const TAG = 'spz-custom-utils'; const DEFAULT_DELAY_TIME = 100; class SpzCustomUtils extends SPZ.BaseElement { constructor(element) { super(element); this.templates_ = SPZServices.templatesForDoc(); } buildCallback() { this.action_ = SPZServices.actionServiceForDoc(this.element); this.templates_ = SPZServices.templatesForDoc(this.element); this.xhr_ = SPZServices.xhrFor(this.win); } static deferredMount() { return false; } mountCallback() { } //判断是否为移动端 isMobile() { /* 判断机型与处理 */ const u = navigator.userAgent const isAndroid = u.indexOf('Android') > -1 || u.indexOf('Adr') > -1; // android终端 const isiOS = !!u.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/); // ios终端 return (isAndroid || isiOS); }; /** * url query param to object * @param {string} url * @returns {object} query object */ params(url) { url = url || window.location.href; let params = {}; url.replace(/[?&]+([^=&]+)=([^&]*)/gi, function (str, key, value) { try { params[key] = decodeURIComponent(value); } catch (e) { params[key] = value; } }); return params; }; /** * @param fn {Function} 实际要执行的函数 * @param delay {Number} 延迟时间,单位是毫秒(ms) * @return {Function} 返回一个“防反跳”了的函数 */ debounce(fn, delay) { // 定时器,用来 setTimeout let timer; // 返回一个函数,这个函数会在一个时间区间结束后的 delay 毫秒时执行 fn 函数 return function () { // 保存函数调用时的上下文和参数,传递给 fn const context = this; const args = arguments; // 每次这个返回的函数被调用,就清除定时器,以保证不执行 fn clearTimeout(timer); // 当返回的函数被最后一次调用后(也就是用户停止了某个连续的操作), // 再过 delay 毫秒就执行 fn timer = setTimeout(function () { fn.apply(context, args); }, delay); }; }; /* 节流防抖 */ throttle(func, wait, mustRun) { var timeout, startTime = new Date(); return function () { var context = this, args = arguments, curTime = new Date(); clearTimeout(timeout); // 如果达到了规定的触发时间间隔,触发 handler if (mustRun && curTime - startTime >= mustRun) { func.apply(context, args); startTime = curTime; // 没达到触发间隔,重新设定定时器 } else { timeout = setTimeout(func, wait); } }; }; //滚动加载方法 isToPageEnd(id) { const $el = document.querySelector(`[data-section-id='${id}']`); const scrollTop = (document.documentElement && document.documentElement.scrollTop) || document.body.scrollTop; //滚动条距离顶部的高度 const clientHeight = window.innerHeight; //当前可视的页面高度 const scrollHeight = document.body.scrollHeight; //当前页面的总高度 const elOffsetTop = $el.getBoundingClientRect().top + window.pageYOffset - document.documentElement.clientTop; // 元素距离文档顶部距离 // 如果改卡片下面还有卡片或者dom,计算滚动加载需要考虑这个高度 const toBottom = scrollHeight - ($el.offsetHeight + elOffsetTop); //元素到浏览器底部的高度 if (scrollTop + clientHeight + toBottom + 100 >= scrollHeight) { return true; } return false; }; /** * url 添加前缀 * @param {string} path , 必须是前面有斜杠前缀的路径 * @returns string */ prefixionPath(prefix,urlPath) { if(typeof prefix !== 'string') return ; if(typeof urlPath !== 'string') return ; if(urlPath.indexOf('/') !== 0){ throw new Error('prefixPath: urlPath must be start with /'); } if(prefix.indexOf('/') !== 0){ throw new Error('prefixPath: prefix must be start with /'); } return prefix+urlPath; } /** * @param {string} urlPath * @returns {string} * @example globalizePath('/path_a/path_b')// => '/en/path_a/path_b' */ globalizePath(urlPath) { if(typeof urlPath !== 'string') return ; if(urlPath.indexOf('/') !== 0){ urlPath = '/'+urlPath; } let prefix = ((SHOPLAZZA && SHOPLAZZA.routes && SHOPLAZZA.routes.root) || ''); if(prefix.length>0){ if(prefix.indexOf('/') !== 0){ prefix = '/'+prefix; } return this.prefixionPath(prefix,urlPath); }else{ return urlPath; } } image_padding_bottom(width, height, origin) { origin = origin || 'limit'; if (width && height) { const hw_ratio = height / width; if (origin == 'limit') { if (hw_ratio < 0.62) { return '62%'; } else if (hw_ratio > 1.6) { return '160%'; } } return parseInt(hw_ratio * 100) + '%'; } return '100%'; } getNumber(str) { str = str + ''; return str.match(/\d+(\.\d+)?/g) ? Number(str.match(/\d+(\.\d+)?/g)[0]) : str; }; // 处理货币符号 finance_money_with_shop_symbol(price, onlyNumber) { const symbol = onlyNumber ? '' : window.SHOPLAZZA.currency_symbol; const position = window.SHOPLAZZA ? window.SHOPLAZZA.currency_symbol_pos : 'left'; const format = window.SHOPLAZZA ? window.SHOPLAZZA.money_format : 'amount'; if (position == 'right') { return Number(Number(price) * 1).format(format) + symbol; } return symbol + Number(Number(price) * 1).format(format); }; triggerEvent_(name, data) { const event = SPZUtils.Event.create(this.win, `${TAG}.${name}`, data || {}); this.action_.trigger(this.element, name, event); } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomUtils) const TAG = 'spz-custom-lang-script'; class SpzCustomlangScript extends SPZ.BaseElement { constructor(element) { super(element); this.currentLangMap = null; } buildCallback() { this.getLang(); } getLang() { const i18nJSON = { 'en-US': { "start_in_text": "Start in", "end_in_text": "End in", "add_to_cart_successfully": "Added successfully", "view_cart": "View Cart", "add": "Add", "product": "Products", "modal_discount_tip": "Add {count} items for discount", "sold_out": "Sold Out", }, 'zh-CN': { "start_in_text": "距开始", "end_in_text": "距结束", "add_to_cart_successfully": "添加成功", "view_cart": "查看购物车", "add": "添加", "product": "个商品", "modal_discount_tip": "添加{count}件商品享受优惠", "sold_out": "已售罄", }, } const lang = sessionStorage._language || document.documentElement.lang || "en-US"; const currentLangMap = i18nJSON[lang] ? i18nJSON[lang] : i18nJSON["en-US"]; this.currentLangMap = currentLangMap; return currentLangMap; } getLangValue({ langKey }) { let currentLangMap = this.currentLangMap || this.getLang(); let langValue = currentLangMap[langKey]; return langValue; } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomlangScript) const TAG = 'spz-custom-lang'; class SpzCustomLang extends SPZ.BaseElement { constructor(element) { super(element); this.langKey = ""; this.count = ""; this.templates_ = SPZServices.templatesForDoc(); } static deferredMount() { return false; } buildCallback() { this.langKey = this.element.getAttribute('langKey'); this.count = this.element.getAttribute('count'); } mountCallback() { const render = async () => { const tempElement = document.getElementById('spz-custom-lang-script'); SPZ.whenApiDefined(tempElement).then(async (api) => { let lang_value = await api.getLangValue({langKey: this.langKey }); if (this.count) { lang_value = lang_value.replace("{count}", this.count); } var spanDom = document.createElement("div"); spanDom.innerHTML = lang_value; this.element.innerHTML = ""; this.element.appendChild(spanDom); }); }; this.mutateElement(render); } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomLang) const TAG = 'spz-custom-sort'; class SpzCustomSort extends SPZ.BaseElement { constructor(element) { super(element); this.spz_custom_id = ''; } static deferredMount() { return false; } buildCallback() { this.action_ = SPZServices.actionServiceForDoc(this.element); this.templates_ = SPZServices.templatesForDoc(this.element); this.xhr_ = SPZServices.xhrFor(this.win); this.setupAction_(); } init() { this.bindEvent(); } bindEvent() { const $selectList = SPZCore.Dom.scopedQuerySelectorAll( this.element, ".sort_custom_content li" ); const $customerSelect = SPZCore.Dom.scopedQuerySelector( this.element, ".sort_custom_select" ); // 选择下拉选项 Array.from($selectList).forEach((node) => { SPZUtils.Event.listen(node, 'click', ()=> { let value = node.getAttribute('value'); let text = node.getAttribute('text'); // 触发selectChange 事件 this.triggerEvent_('selectChange', { value: value, name: value }) $customerSelect.innerHTML = text; const panelChilds = this.element.querySelectorAll(".sort_custom_panel li"); // 清空其他选项的勾选状态 Array.from(panelChilds).forEach((el) => { if(el.getAttribute('value') == value) { el.classList.add("active") } else { el.classList.remove('active'); } }) }); }) } // 渲染界面 async doRender_(data) { // 操作该组件的dom id this.spz_custom_id = data.id; return this.templates_ .findAndRenderTemplate(this.element, data) .then((el) => { const children = this.element.querySelector('*:not(template)'); children && SPZCore.Dom.removeElement(children); this.element.appendChild(el); }).then(() => { this.init(); }); } setupAction_() { this.registerAction('render', async(invocation) => { const data = invocation.args.data; this.doRender_(data) }); this.registerAction('handleSelect', async(invocation) => { const data = invocation.args.data; }); this.registerAction('handleDropdownOpen', async(invocation) => { const $selectDropDown = SPZCore.Dom.scopedQuerySelector( this.element, ".select_drop_down" ); $selectDropDown.classList.add('select_drop_down_rotate'); }); this.registerAction('handleDropdownClose', async(invocation) => { const $selectDropDown = SPZCore.Dom.scopedQuerySelector( this.element, ".select_drop_down" ); $selectDropDown.classList.remove('select_drop_down_rotate'); }); } triggerEvent_(name, data) { const event = SPZUtils.Event.create(this.win, `${ TAG }.${ name }`, data || {}); this.action_.trigger(this.element, name, event); } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomSort) const TAG = "spz-custom-render-products"; class SpzCustomProducts extends SPZ.BaseElement { constructor(element) { super(element); } buildCallback() { this.action_ = SPZServices.actionServiceForDoc(this.element); this.templates_ = SPZServices.templatesForDoc(this.element); this.xhr_ = SPZServices.xhrFor(this.win); this.setupAction_(); } doRender_(data) { return this.templates_ .findAndRenderTemplate(this.element, data) .then((el) => { const children = this.element.querySelector('*:not(template)'); children && SPZCore.Dom.removeElement(children); this.element.appendChild(el); }); } getRenderTemplate(data) { const renderData = data || {}; return this.templates_ .findAndRenderTemplate(this.element, renderData) .then((el) => { const children = this.element.querySelector('*:not(template)'); children && SPZCore.Dom.removeElement(children); return el; }); } setupAction_() { this.registerAction('test', (invocation) => { }); } triggerEvent_(name, data) { const event = SPZUtils.Event.create(this.win, `${ TAG }.${ name }`, data || {}); this.action_.trigger(this.element, name, event); } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomProducts) const TAG = 'spz-custom-discount-default'; const E_DISCOUNT_PROGRESS = { ProgressFinished : "PROGRESS_FINISHED", ProgressNotStarted : "PROGRESS_NOT_STARTED", ProgressOngoing : "PROGRESS_ONGOING" }; class SpzCustomDiscountDefault extends SPZ.BaseElement { constructor(element) { super(element); this.templates_ = null; let discountDefaultData = {"discount_info":{"id":"440767132083436969","center_id":"1199785","discount_name":"BOGO 50% OFF","display_name":"BOGO 50% OFF","discount_target":"DTT_PRODUCT","discount_type":"DT_BUY_ONE_GET_ONE","discount_method":"DM_AUTOMATIC","discount_code":"","starts_at":1730451599,"ends_at":1732784399,"progress":"PROGRESS_FINISHED","discount_layer":{"condition_type":"CT_PURCHASE_QUANTITY","obtain_type":"OT_PERCENT","layers":[{"condition_value":"1","obtain_count":1,"obtain_value":"50","condition_max_value":"-1"}],"layer_type":"LT_SIGNLE"},"starts_remaining_seconds":0,"ends_remaining_seconds":0,"enable_min_purchase_qty":false,"min_purchase_qty_type":""},"landing_page_info":{"customer":{"is_entitled_customer":false},"banner":{"text":"","config":"{\"color\":{\"banner_bg_start\":\"rgba(255, 136, 26, 1)\",\"banner_bg_end\":\"rgba(241, 48, 83, 1)\",\"banner_text\":\"rgba(255, 255, 255, 1)\",\"countdown_text\":\"rgba(34, 34, 34, 1)\",\"countdown_bg\":\"rgba(255, 246, 219, 1)\"},\"countdown\":{\"start_opened\":true,\"start_format\":\"DD:HH:mm:ss:SSS\",\"end_opened\":true,\"end_format\":\"DD:HH:mm:ss:SSS\"}}"},"poster":{"config":"{\"desktop\":\"\",\"mobile\":\"\",\"image_render\":\"fill\"}"},"additional":{"text":"","config":"{\"text_opened\":false,\"color\":{\"background\":\"rgba(230, 230, 230, 1)\",\"text\":\"rgba(34, 34, 34, 1)\"}}"},"product":{"buy_product":{"text":""},"obtain_product":{"text":""}},"sort":{"by":"title","direction":"asc"},"price_text_config":{"text":"","config":""},"button":{"text":"","config":"","redirect_page":""}},"buy_product_info":{"page":1,"has_more":false,"product":[{"id":"a729ac42-c525-4367-85b9-993a7367e362","title":" Performix\u00ae SST AbCuts CLA","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"11","inventory_tracking":true,"published":true,"handle":"performix\u00ae-sst-abcuts-cla","spu":"46994","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"19.99","price_max":"19.99","price":"19.99","compare_at_price":"19.99","compare_at_price_min":"19.99","compare_at_price_max":"19.99","url":"\/products\/performix\u00ae-sst-abcuts-cla","sales":"6","image":{"src":"\/\/img.staticdj.com\/6d475b221215831ce6418470e294c16f.png","alt":"","path":"6d475b221215831ce6418470e294c16f.png","width":2000,"height":2000},"variants":[{"id":"74d64a6c-e5fc-4bf7-a212-717e16fbceea","title":"","weight_unit":"oz","inventory_quantity":"11","sku":"46994","barcode":"811274025787","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"5.30","compare_at_price":"19.99","price":"19.99","available":true,"url":"\/products\/performix\u00ae-sst-abcuts-cla?variant=74d64a6c-e5fc-4bf7-a212-717e16fbceea","available_quantity":"11","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/6d475b221215831ce6418470e294c16f.png","alt":"","path":"6d475b221215831ce6418470e294c16f.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/13b54f0393c78254fc21917e778a73b0.png","alt":"","path":"13b54f0393c78254fc21917e778a73b0.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/46b33daf1a8efb8761ad9656eb88da99.png","alt":"","path":"46b33daf1a8efb8761ad9656eb88da99.png","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"8db25ba3-6fd1-4506-8271-24fcbc78f247","title":"AbCuts CLA with Collagen","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"performix-abcuts-cla-with-collagen","spu":"99592","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"29.99","price_max":"29.99","price":"29.99","compare_at_price":"29.99","compare_at_price_min":"29.99","compare_at_price_max":"29.99","url":"\/products\/performix-abcuts-cla-with-collagen","sales":"13","image":{"src":"\/\/img.staticdj.com\/899315fd230b7d652e0633539faefb6b.jpeg","alt":"","path":"899315fd230b7d652e0633539faefb6b.jpeg","width":2000,"height":2000},"variants":[{"id":"a525efc8-903d-40ba-9834-98b5b49c33c0","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"99592","barcode":"811274023387","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/performix-abcuts-cla-with-collagen?variant=a525efc8-903d-40ba-9834-98b5b49c33c0","available_quantity":"0","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/899315fd230b7d652e0633539faefb6b.jpeg","alt":"","path":"899315fd230b7d652e0633539faefb6b.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/db272d36fb20affed234dae3469faba9.jpeg","alt":"","path":"db272d36fb20affed234dae3469faba9.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/37e0df10f507528d28814b1d14ac1cd5.jpeg","alt":"","path":"37e0df10f507528d28814b1d14ac1cd5.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"f380f142-8251-47a1-80b0-3648f777c918","title":"Creatine Powder Unflavored 30 Servings","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"92","inventory_tracking":true,"published":true,"handle":"creatine-powder-unflavored-30-servings","spu":"99800","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"29.99","price_max":"29.99","price":"29.99","compare_at_price":"29.99","compare_at_price_min":"29.99","compare_at_price_max":"29.99","url":"\/products\/creatine-powder-unflavored-30-servings","sales":"13","image":{"src":"\/\/img.staticdj.com\/e3ac7903ed03b21509e249a767953a84.jpeg","alt":"","path":"e3ac7903ed03b21509e249a767953a84.jpeg","width":2000,"height":2000},"variants":[{"id":"9323e3f4-f0e4-4cbb-a564-2e2b57d402da","title":"","weight_unit":"lb","inventory_quantity":"92","sku":"99800","barcode":"811274025817","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":true,"url":"\/products\/creatine-powder-unflavored-30-servings?variant=9323e3f4-f0e4-4cbb-a564-2e2b57d402da","available_quantity":"92","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/e3ac7903ed03b21509e249a767953a84.jpeg","alt":"","path":"e3ac7903ed03b21509e249a767953a84.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/79181b1afa4a3a67eea7dd6b291225e6.jpeg","alt":"","path":"79181b1afa4a3a67eea7dd6b291225e6.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"e688d8c0-b882-4073-880f-908a4fd2b02c","title":"ION Pre-Workout - Blue Ice","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"47","inventory_tracking":true,"published":true,"handle":"ion-pre-workout-blue-ice","spu":"44225","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"34.99","price_max":"34.99","price":"34.99","compare_at_price":"34.99","compare_at_price_min":"34.99","compare_at_price_max":"34.99","url":"\/products\/ion-pre-workout-blue-ice","sales":"41","image":{"src":"\/\/img.staticdj.com\/4d8053d614b4473f11cd7319368beff0.jpeg","alt":"","path":"4d8053d614b4473f11cd7319368beff0.jpeg","width":2000,"height":2000},"variants":[{"id":"b090d4ff-fbe3-4eb3-87b0-3a5d2f4ea125","title":"","weight_unit":"lb","inventory_quantity":"47","sku":"44225","barcode":"811274026104","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"34.99","price":"34.99","available":true,"url":"\/products\/ion-pre-workout-blue-ice?variant=b090d4ff-fbe3-4eb3-87b0-3a5d2f4ea125","available_quantity":"47","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/4d8053d614b4473f11cd7319368beff0.jpeg","alt":"","path":"4d8053d614b4473f11cd7319368beff0.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/41aff8f89872fc3b2e17384e020831f7.jpeg","alt":"","path":"41aff8f89872fc3b2e17384e020831f7.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/18155d7a3ae80aac1a3309b742439a79.jpeg","alt":"","path":"18155d7a3ae80aac1a3309b742439a79.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"e4ea026b-9bad-47c0-81de-1cf6af90f7cb","title":"ISO EAA & Recovery Powder Watermelon","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"13","inventory_tracking":true,"published":true,"handle":"iso-eaa-recovery-powder-watermelon","spu":"99743","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"39.99","price_max":"39.99","price":"39.99","compare_at_price":"39.99","compare_at_price_min":"39.99","compare_at_price_max":"39.99","url":"\/products\/iso-eaa-recovery-powder-watermelon","sales":"0","image":{"src":"\/\/img.staticdj.com\/b26177d4df6d6b6920738d52744e6f70.jpeg","alt":"","path":"b26177d4df6d6b6920738d52744e6f70.jpeg","width":2000,"height":2000},"variants":[{"id":"74cc5c21-5858-4d05-ba8e-25efb5654d49","title":"","weight_unit":"lb","inventory_quantity":"13","sku":"99743","barcode":"811274026111","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"39.99","price":"39.99","available":true,"url":"\/products\/iso-eaa-recovery-powder-watermelon?variant=74cc5c21-5858-4d05-ba8e-25efb5654d49","available_quantity":"13","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/b26177d4df6d6b6920738d52744e6f70.jpeg","alt":"","path":"b26177d4df6d6b6920738d52744e6f70.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/fbc196b5e11f236fa63f39c078e01be6.jpeg","alt":"","path":"fbc196b5e11f236fa63f39c078e01be6.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"69810be4-9781-4e07-99f3-15d84d177d2b","title":"Performix SST Stim Free","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"18","inventory_tracking":true,"published":true,"handle":"performix-sst-stim-free","spu":"99561","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"59.99","price_max":"59.99","price":"59.99","compare_at_price":"59.99","compare_at_price_min":"59.99","compare_at_price_max":"59.99","url":"\/products\/performix-sst-stim-free","sales":"13","image":{"src":"\/\/img.staticdj.com\/a9a2a793e41e2e04f14676434fab7d09.jpeg","alt":"","path":"a9a2a793e41e2e04f14676434fab7d09.jpeg","width":2000,"height":2000},"variants":[{"id":"d4fb3fb8-9c90-49f3-a698-ce435197b293","title":"","weight_unit":"lb","inventory_quantity":"18","sku":"99561","barcode":"811274023295","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"59.99","price":"59.99","available":true,"url":"\/products\/performix-sst-stim-free?variant=d4fb3fb8-9c90-49f3-a698-ce435197b293","available_quantity":"18","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/a9a2a793e41e2e04f14676434fab7d09.jpeg","alt":"","path":"a9a2a793e41e2e04f14676434fab7d09.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/6e891307a857489bf066eee179b91c12.png","alt":"","path":"6e891307a857489bf066eee179b91c12.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/28f10df9e0c61c56c7f75c748068e20f.png","alt":"","path":"28f10df9e0c61c56c7f75c748068e20f.png","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"c20f1d3a-85dc-4354-b4cf-7a647c9120c1","title":"Performix Supercharged Preworkout 60ct","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"56","inventory_tracking":true,"published":true,"handle":"performix-supercharged-preworkout-60ct","spu":"99891","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"49.99","price_max":"49.99","price":"49.99","compare_at_price":"49.99","compare_at_price_min":"49.99","compare_at_price_max":"49.99","url":"\/products\/performix-supercharged-preworkout-60ct","sales":"3","image":{"src":"\/\/img.staticdj.com\/8fa59cde5637afa78884db3c9185ce60.png","alt":"","path":"8fa59cde5637afa78884db3c9185ce60.png","width":2000,"height":2000},"variants":[{"id":"5be86678-6fa3-4324-9f61-12a220778b78","title":"","weight_unit":"lb","inventory_quantity":"56","sku":"99891","barcode":"811274027033","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"49.99","price":"49.99","available":true,"url":"\/products\/performix-supercharged-preworkout-60ct?variant=5be86678-6fa3-4324-9f61-12a220778b78","available_quantity":"56","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/8fa59cde5637afa78884db3c9185ce60.png","alt":"","path":"8fa59cde5637afa78884db3c9185ce60.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/1ff8b0594881e8a3d7eb49364f4f0c1a.png","alt":"","path":"1ff8b0594881e8a3d7eb49364f4f0c1a.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/f476d84c66fefdb42a2649c4483d6a35.png","alt":"","path":"f476d84c66fefdb42a2649c4483d6a35.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/077011a9a8ec2f62bcaaf3274fe960c2.png","alt":"","path":"077011a9a8ec2f62bcaaf3274fe960c2.png","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"9518324b-af97-459a-9183-0990f24e5933","title":"Performix Supercharged Pump 80ct","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"36","inventory_tracking":true,"published":true,"handle":"performix-supercharged-pump-80ct","spu":"99890","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"49.99","price_max":"49.99","price":"49.99","compare_at_price":"49.99","compare_at_price_min":"49.99","compare_at_price_max":"49.99","url":"\/products\/performix-supercharged-pump-80ct","sales":"1","image":{"src":"\/\/img.staticdj.com\/0771f5d4707d61a45509b64005b8770d.png","alt":"","path":"0771f5d4707d61a45509b64005b8770d.png","width":2000,"height":2000},"variants":[{"id":"df342ff2-ded8-4231-9cfb-fbb1a3c31615","title":"","weight_unit":"lb","inventory_quantity":"36","sku":"99890","barcode":"811274027088","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"49.99","price":"49.99","available":true,"url":"\/products\/performix-supercharged-pump-80ct?variant=df342ff2-ded8-4231-9cfb-fbb1a3c31615","available_quantity":"36","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/0771f5d4707d61a45509b64005b8770d.png","alt":"","path":"0771f5d4707d61a45509b64005b8770d.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/15d170b4b1b8847ccc8fb3fba4677f21.png","alt":"","path":"15d170b4b1b8847ccc8fb3fba4677f21.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/240580663576276322cb64416b20d4fd.png","alt":"","path":"240580663576276322cb64416b20d4fd.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4c5fc2d3817f1a3fd800c44cbd2ca889.png","alt":"","path":"4c5fc2d3817f1a3fd800c44cbd2ca889.png","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"bae4d806-5e51-42c0-b2fb-72dc3685a6d1","title":"Performix Supercharged Recovery 60ct","brief":"Performix Supercharged Recovery 60ct","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"25","inventory_tracking":true,"published":true,"handle":"performix-supercharged-recovery-60ct","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"49.99","price_max":"49.99","price":"49.99","compare_at_price":"49.99","compare_at_price_min":"49.99","compare_at_price_max":"49.99","url":"\/products\/performix-supercharged-recovery-60ct","sales":"0","image":{"src":"\/\/img.staticdj.com\/b13fd1c824aea1daaaa9c91de0fe202d.jpg","alt":"","path":"b13fd1c824aea1daaaa9c91de0fe202d.jpg","width":2000,"height":2000},"variants":[{"id":"11240dd4-e2b9-43e0-8d85-eafb1a08de5d","title":"","weight_unit":"kg","inventory_quantity":"25","sku":"99889","barcode":"811274027026","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"49.99","price":"49.99","available":true,"url":"\/products\/performix-supercharged-recovery-60ct?variant=11240dd4-e2b9-43e0-8d85-eafb1a08de5d","available_quantity":"25","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/b13fd1c824aea1daaaa9c91de0fe202d.jpg","alt":"","path":"b13fd1c824aea1daaaa9c91de0fe202d.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/435e7779f5c49669bbcf7cfae99be537.jpg","alt":"","path":"435e7779f5c49669bbcf7cfae99be537.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/78ac49a048cbe36c6d20057d01786d7c.jpg","alt":"","path":"78ac49a048cbe36c6d20057d01786d7c.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4b70646399368f41a0c15d06ea13b0f5.jpg","alt":"","path":"4b70646399368f41a0c15d06ea13b0f5.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"d785df95-76bd-4b9c-ad75-023ea92da3f3","title":"Performix Whey Protein 1.98lb","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"1","inventory_tracking":true,"published":true,"handle":"performix-whey-protein-1-98lb","spu":"99744_master","note":"","need_variant_image":true,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"39.99","price_max":"39.99","price":"39.99","compare_at_price":"39.99","compare_at_price_min":"39.99","compare_at_price_max":"39.99","url":"\/products\/performix-whey-protein-1-98lb","sales":"50","image":{"src":"\/\/img.staticdj.com\/4300175d3b2b083d841517babc50be1d.jpeg","alt":"","path":"4300175d3b2b083d841517babc50be1d.jpeg","width":2000,"height":2000},"variants":[{"id":"4506bf8c-aa4d-4edc-a4cb-77a674f1af46","title":"Chocolate","weight_unit":"lb","inventory_quantity":"0","sku":"99744","barcode":"811274026982","position":1,"option1":"Chocolate","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/4300175d3b2b083d841517babc50be1d.jpeg","alt":"","path":"4300175d3b2b083d841517babc50be1d.jpeg","width":2000,"height":2000},"weight":"2.00","compare_at_price":"39.99","price":"39.99","available":false,"url":"\/products\/performix-whey-protein-1-98lb?variant=4506bf8c-aa4d-4edc-a4cb-77a674f1af46","available_quantity":"0","options":[{"name":"Flavor","value":"Chocolate"}],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/4300175d3b2b083d841517babc50be1d.jpeg","alt":"","path":"4300175d3b2b083d841517babc50be1d.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/d70ef7453db56c1332f5f887fd63c046.jpeg","alt":"","path":"d70ef7453db56c1332f5f887fd63c046.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4a25b38bbd918843e9be458eb7d94fe9.jpeg","alt":"","path":"4a25b38bbd918843e9be458eb7d94fe9.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/8dd8f7ea4009fb1b035f1f50940b91f0.jpeg","alt":"","path":"8dd8f7ea4009fb1b035f1f50940b91f0.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a03ffac3aa4e9ff7f5ffc9c22dc2ed6b.jpeg","alt":"","path":"a03ffac3aa4e9ff7f5ffc9c22dc2ed6b.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/230d9f5852a17929e826fd6e87bc1e25.jpeg","alt":"","path":"230d9f5852a17929e826fd6e87bc1e25.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4300175d3b2b083d841517babc50be1d.jpeg","alt":"","path":"4300175d3b2b083d841517babc50be1d.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/8dd8f7ea4009fb1b035f1f50940b91f0.jpeg","alt":"","path":"8dd8f7ea4009fb1b035f1f50940b91f0.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"91f32c0b-c9cc-4af7-afa6-5b6eb764a5a6","title":"Performix\u2122 SST Cuts","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"10","inventory_tracking":true,"published":true,"handle":"performix\u2122-sst-cuts","spu":"43381","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"59.99","price_max":"59.99","price":"59.99","compare_at_price":"59.99","compare_at_price_min":"59.99","compare_at_price_max":"59.99","url":"\/products\/performix\u2122-sst-cuts","sales":"19","image":{"src":"\/\/img.staticdj.com\/f7c99d76fa9c78751269fa9f2ba43295.png","alt":"","path":"f7c99d76fa9c78751269fa9f2ba43295.png","width":2000,"height":2000},"variants":[{"id":"8980a164-a13b-44ea-904f-6e0a9f0bb140","title":"","weight_unit":"lb","inventory_quantity":"10","sku":"43381","barcode":"811274024537","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"59.99","price":"59.99","available":true,"url":"\/products\/performix\u2122-sst-cuts?variant=8980a164-a13b-44ea-904f-6e0a9f0bb140","available_quantity":"10","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/f7c99d76fa9c78751269fa9f2ba43295.png","alt":"","path":"f7c99d76fa9c78751269fa9f2ba43295.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/6153dd7285aaa0287d1c3dd8c9d4d0a8.png","alt":"","path":"6153dd7285aaa0287d1c3dd8c9d4d0a8.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/67ee12ceb906d3cea392842c6c1296e1.png","alt":"","path":"67ee12ceb906d3cea392842c6c1296e1.png","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"461fd9d5-9d22-4717-90ca-d66ccbc01434","title":"SST Multi+ Multivitamin","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"25","inventory_tracking":true,"published":true,"handle":"performix-sst-multivitamin","spu":"99742","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"49.99","price_max":"49.99","price":"49.99","compare_at_price":"49.99","compare_at_price_min":"49.99","compare_at_price_max":"49.99","url":"\/products\/performix-sst-multivitamin","sales":"7","image":{"src":"\/\/img.staticdj.com\/32cc63da81e4fd280f105cf25f8136ec.jpeg","alt":"","path":"32cc63da81e4fd280f105cf25f8136ec.jpeg","width":2000,"height":2000},"variants":[{"id":"9a2deb84-bf97-439d-b811-1b92479dd46d","title":"","weight_unit":"lb","inventory_quantity":"25","sku":"99742","barcode":"811274026456","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"49.99","price":"49.99","available":true,"url":"\/products\/performix-sst-multivitamin?variant=9a2deb84-bf97-439d-b811-1b92479dd46d","available_quantity":"25","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/32cc63da81e4fd280f105cf25f8136ec.jpeg","alt":"","path":"32cc63da81e4fd280f105cf25f8136ec.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/0c65400576916ac3358d2bd5bfb6d0d6.jpeg","alt":"","path":"0c65400576916ac3358d2bd5bfb6d0d6.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"1355fa41-3d2d-423d-a194-1e004a2b2cdd","title":"SST Original","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"58","inventory_tracking":true,"published":true,"handle":"sst-original","spu":"42804","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"54.99","price_max":"54.99","price":"54.99","compare_at_price":"54.99","compare_at_price_min":"54.99","compare_at_price_max":"54.99","url":"\/products\/sst-original","sales":"4","image":{"src":"\/\/img.staticdj.com\/e861ce0d7025645da8843a5911454e9e.png","alt":"","path":"e861ce0d7025645da8843a5911454e9e.png","width":2000,"height":2000},"variants":[{"id":"18927046-50e7-40ef-9f56-27d5250832e0","title":"","weight_unit":"kg","inventory_quantity":"58","sku":"42804","barcode":"811274023530","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"54.99","price":"54.99","available":true,"url":"\/products\/sst-original?variant=18927046-50e7-40ef-9f56-27d5250832e0","available_quantity":"58","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/e861ce0d7025645da8843a5911454e9e.png","alt":"","path":"e861ce0d7025645da8843a5911454e9e.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/5d52adf12c57479223627082a3ab4b71.png","alt":"","path":"5d52adf12c57479223627082a3ab4b71.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/7ccd32c33b6125fa241260d9fdbca66e.png","alt":"","path":"7ccd32c33b6125fa241260d9fdbca66e.png","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/8e12b8ef1d800e0f4e789abd8681249e.jpg","alt":"","path":"8e12b8ef1d800e0f4e789abd8681249e.jpg","width":1500,"height":1500},{"src":"\/\/img.staticdj.com\/265d071772a39db8c608991b04972fba.jpg","alt":"","path":"265d071772a39db8c608991b04972fba.jpg","width":1080,"height":1080}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"f0aa832e-587c-4ab2-b1d5-b2bec97806b5","title":"SST Support Burn","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"108","inventory_tracking":true,"published":true,"handle":"performix-sst-support-burn","spu":"43382","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"29.99","price_max":"29.99","price":"29.99","compare_at_price":"29.99","compare_at_price_min":"29.99","compare_at_price_max":"29.99","url":"\/products\/performix-sst-support-burn","sales":"21","image":{"src":"\/\/img.staticdj.com\/411a8a4ef5cc28e8dfd16a8d8c82baff.jpeg","alt":"","path":"411a8a4ef5cc28e8dfd16a8d8c82baff.jpeg","width":2000,"height":2000},"variants":[{"id":"1b533e8f-9178-40d5-add0-df8b32e95a6a","title":"","weight_unit":"lb","inventory_quantity":"108","sku":"43382","barcode":"811274024254","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":true,"url":"\/products\/performix-sst-support-burn?variant=1b533e8f-9178-40d5-add0-df8b32e95a6a","available_quantity":"108","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/411a8a4ef5cc28e8dfd16a8d8c82baff.jpeg","alt":"","path":"411a8a4ef5cc28e8dfd16a8d8c82baff.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/dfeff191746b48fc316299f6fcdc05e2.jpeg","alt":"","path":"dfeff191746b48fc316299f6fcdc05e2.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/129a2eec4247d9c333dc812926b6b6de.jpeg","alt":"","path":"129a2eec4247d9c333dc812926b6b6de.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"da3cf462-7717-457e-82f1-110c50acd6be","title":"SST Support Lean Sleep","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"performix-sst-lean-sleep","spu":"46997","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"24.99","price_max":"24.99","price":"24.99","compare_at_price":"24.99","compare_at_price_min":"24.99","compare_at_price_max":"24.99","url":"\/products\/performix-sst-lean-sleep","sales":"32","image":{"src":"\/\/img.staticdj.com\/5667e75fcaa8a8267635961f1bd72903.jpeg","alt":"","path":"5667e75fcaa8a8267635961f1bd72903.jpeg","width":2000,"height":2000},"variants":[{"id":"780844b8-acfc-4f54-a461-7e6efc27370c","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"46997","barcode":"811274026043","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"24.99","price":"24.99","available":false,"url":"\/products\/performix-sst-lean-sleep?variant=780844b8-acfc-4f54-a461-7e6efc27370c","available_quantity":"0","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/5667e75fcaa8a8267635961f1bd72903.jpeg","alt":"","path":"5667e75fcaa8a8267635961f1bd72903.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/b04b6f697fad01ce86ef3b6189c1a49c.jpeg","alt":"","path":"b04b6f697fad01ce86ef3b6189c1a49c.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/6ebf64eb5d4917414acc3eb0b4c80d3b.jpeg","alt":"","path":"6ebf64eb5d4917414acc3eb0b4c80d3b.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/b512d9065284c4abde2b872739ba0523.jpeg","alt":"","path":"b512d9065284c4abde2b872739ba0523.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/2b674503631851cda149dce288e95ca3.jpeg","alt":"","path":"2b674503631851cda149dce288e95ca3.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/ab50d56e732d8d1a5e34bf8c32d9d110.jpeg","alt":"","path":"ab50d56e732d8d1a5e34bf8c32d9d110.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"91e33d7d-80ae-4d11-bdc9-b44d4aa3e467","title":"Super Male T","brief":"","vendor":"PERFORMIX","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"27","inventory_tracking":true,"published":true,"handle":"performix-super-male-t","spu":"42806","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"89.99","price_max":"89.99","price":"89.99","compare_at_price":"89.99","compare_at_price_min":"89.99","compare_at_price_max":"89.99","url":"\/products\/performix-super-male-t","sales":"34","image":{"src":"\/\/img.staticdj.com\/8456a81bf4210a8e1b1e0e1bb64908ed.jpeg","alt":"","path":"8456a81bf4210a8e1b1e0e1bb64908ed.jpeg","width":2000,"height":2000},"variants":[{"id":"bee8515c-9fde-43ce-8f4e-d7a3be453f93","title":"","weight_unit":"lb","inventory_quantity":"27","sku":"42806","barcode":"811274024711","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"89.99","price":"89.99","available":true,"url":"\/products\/performix-super-male-t?variant=bee8515c-9fde-43ce-8f4e-d7a3be453f93","available_quantity":"27","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/8456a81bf4210a8e1b1e0e1bb64908ed.jpeg","alt":"","path":"8456a81bf4210a8e1b1e0e1bb64908ed.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/66a951ab606fc6bc82fd46d032219e4f.jpeg","alt":"","path":"66a951ab606fc6bc82fd46d032219e4f.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/708089df25ef508f77fae9cc8d68a8c2.jpeg","alt":"","path":"708089df25ef508f77fae9cc8d68a8c2.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0}],"sort":{"by":"title","direction":"asc"},"has_removed":false,"has_insufficient_inventory":false,"count":16,"total_price":null},"obtain_product_info":{"page":1,"has_more":false,"product":[],"sort":null,"count":0},"discount_template_name":"default"} || {}; discountDefaultData.section_id = 15890337540001 || 1; this.discountDefaultData = discountDefaultData; this.discount_id = this.discountDefaultData.discount_info.id;// 活动id this.discountI18n = {}; const THEME_NAME = window.SHOPLAZZA.theme.merchant_theme_name || window.SHOP_PARAMS.theme_name; this.isHero = /Hero/.test(THEME_NAME); // PROGRESS_ONGOING: 进行中 PROGRESS_NOT_STARTED: 未开始 PROGRESS_FINISHED: 已结束 this.E_DISCOUNT_PROGRESS = { ProgressFinished : "PROGRESS_FINISHED", ProgressNotStarted : "PROGRESS_NOT_STARTED", ProgressOngoing : "PROGRESS_ONGOING" }; this.E_TAB_MAP = { scenario_buy : { value: "1", domId: "product_list_buy_products" }, scenario_obtain : { value: "2", domId: "product_list_obtain_products" } } this.tabContentIdMap = {}; this.currentTab = this.E_TAB_MAP.scenario_buy.value; this.model_buy = { discount_id: this.discount_id, //活动id scenario: 1, // 枚举值,1:购买商品,2:获得商品 sort: { by: 'recommend', direction: 'asc' }, page: 2, //分页码 limit: 20, // 每页数量 loading: false, // 请求数据标示 has_more: this.discountDefaultData.buy_product_info.has_more // 是否还有数据 }; this.model_get = { discount_id: this.discount_id, //活动id scenario: 2, // 枚举值,1:购买商品,2:获得商品 sort: { by: 'recommend', direction: 'asc' }, page: 2, limit: 20, loading: false, has_more: this.discountDefaultData.obtain_product_info?.has_more }; this.modelMap = { [this.E_TAB_MAP.scenario_buy.value]: this.model_buy, [this.E_TAB_MAP.scenario_obtain.value]: this.model_get, } this.sortDict = { recommend_asc: { by: 'recommend', direction: 'asc' }, title_asc: { by: 'title', direction: 'asc' }, title_desc: { by: 'title', direction: 'desc' }, price_asc: { by: 'price', direction: 'asc' }, price_desc: { by: 'price', direction: 'desc' }, created_at_desc: { by: 'created_at', direction: 'desc' }, sales_desc: { by: 'sales', direction: 'desc' }, add_to_cart_count_desc: { by: 'add_to_cart_count', direction: 'desc' }, views_desc: { by: 'views', direction: 'desc' } }; this.sortOptions = [ { value: 'recommend_asc', text: "Recommend" }, { value: 'price_asc', text: "Price, low to high" }, { value: 'price_desc', text: "Price, high to low" }, { value: 'title_asc', text: "Name, A to Z" }, { value: 'title_desc', text: "Name, Z to A" }, { value: 'created_at_desc', text: "Newest in" }, { value: 'sales_desc', text: "Total sales, high to low" }, { value: 'add_to_cart_count_desc', text: "Purchases, high to low" }, { value: 'views_desc', text: "Page views, high to low" } ]; // 直出商品数据 + 异步请求商品数据 this.products = this.discountDefaultData.buy_product_info.product; // 款式信息集合 this.productStyleInfo = []; // 弹窗内选择款式集合 this.modalVariantInfo = []; // 加购商品列表 this.lineItems = []; this.buyNowApi = "\/api\/checkout\/order"; this.batchAtcApi = "\/api\/cart\/batch"; this.debounceTimer = null; this.discount_type = this.discountDefaultData.discount_info.discount_type; this.discount_info = this.discountDefaultData.discount_info; } static deferredMount() { return false; } buildCallback() { this.action_ = SPZServices.actionServiceForDoc(this.element); this.templates_ = SPZServices.templatesForDoc(this.element); this.xhr_ = SPZServices.xhrFor(this.win); this.setupAction_(); Object.entries(this.E_TAB_MAP).forEach(([key, valueObj]) => { this.tabContentIdMap[valueObj.value] = valueObj.domId; }) this.getLocalLang(); console.log(this.discountDefaultData, 'discountDefaultData'); } async mountCallback() { this.utilsApi_ = await SPZ.whenApiDefined(document.querySelector('#spz_custom_utils')); this.init(); this.handleRenderSort(); } init() { this.xhr_.fetchJson(`/api/discount-i18n`, { method: "get", }).then((res)=>{ this.discountI18n = res; this.bindEvent(); }) // url 携带 sort_by参数 var queryParams = this.utilsApi_.params(); var sortValue = queryParams.sort_by; if (sortValue) { this.model_buy.sort = this.sortDict[sortValue]; } // 经典捆绑初始化商品数据 if(this.discount_type == 'DT_CLASSIC_BUNDLE') { this.productStyleInfo.push(...this.discountDefaultData.buy_product_info.product.map((item) => { return this.getFilteredVariants_(item, 'single'); })); } } // 获取本地的多语言 async getLocalLang() { const tempElement = document.getElementById('spz-custom-lang-script'); SPZ.whenApiDefined(tempElement).then(async (api) => { this.currentLangMap = await api.getLang(); }); } handleRenderSort() { // 渲染排序 const sort_x_id = 'promotionSortProductsX'; const $sortX = document.getElementById(sort_x_id) $sortX && SPZ.whenApiDefined($sortX).then((api) => { // 渲染排序列表 api.doRender_({options: this.sortOptions, defaultValue: 'recommend_asc', id: sort_x_id }); }) const sort_y_id = 'promotionSortProductsY'; const $sortY = document.getElementById(sort_y_id) $sortY && SPZ.whenApiDefined($sortY).then((api) => { // 渲染排序列表 api.doRender_({options: this.sortOptions, defaultValue: 'recommend_asc' , id: sort_y_id}); }) } // 获取数据,拼接html模板 async getData() { // 请求数据 let model = this.modelMap[this.currentTab]; if (!model.has_more || model.loading) { return; } model.loading = true; this.handleLoading_({type: 'product', action: 'show'}); let $content = document.querySelector(`#${this.tabContentIdMap[this.currentTab]} .discount-default__productlist-wrap`) || document.querySelector(`.discount-default__productlist-wrap`); let $defaultEmpty = $content && $content.querySelector('.discount_default_empty'); //查询活动商品接口 const reqBody = { discount_id: model.discount_id, page: model.page, limit: model.limit, apply_scenario: model.scenario, sort: model.sort, sales_channel: { sale_channel_type: "online", sale_channel_id: '1199785' } } this.xhr_.fetchJson(`/api/storefront/promotion/landing_page/product/list`, { method: "post", body: reqBody }).then(async(res)=>{ // 更新参与活动的商品数量 const productCount = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionEventProductCount`); productCount && SPZ.whenApiDefined(productCount).then((api) => { api.render(res.count, true); }); this.products.push(...res.products); this.handleLoading_({type: 'product', action: 'close'}); const count = res.count; model.has_more = res.has_more; if (count > 0) { $defaultEmpty && ($defaultEmpty.style.display = 'none'); model.page++; if (res.products && res.products.length > 0) { let products = res.products.map((product) => { return { ...product, url: this.utilsApi_.globalizePath(product.url), image_padding_bottom: this.utilsApi_.image_padding_bottom(product.image.width, product.image.height,'no-limit') } }); // 获取商品列表渲染模板, dom挂载 const renderApi = await SPZ.whenApiDefined(document.querySelector('#discounts_products_render')); const el = await renderApi.getRenderTemplate({ products: products, discountI18n: this.discountI18n, discount_info: this.discountDefaultData.discount_info, }); const childNodes = el.querySelectorAll('.as-render-product-item'); if (childNodes && childNodes.length > 0) { $content.append(...el.childNodes); } if(this.discount_type == 'DT_CLASSIC_BUNDLE') { // 遍历$content 插入商品垂直虚线分割 const productListAsync = $content.querySelectorAll('.as-render-product-item'); if (productListAsync.length > 0) { productListAsync.forEach((item, index) => { const htmlStr = `<span class="promotion_dotted_line"></span> <div class="promotion_plus_bundle"> <svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 12 12" fill="none"> <path d="M1 6H11M6 1L6 11" stroke="black" stroke-width="1.6" stroke-linecap="round"/> </svg> </div> <span class="promotion_dotted_line"></span>`; this.createAndInsertSeparator_('promotion_separator md:hidden', (index + 1) % 4 !== 0 && index !== productListAsync.length - 1, htmlStr, $content, item); this.createAndInsertSeparator_('promotion_separator lg:hidden', (index + 1) % 2 !== 0 && index !== productListAsync.length - 1, htmlStr, $content, item); }); } } if(this.discount_type == 'DT_MIX_MATCH_BUNDLE') { const productSelector = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionProductSelector`); productSelector && SPZ.whenApiDefined(productSelector).then((api) => { api.init(); }); const currentPageSelectedProducts = res.products.filter(item => this.productStyleInfo.map(item => item.product_id).includes(item.id)); this.updateProductPrice_(currentPageSelectedProducts); } } } else { // 空列表 const $emptyTemplate = document.querySelector('#promotionDiscountEmpty .discount_default_empty'); const $cloneEmptyTemplate = $emptyTemplate.cloneNode(true); $content.innerHTML = ''; $content.append($cloneEmptyTemplate); $defaultEmpty && ($defaultEmpty.style.display = 'flex'); } model.loading = false; }).catch((err)=>{ this.handleRequestError_(err); }).finally(()=>{ model.loading = false; this.handleLoading_({type: 'product', action: 'close'}); // 经典spu纬度需要该商品信息: is_classic_bundle_product_list_variant_tag if(this.discount_type == 'DT_CLASSIC_BUNDLE' && this.discount_info.enable_min_purchase_qty == true && this.discount_info.min_purchase_qty_type == 'spu') { this.productStyleInfo = this.productStyleInfo.map((item) => { return { ...item, is_classic_bundle_product_list_variant_tag: true, } }); } const result = this.productStyleInfo.reduce((map, item) => { if (!map[item.product_id]) { map[item.product_id] = []; } map[item.product_id].push(item); return map; }, {}); // 渲染变体tags if(this.discount_type == 'DT_MIX_MATCH_BUNDLE' || this.discount_type == 'DT_CLASSIC_BUNDLE') { Object.values(result).forEach((item) => { this.handleSpzVariantRender_(item, item[0].product_id); this.handleProductOption_(item[0].product_id, true); }); } // 渲染经典额外变体 if(this.discount_type == 'DT_CLASSIC_BUNDLE' && this.discount_info.enable_min_purchase_qty == true && this.discount_info.min_purchase_qty_type == 'spu') { Object.values(result).forEach((item) => { if(item[0].is_multi_style && item[0].discount_min_purchase_qty > 1) { const classicSpuTag = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionClassicSpuTags-${item[0].product_id}`); classicSpuTag && SPZ.whenApiDefined(classicSpuTag).then((api) => { api.render(item, true); }); } }); } // 渲染经典捆绑商品最低购买数量 if(this.discount_type == 'DT_CLASSIC_BUNDLE') { Object.values(result).forEach((item) => { this.handleMinPurchaseQtyUpdate_({discount_min_purchase_qty: item[0].discount_min_purchase_qty}, item[0].product_id); }); } }) } createAndInsertSeparator_(className, condition, htmlStr, $content, item) { if (condition) { const separator = document.createElement('div'); separator.className = className; separator.innerHTML = htmlStr; $content.insertBefore(separator, item.nextSibling); } } bindEvent() { // 监听滚动,请求数据 window.addEventListener("scroll", this.utilsApi_.debounce( () => { // 判断是否到底 const model = this.modelMap[this.currentTab]; if (!model.loading && model.has_more && this.utilsApi_.isToPageEnd(this.discountDefaultData.section_id)) { this.getData(); } }, 10, 50 )) } // 商品排序 handleSort_(data) { let sortKey = data.value; this.modelMap[this.currentTab].sort = this.sortDict[sortKey || 'recommend_asc']; this.modelMap[this.currentTab].page = 1; this.modelMap[this.currentTab].has_more = true; this.productStyleInfo = this.handleMixMatchBundleFilterSelected_(this.productStyleInfo); // 清空商品列表dom, 重新请求排序数据渲染 let $productList = document.querySelector(`#${this.tabContentIdMap[this.currentTab]} .discount-default__productlist-wrap`) || document.querySelector(`.discount-default__productlist-wrap`);; $productList && ($productList.innerHTML = ''); this.getData(); } // tab 切换 tabChange_(value) { this.currentTab = value || this.E_TAB_MAP.scenario_buy.value; } // 渲染界面 doRender_(data) { return this.templates_ .findAndRenderTemplate(this.element, data) .then((el) => { const children = this.element.querySelector('*:not(template)'); children && SPZCore.Dom.removeElement(children); this.element.appendChild(el); }); } // 捆绑商品加购/立即购买 handleBundleAddToCart_(data) { const { action } = data; if(this.discount_type == 'DT_CLASSIC_BUNDLE') { this.lineItems = this.productStyleInfo; } else { this.lineItems = this.handleMixMatchBundleFilterSelected_(this.productStyleInfo); } if(action == 'cart') { //add to cart this.xhr_ .fetchJson(this.batchAtcApi, { method: 'POST', body: { line_items: this.lineItems.map((item) => { return { product_id: item.product_id, variant_id: item.variant_id, quantity: item.quantity } }) } }) .then(data => { setTimeout(() => { window.location.href = '/cart'; }); }) .catch(async (error) => { await error.then((data) => { this.handleRequestError_(data); }); }); } else { //checkout this.xhr_ .fetchJson(this.buyNowApi, { method: 'POST', body: { line_items: (this.lineItems || []).map((product) => { return { quantity: product.quantity, variant_id: product.variant_id, note: product.note || '', properties: product.properties || {} } }), refer_info: { source: 'buy_now' } } }) .then(async (data) => { if (data.state === 'success') { window.location.href = data.data?.checkout_url; } else { this.handleRequestError_(data); } }) .catch(async (error) => { await error.then((data) => { this.handleRequestError_(data); }); }); } } handleRequestError_(data) { const message = data?.message || data?.errors?.[0] || 'Unknown error'; const toast = SPZCore.Dom.scopedQuerySelector(document.body, '#discount_toast'); toast && SPZ.whenApiDefined(toast).then((api) => { api.showToast(message); }); }; // 渲染加购弹窗内容 async renderQuickShop(data) { this.handleLoading_({type: 'whole', action: 'show'}); const apply_scenario = this.modelMap[this.currentTab].scenario; this.xhr_.fetchJson(`/api/storefront/promotion/landing_page/product?product_id=${data.product_id}&discount_id=${this.discount_id}&apply_scenario=${apply_scenario}`, { method: "get", }).then(async(res)=>{ this.handleLoading_({type: 'whole', action: 'close'}); const $quickShop = await SPZ.whenApiDefined(document.querySelector('#promotion-quick-view-render')); // 定义默认渲染的子款式 const selectedVariant = res.product.variants.find((v)=> (v.available && v.is_hit_discount == true)) || res.product.variants[0]; let selectedValues = {}; selectedVariant.options.length && selectedVariant.options.forEach(item => { selectedValues[item.name] = item.value; }) // 默认选中的 子款式、 options res.product.defaultSelectValues = selectedValues; let data = {...res.product, product:res.product, selectedVariant}; $quickShop.render(data); // 打开加购弹窗 SPZ.whenApiDefined(document.querySelector(`#promotion-quick-view`)).then((api)=>{ api.open(); }); }).catch((err)=>{ this.handleLoading_({type: 'whole', action: 'close'}); }) } // 单变体点击添加按钮 renderSingleVariant(data) { const { product_id } = data; const currentProduct = this.products.find((product) => product.id == product_id); // 若当前商品已存在,则不再添加 而是更新数量 const index = this.productStyleInfo.findIndex((item) => item.product_id == product_id); if (index != -1) { this.productStyleInfo[index].quantity = Number(this.productStyleInfo[index].quantity) + 1; this.updateProductPrice_(this.productStyleInfo); } else { this.productStyleInfo.push(this.getFilteredVariants_(currentProduct, 'single')); } const renderProductArr = this.productStyleInfo.filter((item) => item.product_id == product_id); this.handleSpzVariantRender_(renderProductArr, product_id); this.handleProductOption_(product_id, true); } // 过滤选中商品的子款式 获取有用的信息 product_id,variant_id,price,compare_at_price,quantity,title,variant_title getFilteredVariants_(data, type = '') { const { id, title, variants, inventory_tracking, inventory_policy, inventory_quantity, product_type, discount_min_purchase_qty } = data; const { product_id, variant_id, variant, quantity, product } = data; const isSingle = type == 'single'; const variantData = isSingle ? (variants[0] || data) : variant; const productData = isSingle ? data : product; let item_quantity = 0; if (this.discount_type === 'DT_MIX_MATCH_BUNDLE') { item_quantity = isSingle ? 1 : Number(quantity); } else if (type === 'classic_spu') { item_quantity = 1; } else { item_quantity = discount_min_purchase_qty || productData.discount_min_purchase_qty || variantData.discount_info.discount_min_purchase_qty || 1; } return { product_id: isSingle ? id : product_id, variant_id: variantData?.id || '', price: variantData?.price || '0.00', compare_at_price: variantData?.compare_at_price || '0.00', quantity: item_quantity, inventory_tracking: productData.inventory_tracking, inventory_policy: productData.inventory_policy, inventory_quantity: productData.inventory_quantity, product_type: productData.product_type || this.products.find((item) => item.id == product_id)?.product_type || this.products.find((item) => item.id == id)?.product_type || '', title: productData.title, variant_title: variantData?.options.map((option) => option.value).join('/') || '', is_multi_style: productData.variants.length > 1, discount_min_purchase_qty: discount_min_purchase_qty || productData.discount_min_purchase_qty || variantData.discount_info.discount_min_purchase_qty || 0, } } // 更新价格方法 updateProductPrice_(data) { const bottomBtnContainer = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionBottomContainer`); if (data.length == 0) { bottomBtnContainer && SPZ.whenApiDefined(bottomBtnContainer).then((api) => { api.render({original_price: 0, received_discounts: 0}, true); }); return; } data = this.handleMixMatchBundleFilterSelected_(data); const reqBody = { discount_id: this.discount_id, customer: { customer_id: '', email: '', }, sales_channel: { sale_channel_type: "online", sale_channel_id: '1199785' }, line_items: data } // 如果已经有一个请求在等待,那么取消这个请求 if (this.debounceTimer) { clearTimeout(this.debounceTimer); } this.handleLoading_({type: 'whole', action: 'show'}); this.debounceTimer = setTimeout(() => { this.xhr_.fetchJson(`/api/storefront/promotion/calculate/discounted_price`, { method: "post", body: reqBody }).then((res)=>{ // 更新商品列表价格 Object.keys(res.line_items).forEach((key) => { const currentProductPrice = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionProductPrice-${key}`); currentProductPrice && SPZ.whenApiDefined(currentProductPrice).then((api) => { api.render(res.line_items[key], true); }); }); // 更新底部按钮总价/总折扣价 const picked_qty = data.reduce((acc, item) => { return acc + item.quantity; }, 0); bottomBtnContainer && SPZ.whenApiDefined(bottomBtnContainer).then((api) => { api.render({...res.total_price, picked_qty}, true); }); }).catch(async (err)=>{ await err.then((data) => { this.handleRequestError_(data); }); }).finally(()=>{ this.handleLoading_({type: 'whole', action: 'close'}); }) }, 100); } // 还原商品价格 resetProductPrice_(data) { const {price, compare_at_price, id} = data; const currentProductPrice = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionProductPrice-${id}`); currentProductPrice && SPZ.whenApiDefined(currentProductPrice).then((api) => { api.render({total_received_discounts: price, total_price: compare_at_price}, true); }); } // 处理与selector组件的交互 handleProductOption_(productId, show) { const currentProductOption = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionSelectOption-${productId}`); if(!currentProductOption) return; currentProductOption.toggleAttribute('show', show); const productSelector = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionProductSelector`); productSelector && SPZ.whenApiDefined(productSelector).then((api) => { api.toggle_({option: productId, value: show}); }); } // 调用spz-tag组件的doRender方法 handleSpzVariantRender_(data, id) { const spzVariantTag = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionSpzVariantTags-${id}`); spzVariantTag && SPZ.whenApiDefined(spzVariantTag).then((api) => { api.render(data, true); }); } // 执行经典捆绑最低购买数量更新 handleMinPurchaseQtyUpdate_(data, id) { const minPruchaseQty = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionMinPurchaseQty-${id}`); minPruchaseQty && SPZ.whenApiDefined(minPruchaseQty).then((api) => { api.render(data, true); }); } // 添加商品子款式 renderVariantTag() { let variantInfo; const quickShopBody = SPZCore.Dom.scopedQuerySelector(document.body, '#promotion-quick-shop-body'); quickShopBody && SPZ.whenApiDefined(quickShopBody).then((api) => { variantInfo = api.getVariantsData(); console.log(variantInfo, 'variantInfo'); const productId = variantInfo.product_id; const variantId = variantInfo.variant_id; const minPruchaseQtyRender = variantInfo.product.discount_min_purchase_qty || variantInfo.variant.discount_info.discount_min_purchase_qty; if(this.discount_type == 'DT_MIX_MATCH_BUNDLE') { const index = this.productStyleInfo.findIndex((item) => item.variant_id == variantInfo.variant_id); if (index != -1) { this.productStyleInfo[index].quantity = Number(this.productStyleInfo[index].quantity) + Number(variantInfo.quantity); this.updateProductPrice_(this.productStyleInfo); } else { this.productStyleInfo.push(this.getFilteredVariants_(variantInfo)); // 若当前商品已选中,更新商品价格 const currentProductOption = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionSelectOption-${productId}`); const isSelected = currentProductOption && currentProductOption.hasAttribute('selected'); isSelected && this.updateProductPrice_(this.productStyleInfo); } const selectedVariantsFilter = this.productStyleInfo.filter((item) => item.product_id == productId); this.handleSpzVariantRender_(selectedVariantsFilter, productId); this.handleProductOption_(productId, true); } else { if(this.discount_info.enable_min_purchase_qty == true && this.discount_info.min_purchase_qty_type == 'spu' && minPruchaseQtyRender > 1) { const index = this.modalVariantInfo.findIndex((item) => item.variant_id == variantId); if (index != -1) { this.modalVariantInfo[index].quantity = Number(this.modalVariantInfo[index].quantity) + 1; } else { this.modalVariantInfo.push(this.getFilteredVariants_(variantInfo, 'classic_spu')); } const modalVariantTag = SPZCore.Dom.scopedQuerySelector(document.body, '#promotionModalVariantTagRender'); modalVariantTag && SPZ.whenApiDefined(modalVariantTag).then((api) => { api.render(this.modalVariantInfo, true); }); this.handleModalInventoryCheck_(variantInfo); const selectedVariantsNum = this.modalVariantInfo.reduce((acc, item) => { return acc + item.quantity; }, 0); if(selectedVariantsNum == minPruchaseQtyRender) { this.handleSpzVariantRender_([this.getFilteredVariants_(variantInfo)], productId); this.productStyleInfo = this.productStyleInfo.filter((item) => item.product_id != productId).concat(this.modalVariantInfo); const renderData = this.productStyleInfo.filter((item) => item.product_id == productId).map((item) => { return { ...item, is_classic_bundle_product_list_variant_tag: true } }); const classicSpuTag = SPZCore.Dom.scopedQuerySelector(document.body, `#promotionClassicSpuTags-${productId}`); classicSpuTag && SPZ.whenApiDefined(classicSpuTag).then((api) => { api.render(renderData, true); }); this.updateProductPrice_(this.productStyleInfo); const quickView = SPZCore.Dom.scopedQuerySelector(document.body, '#promotion-quick-view'); quickView && SPZ.whenApiDefined(quickView).then((api)=>{ api.close(); }); this.modalVariantInfo = []; } else { return; } } // this.productStyleInfo 中已存在与productId, variantId都相同的商品 则直接return 关闭弹窗 const isExist = this.productStyleInfo.some((item) => item.product_id == productId && item.variant_id == variantId); if (isExist) { const quickView = SPZCore.Dom.scopedQuerySelector(document.body, '#promotion-quick-view'); quickView && SPZ.whenApiDefined(quickView).then((api)=>{ api.close(); }); return; } // 更新this.productStyleInfo中的商品款式信息 const index = this.productStyleInfo.findIndex((item) => item.product_id == productId); if (index != -1) { this.productStyleInfo[index] = this.getFilteredVariants_(variantInfo); } const selectedVariantsFilter = this.productStyleInfo.filter((item) => item.product_id == productId); this.handleSpzVariantRender_(selectedVariantsFilter, productId); this.handleMinPurchaseQtyUpdate_({discount_min_purchase_qty: minPruchaseQtyRender}, productId); this.updateProductPrice_(this.productStyleInfo); } const quickView = SPZCore.Dom.scopedQuerySelector(document.body, '#promotion-quick-view'); quickView && SPZ.whenApiDefined(quickView).then((api)=>{ api.close(); }); }); } // 混搭弹窗内的前端库存校验 handleModalInventoryCheck_(data) { if(this.discount_type == 'DT_CLASSIC_BUNDLE') { const currentVariantAddNum = this.modalVariantInfo.find((item) => item.variant_id == data.variant_id)?.quantity || 0; const quickShopBody = SPZCore.Dom.scopedQuerySelector(document.body, '#promotion-quick-shop-body'); if(!!data.variant && currentVariantAddNum == Number(data.variant.available_quantity)) { quickShopBody && quickShopBody.setAttribute('status', 'soldout'); } else { quickShopBody && quickShopBody.setAttribute('status', 'available'); } } } // 删除商品子款式 deleteVariantTag(data) { const { product_id, variant_id } = data; if(this.discount_info.enable_min_purchase_qty == true && this.discount_info.min_purchase_qty_type == 'spu') { const modalProductVariants = this.modalVariantInfo.filter((item) => item.product_id == product_id && item.variant_id != variant_id); const modalVariantTag = SPZCore.Dom.scopedQuerySelector(document.body, '#promotionModalVariantTagRender'); modalVariantTag && SPZ.whenApiDefined(modalVariantTag).then((api) => { api.render(modalProductVariants, true); }); this.handleModalInventoryCheck_(data); this.modalVariantInfo = modalProductVariants; return; } const currentProductVariants = this.productStyleInfo.filter((item) => item.product_id == product_id && item.variant_id != variant_id); this.handleSpzVariantRender_(currentProductVariants, product_id); // 更新selectedVariants this.productStyleInfo = this.productStyleInfo.filter((item) => item.variant_id != variant_id); if(currentProductVariants.length > 0) { // currentProductVariants 中只要有一项是多款式商品,就更新价格 const isMultiStyle = currentProductVariants.some((item) => item.is_multi_style); isMultiStyle && this.updateProductPrice_(this.productStyleInfo); this.handleProductOption_(product_id, true); } else { this.handleProductOption_(product_id, false); this.resetProductPrice_(this.products.find((item) => item.id == product_id)); } } // 加购弹窗未参与活动 加购按钮不可点击 handleNotHitDiscount_(data) { const $quickShopBody = document.querySelector('#promotion-quick-shop-body'); //当前子框式未命中活动 if(data.variant.is_hit_discount == false) { $quickShopBody.setAttribute('variantstatus', 'notHitDiscount') } else { $quickShopBody.setAttribute('variantstatus', '') } } // loading handleLoading_(event) { const { type, action } = event; const loadingElementId = type == 'product' ? '#promotionProductsLoading' : '#promotionWholeLoading'; const loadingElement = document.querySelector(loadingElementId); if (loadingElement) { SPZ.whenApiDefined(loadingElement).then((api) => { if (action == 'show') { api.show_(); } else { api.close_(); } }); } } handleSelectProduct(productArr) { // 从this.productStyleInfo 过滤出选中的商品 const selectedProducts = this.productStyleInfo.filter((item) => productArr.includes(item.product_id)); this.updateProductPrice_(selectedProducts); } handleMixMatchBundleFilterSelected_(data) { const selectedOptions = SPZCore.Dom.scopedQuerySelectorAll(document.body, '[id^="promotionSelectOption-"]'); const idArr = [...selectedOptions].reduce((acc, item) => { if (item.hasAttribute('selected')) { const optionValue = item.getAttribute('option'); if (optionValue) { acc.push(optionValue); } } return acc; }, []); if(this.discount_type == 'DT_MIX_MATCH_BUNDLE') { return data.filter((item) => idArr.includes(item.product_id)); } return data; } setupAction_() { this.registerAction('handleTabChange', (invocation) => { const { panelId } = invocation.args.data; this.tabChange_(panelId); }); // 监听排序组件 选中选项 this.registerAction('handleSort', (invocation) => { const data = invocation.args.data; this.handleSort_(data); }); // 渲染加购弹窗 this.registerAction('renderQuickShop', (invocation) => { const data = invocation.args; this.renderQuickShop(data); }); this.registerAction('renderSingleVariant', (invocation) => { const data = invocation.args; this.renderSingleVariant(data); }); // 捆绑商品加购/立即购买 this.registerAction('handleBundleAddToCart', (invocation) => { const data = invocation.args; this.handleBundleAddToCart_(data); }); // 子款式 未参与活动 this.registerAction('handleNotHitDiscount', (invocation) => { const data = invocation.args.data; this.handleNotHitDiscount_(data); }); // 加购提示 this.registerAction('handleAddToCartToast', (invocation) => { const langValue = this.currentLangMap['add_to_cart_successfully']; this.triggerEvent_("addToCartToast", langValue); }); this.registerAction('getVariantInfo', (invocation) => { this.renderVariantTag(); }); this.registerAction('deleteVariantTag', (invocation) => { const data = invocation.args; this.deleteVariantTag(data); }); this.registerAction('getSelectedProduct', (invocation) => { const data = invocation.args.data; this.handleSelectProduct(data); }); this.registerAction('pageReload', () => { window.location.reload(); }); this.registerAction('resetModalVariantInfo', () => { this.modalVariantInfo = []; }); this.registerAction('handleModalInventoryCheck', (invocation) => { const data = invocation.args.data; this.handleModalInventoryCheck_(data); }); this.registerAction('changeTextSoldOut', (invocation) => { const data = invocation.args.data; const addBtn = SPZCore.Dom.scopedQuerySelector(document.body, '.promotion-shop-btn[role="confirm"]'); if(!addBtn) return; const content = SPZCore.Dom.scopedQuerySelector(addBtn, '[role="content"]'); if (content) { const langValue = this.currentLangMap[data.variant.available ? 'add' : 'sold_out']; content.innerHTML = langValue; } }); } triggerEvent_(name, data) { const event = SPZUtils.Event.create(this.win, `${ TAG }.${ name }`, data || {}); this.action_.trigger(this.element, name, event); } isLayoutSupported(layout) { return layout == SPZCore.Layout.CONTAINER; } } SPZ.defineElement(TAG, SpzCustomDiscountDefault)
16
Sort by