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":"440764793998029417","center_id":"1199785","discount_name":"BOGO 50% OFF - Select PE","display_name":"BOGO 50% OFF - Select PE","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":"1856ea20-656a-4aed-91c5-98009eed2c39","title":"BCAA + Energy with Electrolytes","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"bcaa-energy-with-electrolytes","spu":"43376","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\/bcaa-energy-with-electrolytes","sales":"0","image":{"src":"\/\/img.staticdj.com\/c38dd80ac9abc12d1e02ae62fa1a3770.jpeg","alt":"","path":"c38dd80ac9abc12d1e02ae62fa1a3770.jpeg","width":2000,"height":2000},"variants":[{"id":"391e8cca-3389-4353-9ce8-ba3d84fe0d57","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"43376","barcode":"191069433769","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"24.99","price":"24.99","available":false,"url":"\/products\/bcaa-energy-with-electrolytes?variant=391e8cca-3389-4353-9ce8-ba3d84fe0d57","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\/c38dd80ac9abc12d1e02ae62fa1a3770.jpeg","alt":"","path":"c38dd80ac9abc12d1e02ae62fa1a3770.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"dd2fb7b1-8a75-4962-aa03-b005ebf5abca","title":"BCAA Lemon Lime","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"bcaa-lemon-lime","spu":"50750_master","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\/bcaa-lemon-lime","sales":"26","image":{"src":"\/\/img.staticdj.com\/85aca697cddfec2c0c66029857ca5d38.jpeg","alt":"","path":"85aca697cddfec2c0c66029857ca5d38.jpeg","width":2000,"height":2000},"variants":[{"id":"6c1e492d-d260-44b5-b2c4-f0a740370cf7","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"50750","barcode":"191069507507","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.84","compare_at_price":"24.99","price":"24.99","available":false,"url":"\/products\/bcaa-lemon-lime?variant=6c1e492d-d260-44b5-b2c4-f0a740370cf7","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\/85aca697cddfec2c0c66029857ca5d38.jpeg","alt":"","path":"85aca697cddfec2c0c66029857ca5d38.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/cfc4adcefe9eb7ddb6a9ee1636f7d437.jpg","alt":"","path":"cfc4adcefe9eb7ddb6a9ee1636f7d437.jpg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"f621f77f-01d7-42e4-9d9b-6436347f24fc","title":"BCAA Powder Blue Raspberry","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"precision-engineered-bcaa-powder","spu":"51714_master","note":"","need_variant_image":true,"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\/precision-engineered-bcaa-powder","sales":"14","image":{"src":"\/\/img.staticdj.com\/f93625dd0fd63de7a64eaf0d687781c9.jpeg","alt":"","path":"f93625dd0fd63de7a64eaf0d687781c9.jpeg","width":2000,"height":2000},"variants":[{"id":"51c24cc5-03a9-4ea2-9f36-caa256ee5603","title":"Watermelon","weight_unit":"lb","inventory_quantity":"0","sku":"52200","barcode":"191069522005","position":1,"option1":"Watermelon","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/052680bba8d03f8114984819b0ada84c.jpeg","alt":"","path":"052680bba8d03f8114984819b0ada84c.jpeg","width":2000,"height":2000},"weight":"0.85","compare_at_price":"24.99","price":"24.99","available":false,"url":"\/products\/precision-engineered-bcaa-powder?variant=51c24cc5-03a9-4ea2-9f36-caa256ee5603","available_quantity":"0","options":[{"name":"Flavor","value":"Watermelon"}],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/f93625dd0fd63de7a64eaf0d687781c9.jpeg","alt":"","path":"f93625dd0fd63de7a64eaf0d687781c9.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/85aca697cddfec2c0c66029857ca5d38.jpeg","alt":"","path":"85aca697cddfec2c0c66029857ca5d38.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/95666dc12dade9ba4843e774cf182b9a.jpeg","alt":"","path":"95666dc12dade9ba4843e774cf182b9a.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/71a2c2067af122b2efb04673efd6bcf4.jpeg","alt":"","path":"71a2c2067af122b2efb04673efd6bcf4.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/88b342bf2ade811641d2265acf0a2496.jpeg","alt":"","path":"88b342bf2ade811641d2265acf0a2496.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/de3635fcaefe067aa0e55918ffe6aa24.jpeg","alt":"","path":"de3635fcaefe067aa0e55918ffe6aa24.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/f93625dd0fd63de7a64eaf0d687781c9.jpeg","alt":"","path":"f93625dd0fd63de7a64eaf0d687781c9.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4f0959b0994dfb4de2e8942ce51c5d47.jpeg","alt":"","path":"4f0959b0994dfb4de2e8942ce51c5d47.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/3563c2df260283ca9c8fccb41f71e0cd.jpeg","alt":"","path":"3563c2df260283ca9c8fccb41f71e0cd.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/052680bba8d03f8114984819b0ada84c.jpeg","alt":"","path":"052680bba8d03f8114984819b0ada84c.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"6f046705-03a2-4619-9670-51e5db68c80b","title":"BCAA+ Citrulline Malate Powder","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"bcaa-citrulline-malate-powder","spu":"59861","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\/bcaa-citrulline-malate-powder","sales":"10","image":{"src":"\/\/img.staticdj.com\/f9e1a81a35d7d848a86763fdfdf42277.jpeg","alt":"","path":"f9e1a81a35d7d848a86763fdfdf42277.jpeg","width":2000,"height":2000},"variants":[{"id":"80b6f76f-8833-4cd2-b4e2-b3050f9dc135","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"59861","barcode":"191069598611","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/bcaa-citrulline-malate-powder?variant=80b6f76f-8833-4cd2-b4e2-b3050f9dc135","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\/f9e1a81a35d7d848a86763fdfdf42277.jpeg","alt":"","path":"f9e1a81a35d7d848a86763fdfdf42277.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"4d81b6fa-3a45-497c-9b6b-15f2a93e7ae8","title":"Chocolate Whey Protein Isolate 5lbs","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"whey-protein-isolate-5lbs","spu":"36250_master","note":"","need_variant_image":true,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"99.99","price_max":"99.99","price":"99.99","compare_at_price":"99.99","compare_at_price_min":"99.99","compare_at_price_max":"99.99","url":"\/products\/whey-protein-isolate-5lbs","sales":"1","image":{"src":"\/\/img.staticdj.com\/1049370988884572748ac00862e360cd.jpeg","alt":"","path":"1049370988884572748ac00862e360cd.jpeg","width":2000,"height":2000},"variants":[{"id":"32211e94-bda0-47a4-81ce-64257ad1b78f","title":"Chocolate","weight_unit":"lb","inventory_quantity":"0","sku":"36250","barcode":"191069362502","position":1,"option1":"Chocolate","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","alt":"","path":"a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","width":2000,"height":2000},"weight":"5.14","compare_at_price":"99.99","price":"99.99","available":false,"url":"\/products\/whey-protein-isolate-5lbs?variant=32211e94-bda0-47a4-81ce-64257ad1b78f","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\/1049370988884572748ac00862e360cd.jpeg","alt":"","path":"1049370988884572748ac00862e360cd.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","alt":"","path":"a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/1049370988884572748ac00862e360cd.jpeg","alt":"","path":"1049370988884572748ac00862e360cd.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","alt":"","path":"a0b8eda169c0460c2a7ad16dfe0f3f62.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"64846bd8-d512-4157-9277-2572a0ac1581","title":"Creatine Extreme Caps 1000MG","brief":"","vendor":"ROBINSON PHARMA","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"7","inventory_tracking":true,"published":true,"handle":"creatine-extreme-caps-1-000mg","spu":"19462","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"37.99","price_max":"37.99","price":"37.99","compare_at_price":"37.99","compare_at_price_min":"37.99","compare_at_price_max":"37.99","url":"\/products\/creatine-extreme-caps-1-000mg","sales":"171","image":{"src":"\/\/img.staticdj.com\/cc6cac06d927973118f154c8bfdea92e.jpeg","alt":"","path":"cc6cac06d927973118f154c8bfdea92e.jpeg","width":2000,"height":2000},"variants":[{"id":"a265e4d7-8448-43a6-8388-636132a9877b","title":"","weight_unit":"lb","inventory_quantity":"7","sku":"19462","barcode":"191069194622","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"37.99","price":"37.99","available":true,"url":"\/products\/creatine-extreme-caps-1-000mg?variant=a265e4d7-8448-43a6-8388-636132a9877b","available_quantity":"7","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/cc6cac06d927973118f154c8bfdea92e.jpeg","alt":"","path":"cc6cac06d927973118f154c8bfdea92e.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"08775d30-159a-489e-8d46-d905eb06eccc","title":"Fireball Inferno CLA","brief":"","vendor":"ROBINSON PHARMA","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"fireball-inferno-cla","spu":"17947","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"26.99","price_max":"26.99","price":"26.99","compare_at_price":"26.99","compare_at_price_min":"26.99","compare_at_price_max":"26.99","url":"\/products\/fireball-inferno-cla","sales":"669","image":{"src":"\/\/img.staticdj.com\/9bb44818dcd56fe0fcae0822ed428d3b.jpeg","alt":"","path":"9bb44818dcd56fe0fcae0822ed428d3b.jpeg","width":2000,"height":2000},"variants":[{"id":"920d5ff8-c9da-40ae-b72a-eb2b4ee76eb9","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"17947","barcode":"191069179476","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"26.99","price":"26.99","available":false,"url":"\/products\/fireball-inferno-cla?variant=920d5ff8-c9da-40ae-b72a-eb2b4ee76eb9","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\/9bb44818dcd56fe0fcae0822ed428d3b.jpeg","alt":"","path":"9bb44818dcd56fe0fcae0822ed428d3b.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"e08fccaf-63d2-4e73-b0f2-79e4e99e7152","title":"Fireball Liqui-Fusion","brief":"","vendor":"ROBINSON PHARMA","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"fireball-liqui-fusion","spu":"16640","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"28.49","price_max":"28.49","price":"28.49","compare_at_price":"28.49","compare_at_price_min":"28.49","compare_at_price_max":"28.49","url":"\/products\/fireball-liqui-fusion","sales":"0","image":{"src":"\/\/img.staticdj.com\/449aa30ee5db5b96a657098065d7941e.jpeg","alt":"","path":"449aa30ee5db5b96a657098065d7941e.jpeg","width":2000,"height":2000},"variants":[{"id":"3907fd11-9378-4ef0-bd22-f328daaf8ca8","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"16640","barcode":"191069166407","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"28.49","price":"28.49","available":false,"url":"\/products\/fireball-liqui-fusion?variant=3907fd11-9378-4ef0-bd22-f328daaf8ca8","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\/449aa30ee5db5b96a657098065d7941e.jpeg","alt":"","path":"449aa30ee5db5b96a657098065d7941e.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"e77a9a2d-9c69-4a79-90b5-787107900ad4","title":"Hardcore Creatine 510 g","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"hardcore-creatine-510-g","spu":"907","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"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\/hardcore-creatine-510-g","sales":"42","image":{"src":"\/\/img.staticdj.com\/70f7b1cbe09bf04b168bd8cac10c74a7.jpeg","alt":"","path":"70f7b1cbe09bf04b168bd8cac10c74a7.jpeg","width":2000,"height":2000},"variants":[{"id":"82566f7d-d629-49ef-9c21-be0f9d008d7b","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"907","barcode":"191069709079","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"49.99","price":"49.99","available":false,"url":"\/products\/hardcore-creatine-510-g?variant=82566f7d-d629-49ef-9c21-be0f9d008d7b","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\/70f7b1cbe09bf04b168bd8cac10c74a7.jpeg","alt":"","path":"70f7b1cbe09bf04b168bd8cac10c74a7.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"3e1a10e0-878f-41ac-b9de-5b662e08626a","title":"Kre-Alkalyn 1500MG","brief":"","vendor":"ADH HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"kre-alkalyn-1500-mg","spu":"13417","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"33.99","price_max":"33.99","price":"33.99","compare_at_price":"33.99","compare_at_price_min":"33.99","compare_at_price_max":"33.99","url":"\/products\/kre-alkalyn-1500-mg","sales":"0","image":{"src":"\/\/img.staticdj.com\/e19b0d84622396711b6407b335d1d13f.jpeg","alt":"","path":"e19b0d84622396711b6407b335d1d13f.jpeg","width":2000,"height":2000},"variants":[{"id":"651f3b41-be84-407d-9c83-f49b3dedc21d","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"13417","barcode":"191069134178","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"33.99","price":"33.99","available":false,"url":"\/products\/kre-alkalyn-1500-mg?variant=651f3b41-be84-407d-9c83-f49b3dedc21d","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\/e19b0d84622396711b6407b335d1d13f.jpeg","alt":"","path":"e19b0d84622396711b6407b335d1d13f.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"fc46ce2d-c8c2-4337-a790-0de3ad3abfe1","title":"L-Glutamine Powder 4500 mg","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"l-glutamine-powder-4500-mg","spu":"67813","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"32.99","price_max":"32.99","price":"32.99","compare_at_price":"32.99","compare_at_price_min":"32.99","compare_at_price_max":"32.99","url":"\/products\/l-glutamine-powder-4500-mg","sales":"0","image":{"src":"\/\/img.staticdj.com\/3a5436b3c3795eba545dacc0cb36452e.jpeg","alt":"","path":"3a5436b3c3795eba545dacc0cb36452e.jpeg","width":2000,"height":2000},"variants":[{"id":"5ae510bb-e677-4ba1-97a3-d133d68b197f","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"67813","barcode":"191069678139","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"32.99","price":"32.99","available":false,"url":"\/products\/l-glutamine-powder-4500-mg?variant=5ae510bb-e677-4ba1-97a3-d133d68b197f","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\/3a5436b3c3795eba545dacc0cb36452e.jpeg","alt":"","path":"3a5436b3c3795eba545dacc0cb36452e.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"66f61667-3a81-4e02-a0b9-eaef3484466c","title":"Liquid L-Carnitine Watermelon 3000MG","brief":"","vendor":"PURE SOURCE","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"liquid-l-carnitine-watermelon-3000mg","spu":"53122","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\/liquid-l-carnitine-watermelon-3000mg","sales":"0","image":{"src":"\/\/img.staticdj.com\/0803e071981a62c709a9c1812769121d.jpeg","alt":"","path":"0803e071981a62c709a9c1812769121d.jpeg","width":2000,"height":2000},"variants":[{"id":"c06f059b-69e3-4b53-b57b-5109e4193473","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"53122","barcode":"191069531229","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/liquid-l-carnitine-watermelon-3000mg?variant=c06f059b-69e3-4b53-b57b-5109e4193473","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\/0803e071981a62c709a9c1812769121d.jpeg","alt":"","path":"0803e071981a62c709a9c1812769121d.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"8fe8a12a-7e0a-432b-8022-dbe705668035","title":"Muscle & Weight Gainer Creamy French Vanilla","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"muscle-weight-gainer-creamy-french-vanilla","spu":"53920_master","note":"","need_variant_image":true,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"74.99","price_max":"74.99","price":"74.99","compare_at_price":"74.99","compare_at_price_min":"74.99","compare_at_price_max":"74.99","url":"\/products\/muscle-weight-gainer-creamy-french-vanilla","sales":"27","image":{"src":"\/\/img.staticdj.com\/26724501a52e5e9b998058759a462411.jpeg","alt":"","path":"26724501a52e5e9b998058759a462411.jpeg","width":2000,"height":2000},"variants":[{"id":"59670917-578a-4aaf-8022-bca4ef568b7f","title":"Creamy French Vanilla","weight_unit":"lb","inventory_quantity":"0","sku":"53920","barcode":"191069539201","position":1,"option1":"Creamy French Vanilla","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/1d945ba3051cbcf1d7a882d58e11dcd2.jpeg","alt":"","path":"1d945ba3051cbcf1d7a882d58e11dcd2.jpeg","width":2000,"height":2000},"weight":"12.24","compare_at_price":"74.99","price":"74.99","available":false,"url":"\/products\/muscle-weight-gainer-creamy-french-vanilla?variant=59670917-578a-4aaf-8022-bca4ef568b7f","available_quantity":"0","options":[{"name":"Flavor","value":"Creamy French Vanilla"}],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/26724501a52e5e9b998058759a462411.jpeg","alt":"","path":"26724501a52e5e9b998058759a462411.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/ed41213937662b07133c76b2899076de.jpeg","alt":"","path":"ed41213937662b07133c76b2899076de.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/eb878da752a5f6f6f243acbec4dea125.jpeg","alt":"","path":"eb878da752a5f6f6f243acbec4dea125.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/1d945ba3051cbcf1d7a882d58e11dcd2.jpeg","alt":"","path":"1d945ba3051cbcf1d7a882d58e11dcd2.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"5aeb76c9-c1b7-4a65-afb2-6542a48c200e","title":"Muscle & Weight Gainer Double Chocolate Supreme","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"muscle-weight-gainer-double-chocolate-supreme","spu":"51422_master","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"74.99","price_max":"74.99","price":"74.99","compare_at_price":"74.99","compare_at_price_min":"74.99","compare_at_price_max":"74.99","url":"\/products\/muscle-weight-gainer-double-chocolate-supreme","sales":"0","image":{"src":"\/\/img.staticdj.com\/26724501a52e5e9b998058759a462411.jpg","alt":"","path":"26724501a52e5e9b998058759a462411.jpg","width":2000,"height":2000},"variants":[{"id":"855ee1a4-d3f0-4e84-bbc8-8befe8c70405","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"51422","barcode":"191069514222","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"12.29","compare_at_price":"74.99","price":"74.99","available":false,"url":"\/products\/muscle-weight-gainer-double-chocolate-supreme?variant=855ee1a4-d3f0-4e84-bbc8-8befe8c70405","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\/26724501a52e5e9b998058759a462411.jpg","alt":"","path":"26724501a52e5e9b998058759a462411.jpg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"75bda99e-dd54-4b27-9738-d58b3f1df856","title":"Muscle & Weight Gainer Double Chocolate Supreme","brief":"Muscle & Weight Gainer Double Chocolate Supreme","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"muscle-weight-gainer-double-chocolate-supreme-2kyd","spu":"5464","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\/muscle-weight-gainer-double-chocolate-supreme-2kyd","sales":"0","image":{"src":"\/\/img.staticdj.com\/c11c3850c1c64d7d62a2991800d2c092.jpg","alt":"","path":"c11c3850c1c64d7d62a2991800d2c092.jpg","width":2000,"height":2000},"variants":[{"id":"068fb673-6ebe-430f-aa5e-f9a2a6f29d91","title":"","weight_unit":"kg","inventory_quantity":"0","sku":"5464","barcode":"191069054643","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/muscle-weight-gainer-double-chocolate-supreme-2kyd?variant=068fb673-6ebe-430f-aa5e-f9a2a6f29d91","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\/c11c3850c1c64d7d62a2991800d2c092.jpg","alt":"","path":"c11c3850c1c64d7d62a2991800d2c092.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"c1a5c891-91f1-4738-9b0a-deb917672b96","title":"NOS Blast Factor Caps","brief":"","vendor":"ROBINSON PHARMA","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"nos-blast-factor-caps","spu":"35721","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"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\/nos-blast-factor-caps","sales":"0","image":{"src":"\/\/img.staticdj.com\/c8d30eb50630bc30a93a36176d64727d.jpeg","alt":"","path":"c8d30eb50630bc30a93a36176d64727d.jpeg","width":2000,"height":2000},"variants":[{"id":"60dca240-654a-4894-bab5-15329a076773","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"35721","barcode":"191069357218","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"34.99","price":"34.99","available":false,"url":"\/products\/nos-blast-factor-caps?variant=60dca240-654a-4894-bab5-15329a076773","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\/c8d30eb50630bc30a93a36176d64727d.jpeg","alt":"","path":"c8d30eb50630bc30a93a36176d64727d.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"0f0f1963-8a5e-4c8b-8baa-3b3f09837604","title":"Perfect Storm Post Workout Creatine","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"perfect-storm-post-workout-creatine","spu":"18795","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"41.99","price_max":"41.99","price":"41.99","compare_at_price":"41.99","compare_at_price_min":"41.99","compare_at_price_max":"41.99","url":"\/products\/perfect-storm-post-workout-creatine","sales":"16","image":{"src":"\/\/img.staticdj.com\/b587f8eda3bb7763f4404ef4c40bf092.jpeg","alt":"","path":"b587f8eda3bb7763f4404ef4c40bf092.jpeg","width":2000,"height":2000},"variants":[{"id":"2aedb5f1-e3db-40b4-8611-1514aa31f299","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"18795","barcode":"191069187952","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"41.99","price":"41.99","available":false,"url":"\/products\/perfect-storm-post-workout-creatine?variant=2aedb5f1-e3db-40b4-8611-1514aa31f299","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\/b587f8eda3bb7763f4404ef4c40bf092.jpeg","alt":"","path":"b587f8eda3bb7763f4404ef4c40bf092.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"2268dc17-90f8-4e7a-98fd-391e521c81ca","title":"Syntha Phase Whey Protein Vanilla Milkshake","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"syntha-phase-whey-protein","spu":"31237_master","note":"","need_variant_image":true,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"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\/syntha-phase-whey-protein","sales":"0","image":{"src":"\/\/img.staticdj.com\/687fa1bb441fa36c6ace3ab1dafd1119.jpeg","alt":"","path":"687fa1bb441fa36c6ace3ab1dafd1119.jpeg","width":2000,"height":2000},"variants":[{"id":"18a8cd20-6a38-4772-9fd9-e3f575cc9e15","title":"Vanilla Milkshake","weight_unit":"lb","inventory_quantity":"0","sku":"31237","barcode":"191069312378","position":1,"option1":"Vanilla Milkshake","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/687fa1bb441fa36c6ace3ab1dafd1119.jpeg","alt":"","path":"687fa1bb441fa36c6ace3ab1dafd1119.jpeg","width":2000,"height":2000},"weight":"3.19","compare_at_price":"39.99","price":"39.99","available":false,"url":"\/products\/syntha-phase-whey-protein?variant=18a8cd20-6a38-4772-9fd9-e3f575cc9e15","available_quantity":"0","options":[{"name":"Flavor","value":"Vanilla Milkshake"}],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/687fa1bb441fa36c6ace3ab1dafd1119.jpeg","alt":"","path":"687fa1bb441fa36c6ace3ab1dafd1119.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/2dfc1d8fa3565654f7aa3afa8494e7bc.jpeg","alt":"","path":"2dfc1d8fa3565654f7aa3afa8494e7bc.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/687fa1bb441fa36c6ace3ab1dafd1119.jpeg","alt":"","path":"687fa1bb441fa36c6ace3ab1dafd1119.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/2dfc1d8fa3565654f7aa3afa8494e7bc.jpeg","alt":"","path":"2dfc1d8fa3565654f7aa3afa8494e7bc.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"ce4322d0-3a6c-4ff6-96f5-96aaf9c161de","title":"Whey Protein Deluxe Chocolate","brief":"","vendor":"NUTRIVO","vendor_url":"","has_only_default_variant":false,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"precision-engineered-whey-protein","spu":"67689_master","note":"","need_variant_image":true,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"79.99","price_max":"79.99","price":"79.99","compare_at_price":"79.99","compare_at_price_min":"79.99","compare_at_price_max":"79.99","url":"\/products\/precision-engineered-whey-protein","sales":"0","image":{"src":"\/\/img.staticdj.com\/4d4609ef0facf2890ff8a8672bea8c30.jpeg","alt":"","path":"4d4609ef0facf2890ff8a8672bea8c30.jpeg","width":2000,"height":2000},"variants":[{"id":"d99f2478-e3d7-450a-820a-e0a51afa20b2","title":"Creamy Vanilla","weight_unit":"lb","inventory_quantity":"0","sku":"67698","barcode":"191069676982","position":1,"option1":"Creamy Vanilla","option2":"","option3":"","note":"","image":{"src":"\/\/img.staticdj.com\/4d4609ef0facf2890ff8a8672bea8c30.jpeg","alt":"","path":"4d4609ef0facf2890ff8a8672bea8c30.jpeg","width":2000,"height":2000},"weight":"5.70","compare_at_price":"79.99","price":"79.99","available":false,"url":"\/products\/precision-engineered-whey-protein?variant=d99f2478-e3d7-450a-820a-e0a51afa20b2","available_quantity":"0","options":[{"name":"Flavor","value":"Creamy Vanilla"}],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/4d4609ef0facf2890ff8a8672bea8c30.jpeg","alt":"","path":"4d4609ef0facf2890ff8a8672bea8c30.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4d4609ef0facf2890ff8a8672bea8c30.jpeg","alt":"","path":"4d4609ef0facf2890ff8a8672bea8c30.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/5a3b56f69f61c399b19b7bbd4debdd8b.jpeg","alt":"","path":"5a3b56f69f61c399b19b7bbd4debdd8b.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":19,"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)
19
Sort by
BCAA + Energy with Electrolytes
$24.99 $24.99
Sold out
BCAA Lemon Lime
$24.99 $24.99
Sold out
BCAA Powder Blue Raspberry
$24.99 $24.99
Sold out
BCAA+ Citrulline Malate Powder
$29.99 $29.99
Sold out
Chocolate Whey Protein Isolate 5lbs
$99.99 $99.99
Sold out
Fireball Inferno CLA
$26.99 $26.99
Sold out
Fireball Liqui-Fusion
$28.49 $28.49
Sold out
Hardcore Creatine 510 g
$49.99 $49.99
Sold out
Kre-Alkalyn 1500MG
$33.99 $33.99
Sold out
L-Glutamine Powder 4500 mg
$32.99 $32.99
Sold out