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":"440759493454740649","center_id":"1199785","discount_name":"BOGO 50% OFF - Windmill","display_name":"BOGO 50% OFF - Windmill","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":"242744ab-43b4-4500-a5c8-f498182e2252","title":"CHLOROPURE\u2122 Liquid Chlorophyll","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"13","inventory_tracking":true,"published":true,"handle":"chloropure-liquid-chlorophyll","spu":"44516","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"11.99","price_max":"11.99","price":"11.99","compare_at_price":"11.99","compare_at_price_min":"11.99","compare_at_price_max":"11.99","url":"\/products\/chloropure-liquid-chlorophyll","sales":"10","image":{"src":"\/\/img.staticdj.com\/077529a0709916ae737b67e18b318cbe.jpeg","alt":"","path":"077529a0709916ae737b67e18b318cbe.jpeg","width":2000,"height":2000},"variants":[{"id":"a8ac4fea-8e67-44ff-a6fb-78d4f26b1d3b","title":"","weight_unit":"lb","inventory_quantity":"13","sku":"44516","barcode":"035046124564","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"11.99","price":"11.99","available":true,"url":"\/products\/chloropure-liquid-chlorophyll?variant=a8ac4fea-8e67-44ff-a6fb-78d4f26b1d3b","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\/077529a0709916ae737b67e18b318cbe.jpeg","alt":"","path":"077529a0709916ae737b67e18b318cbe.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"6b30db5a-f70f-4df8-8576-b3a62ad2d559","title":"Collagen Beauty Formula Liquid","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"42","inventory_tracking":true,"published":true,"handle":"collagen-beauty-formula-liquid","spu":"34870","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"12.99","price_max":"12.99","price":"12.99","compare_at_price":"12.99","compare_at_price_min":"12.99","compare_at_price_max":"12.99","url":"\/products\/collagen-beauty-formula-liquid","sales":"8","image":{"src":"\/\/img.staticdj.com\/1b25d5b9dd99d20481b6779ee13b6d0c.jpeg","alt":"","path":"1b25d5b9dd99d20481b6779ee13b6d0c.jpeg","width":2000,"height":2000},"variants":[{"id":"c8767401-ff80-44b5-87a6-687bd4700291","title":"","weight_unit":"lb","inventory_quantity":"42","sku":"34870","barcode":"035046074210","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"12.99","price":"12.99","available":true,"url":"\/products\/collagen-beauty-formula-liquid?variant=c8767401-ff80-44b5-87a6-687bd4700291","available_quantity":"42","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/1b25d5b9dd99d20481b6779ee13b6d0c.jpeg","alt":"","path":"1b25d5b9dd99d20481b6779ee13b6d0c.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/d78809d6d912488a20986b1daa3e170a.jpeg","alt":"","path":"d78809d6d912488a20986b1daa3e170a.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/53dfaeb4b32bed17ad0e175757c5f8b7.jpeg","alt":"","path":"53dfaeb4b32bed17ad0e175757c5f8b7.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"e2c03dd7-9a67-45ad-a004-737261416e7c","title":"Country Farms Bountiful Beets Powder - Cherry","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"17","inventory_tracking":true,"published":true,"handle":"country-farms-bountiful-beets-powder-cherry","spu":"","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\/country-farms-bountiful-beets-powder-cherry","sales":"0","image":{"src":"\/\/img.staticdj.com\/79cd97298ddc642f3152ffb22fab5ff8.jpg","alt":"","path":"79cd97298ddc642f3152ffb22fab5ff8.jpg","width":2000,"height":2000},"variants":[{"id":"bf7e6423-dc3d-4494-9b5c-170da2595625","title":"","weight_unit":"kg","inventory_quantity":"17","sku":"77031","barcode":"035046098308","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"19.99","price":"19.99","available":true,"url":"\/products\/country-farms-bountiful-beets-powder-cherry?variant=bf7e6423-dc3d-4494-9b5c-170da2595625","available_quantity":"17","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/79cd97298ddc642f3152ffb22fab5ff8.jpg","alt":"","path":"79cd97298ddc642f3152ffb22fab5ff8.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/ec73ad9c11654853aec3ee1b1c2eabb2.jpg","alt":"","path":"ec73ad9c11654853aec3ee1b1c2eabb2.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/451784a47c426e10b95a328188a3c432.jpg","alt":"","path":"451784a47c426e10b95a328188a3c432.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/3159759df3b4860f11b76aa100790928.jpg","alt":"","path":"3159759df3b4860f11b76aa100790928.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/d3ea194146e1d4e42bf77c29f0f36141.jpg","alt":"","path":"d3ea194146e1d4e42bf77c29f0f36141.jpg","width":2000,"height":2500}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"57d3ee02-37c0-472f-987e-e0e59d80c413","title":"Country Farms Super Greens - Berry","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"17","inventory_tracking":true,"published":true,"handle":"country-farms-super-greens-berry","spu":"","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\/country-farms-super-greens-berry","sales":"1","image":{"src":"\/\/img.staticdj.com\/69ed0270961ce142c4f91d72b101e9b4.jpg","alt":"","path":"69ed0270961ce142c4f91d72b101e9b4.jpg","width":2000,"height":2000},"variants":[{"id":"765ce5ea-107c-4300-8d7e-1f31397d42f3","title":"","weight_unit":"kg","inventory_quantity":"17","sku":"77032","barcode":"035046090593","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"19.99","price":"19.99","available":true,"url":"\/products\/country-farms-super-greens-berry?variant=765ce5ea-107c-4300-8d7e-1f31397d42f3","available_quantity":"17","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/69ed0270961ce142c4f91d72b101e9b4.jpg","alt":"","path":"69ed0270961ce142c4f91d72b101e9b4.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/aaa0d675c05c6a35b5efb4199e90026e.jpg","alt":"","path":"aaa0d675c05c6a35b5efb4199e90026e.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/f54c797b950e2b58447f8a3ffab07051.jpg","alt":"","path":"f54c797b950e2b58447f8a3ffab07051.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/765c2e6a8d80fdb08bc287c914035ace.jpg","alt":"","path":"765c2e6a8d80fdb08bc287c914035ace.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/413ede16b68778d578fbb690c6f9894b.jpg","alt":"","path":"413ede16b68778d578fbb690c6f9894b.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"c0e1cec4-64bb-410c-97f1-b6106e4bf31f","title":"Country Farms Super Greens - Unflavored","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"19","inventory_tracking":true,"published":true,"handle":"country-farms-super-greens-unflavored","spu":"","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\/country-farms-super-greens-unflavored","sales":"0","image":{"src":"\/\/img.staticdj.com\/a8da37e91d105067419ca59c14902dcc.jpg","alt":"","path":"a8da37e91d105067419ca59c14902dcc.jpg","width":2000,"height":2000},"variants":[{"id":"2a9b56c0-61f7-461a-a0a4-1fc49c8ad66c","title":"","weight_unit":"kg","inventory_quantity":"19","sku":"77033","barcode":"035046092672","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"19.99","price":"19.99","available":true,"url":"\/products\/country-farms-super-greens-unflavored?variant=2a9b56c0-61f7-461a-a0a4-1fc49c8ad66c","available_quantity":"19","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/a8da37e91d105067419ca59c14902dcc.jpg","alt":"","path":"a8da37e91d105067419ca59c14902dcc.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/e9a131a9061aa4f344a364beab0f5d2e.jpg","alt":"","path":"e9a131a9061aa4f344a364beab0f5d2e.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/33f43e3be476d217490a3576f3fdb56b.jpg","alt":"","path":"33f43e3be476d217490a3576f3fdb56b.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/2f3faa32b1352dcfc33e56cc9a82c0c7.jpg","alt":"","path":"2f3faa32b1352dcfc33e56cc9a82c0c7.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4de9be9e88433d861e93b1a402b4b6d9.jpg","alt":"","path":"4de9be9e88433d861e93b1a402b4b6d9.jpg","width":2000,"height":2500}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"2ba39230-8d91-41ad-a61d-9c3d322bf29e","title":"Debloat","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"20","inventory_tracking":true,"published":true,"handle":"debloat","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"17.99","price_max":"17.99","price":"17.99","compare_at_price":"17.99","compare_at_price_min":"17.99","compare_at_price_max":"17.99","url":"\/products\/debloat","sales":"0","image":{"src":"\/\/img.staticdj.com\/b15163813b0638c9cbb24aec647d46e9.jpg","alt":"","path":"b15163813b0638c9cbb24aec647d46e9.jpg","width":2000,"height":2000},"variants":[{"id":"dd1a8a73-fb91-4682-98c1-18702f382697","title":"","weight_unit":"kg","inventory_quantity":"20","sku":"77035","barcode":"035046128845","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"17.99","price":"17.99","available":true,"url":"\/products\/debloat?variant=dd1a8a73-fb91-4682-98c1-18702f382697","available_quantity":"20","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/b15163813b0638c9cbb24aec647d46e9.jpg","alt":"","path":"b15163813b0638c9cbb24aec647d46e9.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/2a8dcc7569caa074dfef551e3dbbf648.jpg","alt":"","path":"2a8dcc7569caa074dfef551e3dbbf648.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/5eddbf3ac314386ed45148d53cf8a0c9.jpg","alt":"","path":"5eddbf3ac314386ed45148d53cf8a0c9.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/19cccf237dbd2e252a14881485c3cd83.jpg","alt":"","path":"19cccf237dbd2e252a14881485c3cd83.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a06625b3e849229867d4301d2e7ffc74.jpg","alt":"","path":"a06625b3e849229867d4301d2e7ffc74.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"a62963c2-1324-4c58-9c9d-861fc58bcd9a","title":"Her Own Pms Mood & Relief Vitamin","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"13","inventory_tracking":true,"published":true,"handle":"her-own-pms-mood-relief-vitamin","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"17.99","price_max":"17.99","price":"17.99","compare_at_price":"17.99","compare_at_price_min":"17.99","compare_at_price_max":"17.99","url":"\/products\/her-own-pms-mood-relief-vitamin","sales":"2","image":{"src":"\/\/img.staticdj.com\/29ce7d357746a36f864187fc56661614.jpg","alt":"","path":"29ce7d357746a36f864187fc56661614.jpg","width":2000,"height":2000},"variants":[{"id":"fa5f3a94-bfb2-4a28-8a22-ad006526e9f0","title":"","weight_unit":"kg","inventory_quantity":"13","sku":"77034","barcode":"035046129255","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"17.99","price":"17.99","available":true,"url":"\/products\/her-own-pms-mood-relief-vitamin?variant=fa5f3a94-bfb2-4a28-8a22-ad006526e9f0","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\/29ce7d357746a36f864187fc56661614.jpg","alt":"","path":"29ce7d357746a36f864187fc56661614.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a00068689299a949ce1e30d3c7c8f540.jpg","alt":"","path":"a00068689299a949ce1e30d3c7c8f540.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/37fb45eebe630fce3b6f6b368eeadc5b.jpg","alt":"","path":"37fb45eebe630fce3b6f6b368eeadc5b.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/0f3ea9d2214d49f0d2ccd5fd9d8dbc59.jpg","alt":"","path":"0f3ea9d2214d49f0d2ccd5fd9d8dbc59.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/1097bde0b3d402cce509064616b5c0f7.jpg","alt":"","path":"1097bde0b3d402cce509064616b5c0f7.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"2590f247-d9f8-4387-9b17-d9078f04293d","title":"Her Own Probiotic Capsules, 30 Ct","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"33","inventory_tracking":true,"published":true,"handle":"her-own-probiotic-capsules-30-ct","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"17.99","price_max":"17.99","price":"17.99","compare_at_price":"17.99","compare_at_price_min":"17.99","compare_at_price_max":"17.99","url":"\/products\/her-own-probiotic-capsules-30-ct","sales":"4","image":{"src":"\/\/img.staticdj.com\/786aa81ca09cd1ea624a9af3c888ceac.jpg","alt":"","path":"786aa81ca09cd1ea624a9af3c888ceac.jpg","width":2000,"height":2000},"variants":[{"id":"91a708ae-7d2b-4bf9-9f7d-f2014a684ff8","title":"","weight_unit":"kg","inventory_quantity":"33","sku":"77036","barcode":"035046128081","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"17.99","price":"17.99","available":true,"url":"\/products\/her-own-probiotic-capsules-30-ct?variant=91a708ae-7d2b-4bf9-9f7d-f2014a684ff8","available_quantity":"33","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/786aa81ca09cd1ea624a9af3c888ceac.jpg","alt":"","path":"786aa81ca09cd1ea624a9af3c888ceac.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/dfd537869a2996871c40b66d2b953777.jpg","alt":"","path":"dfd537869a2996871c40b66d2b953777.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/146a5ee2c8ed520ce60e14460f2fd2db.jpg","alt":"","path":"146a5ee2c8ed520ce60e14460f2fd2db.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/c3eb9bf6301477fb6d0ffae3e4656183.jpg","alt":"","path":"c3eb9bf6301477fb6d0ffae3e4656183.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/6ad98b5f0cacd740bcaa63e28866c23b.jpg","alt":"","path":"6ad98b5f0cacd740bcaa63e28866c23b.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"8647b3f9-9972-4ba8-95c4-c3d86c7931b0","title":"Her Own Probiotic Gummies, 60 Ct","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"17","inventory_tracking":true,"published":true,"handle":"her-own-probiotic-gummies-60-ct","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"17.99","price_max":"17.99","price":"17.99","compare_at_price":"17.99","compare_at_price_min":"17.99","compare_at_price_max":"17.99","url":"\/products\/her-own-probiotic-gummies-60-ct","sales":"1","image":{"src":"\/\/img.staticdj.com\/32fecc616c634bcfd10b4b575a514d1c.jpg","alt":"","path":"32fecc616c634bcfd10b4b575a514d1c.jpg","width":2000,"height":2000},"variants":[{"id":"6c0886b7-0a44-4804-b4aa-d6b1ace6ac24","title":"","weight_unit":"kg","inventory_quantity":"17","sku":"77037","barcode":"035046128838","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"17.99","price":"17.99","available":true,"url":"\/products\/her-own-probiotic-gummies-60-ct?variant=6c0886b7-0a44-4804-b4aa-d6b1ace6ac24","available_quantity":"17","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/32fecc616c634bcfd10b4b575a514d1c.jpg","alt":"","path":"32fecc616c634bcfd10b4b575a514d1c.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/c17daa8c47b1b836295234b870b28e09.jpg","alt":"","path":"c17daa8c47b1b836295234b870b28e09.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/5af54d3bed1b7ad9fe2ec551e5ddc5c4.jpg","alt":"","path":"5af54d3bed1b7ad9fe2ec551e5ddc5c4.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/e0516bd642b394843eda0912ac78f0e3.jpg","alt":"","path":"e0516bd642b394843eda0912ac78f0e3.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/ebb2eb29f7a517073a1b761b070f410a.jpg","alt":"","path":"ebb2eb29f7a517073a1b761b070f410a.jpg","width":2000,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"9d51d976-20cb-452e-9cb2-fafc330f9244","title":"Keto Burner Dual Action Capsules","brief":"","vendor":"WINDMILL 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":"keto-burner-dual-action-capsules","spu":"44298","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"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\/keto-burner-dual-action-capsules","sales":"8","image":{"src":"\/\/img.staticdj.com\/d3463e605a4c6dda8f964e81c822689d.jpeg","alt":"","path":"d3463e605a4c6dda8f964e81c822689d.jpeg","width":2000,"height":2000},"variants":[{"id":"ff0c54db-86b4-4b7a-8286-1f47257af624","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"44298","barcode":"035046111489","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"19.99","price":"19.99","available":false,"url":"\/products\/keto-burner-dual-action-capsules?variant=ff0c54db-86b4-4b7a-8286-1f47257af624","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\/d3463e605a4c6dda8f964e81c822689d.jpeg","alt":"","path":"d3463e605a4c6dda8f964e81c822689d.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/f89223730f5c99e3817b0033eecdcc11.jpeg","alt":"","path":"f89223730f5c99e3817b0033eecdcc11.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/66d3d184a103c91d8d68dde9c703264c.jpeg","alt":"","path":"66d3d184a103c91d8d68dde9c703264c.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/b995144279b415c76bb95038e598aa87.jpeg","alt":"","path":"b995144279b415c76bb95038e598aa87.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/cccf847af956d6093db0180cdd2ff94c.jpeg","alt":"","path":"cccf847af956d6093db0180cdd2ff94c.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/0f2a693f1f85247f90effa2a0a94c485.jpeg","alt":"","path":"0f2a693f1f85247f90effa2a0a94c485.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/890331d9366155c92249872268286ac1.jpeg","alt":"","path":"890331d9366155c92249872268286ac1.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"134ce9bb-efda-4934-8f68-29b227bf3e2b","title":"Ketogenic Meal Shake Chocolate Cream","brief":"","vendor":"WINDMILL 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":"ketogenic-meal-shake-chocolate-cream","spu":"44294","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\/ketogenic-meal-shake-chocolate-cream","sales":"1","image":{"src":"\/\/img.staticdj.com\/ad0a1282c818d28ed4b387b6f34c80b1.jpeg","alt":"","path":"ad0a1282c818d28ed4b387b6f34c80b1.jpeg","width":2000,"height":2000},"variants":[{"id":"8638e4f5-4287-4e0a-a9d9-ba17cfc0eb2b","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"44294","barcode":"035046107420","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/ketogenic-meal-shake-chocolate-cream?variant=8638e4f5-4287-4e0a-a9d9-ba17cfc0eb2b","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\/ad0a1282c818d28ed4b387b6f34c80b1.jpeg","alt":"","path":"ad0a1282c818d28ed4b387b6f34c80b1.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/be11253b53f27f5dff7e8982eddeb709.jpeg","alt":"","path":"be11253b53f27f5dff7e8982eddeb709.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/8f8d181dbd5b02f774ed3734f89d7479.jpeg","alt":"","path":"8f8d181dbd5b02f774ed3734f89d7479.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"08478bc5-722a-49f2-a0ea-42fc23209565","title":"Ketogenic Meal Shake Vanilla Cream","brief":"","vendor":"WINDMILL 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":"ketogenic-meal-shake-vanilla-cream","spu":"44292","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\/ketogenic-meal-shake-vanilla-cream","sales":"1","image":{"src":"\/\/img.staticdj.com\/7e143b7b3fba53b2e4865a1aa237c2c7.jpeg","alt":"","path":"7e143b7b3fba53b2e4865a1aa237c2c7.jpeg","width":2000,"height":2000},"variants":[{"id":"03043085-2ebf-4698-9124-06c68fa7dc16","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"44292","barcode":"035046107437","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"29.99","price":"29.99","available":false,"url":"\/products\/ketogenic-meal-shake-vanilla-cream?variant=03043085-2ebf-4698-9124-06c68fa7dc16","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\/7e143b7b3fba53b2e4865a1aa237c2c7.jpeg","alt":"","path":"7e143b7b3fba53b2e4865a1aa237c2c7.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/d33c251c239d58f23e32e3e0e1e1a335.jpeg","alt":"","path":"d33c251c239d58f23e32e3e0e1e1a335.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/5f0e359421e8ed6ac34bfdb2a64f946e.jpeg","alt":"","path":"5f0e359421e8ed6ac34bfdb2a64f946e.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"6e40a989-56a7-4987-8576-f83839e35066","title":"RapidFire\u2122 MCT Coconut Oil","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":true,"inventory_policy":"deny","inventory_quantity":"19","inventory_tracking":true,"published":true,"handle":"rapidfire-mct-coconut-oil","spu":"42924","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":true,"price_min":"14.99","price_max":"14.99","price":"14.99","compare_at_price":"14.99","compare_at_price_min":"14.99","compare_at_price_max":"14.99","url":"\/products\/rapidfire-mct-coconut-oil","sales":"4","image":{"src":"\/\/img.staticdj.com\/e68c48628a5c937c75db12c44e4c6c22.jpeg","alt":"","path":"e68c48628a5c937c75db12c44e4c6c22.jpeg","width":2000,"height":2000},"variants":[{"id":"6cb01f5a-c341-4482-a090-a42e5401117e","title":"","weight_unit":"lb","inventory_quantity":"19","sku":"42924","barcode":"035046102548","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"14.99","price":"14.99","available":true,"url":"\/products\/rapidfire-mct-coconut-oil?variant=6cb01f5a-c341-4482-a090-a42e5401117e","available_quantity":"19","options":[],"is_hit_discount":true,"discount_info":{"total_price":"","total_received_discounts":"","discount_min_purchase_qty":0}}],"images":[{"src":"\/\/img.staticdj.com\/e68c48628a5c937c75db12c44e4c6c22.jpeg","alt":"","path":"e68c48628a5c937c75db12c44e4c6c22.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4c4094d100234133c052337e1e897815.jpeg","alt":"","path":"4c4094d100234133c052337e1e897815.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/7f208dd83b555841d81b2a31194e372b.jpeg","alt":"","path":"7f208dd83b555841d81b2a31194e372b.jpeg","width":2000,"height":2000}],"options":[],"product_type":"","discount_min_purchase_qty":0},{"id":"cce085ac-8343-4b63-978c-cc51e27a334c","title":"Super Fruits & Veggies","brief":"","vendor":"WINDMILL HEALTH PRODUCTS","vendor_url":"","has_only_default_variant":true,"requires_shipping":true,"taxable":false,"inventory_policy":"deny","inventory_quantity":"0","inventory_tracking":true,"published":true,"handle":"super-fruits-veggies","spu":"","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"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\/super-fruits-veggies","sales":"0","image":{"src":"\/\/img.staticdj.com\/498dddb5a50e8915676817759069d2ff.jpg","alt":"","path":"498dddb5a50e8915676817759069d2ff.jpg","width":2000,"height":2000},"variants":[{"id":"26f17601-1149-4278-a611-be4f86265421","title":"","weight_unit":"kg","inventory_quantity":"0","sku":"77038","barcode":"035046131166","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"19.99","price":"19.99","available":false,"url":"\/products\/super-fruits-veggies?variant=26f17601-1149-4278-a611-be4f86265421","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\/498dddb5a50e8915676817759069d2ff.jpg","alt":"","path":"498dddb5a50e8915676817759069d2ff.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/f5acdc091e0f73193f09d805c47de38f.jpg","alt":"","path":"f5acdc091e0f73193f09d805c47de38f.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/4b190b52727b0a5fa922929684c94ca0.jpg","alt":"","path":"4b190b52727b0a5fa922929684c94ca0.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/144e64f0e81872f46209f03993f00b82.jpg","alt":"","path":"144e64f0e81872f46209f03993f00b82.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/30a78650cd6f2efbbd434181a8107da6.jpg","alt":"","path":"30a78650cd6f2efbbd434181a8107da6.jpg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/c5f09f4cc9850f87637b732b3cccc62d.jpg","alt":"","path":"c5f09f4cc9850f87637b732b3cccc62d.jpg","width":3233,"height":2000}],"options":[],"product_type":"default","discount_min_purchase_qty":0},{"id":"156f01c6-0191-4263-919f-f45c2f14a3f9","title":"VitaDrops Energy B12 Jellybeans 90ct","brief":"","vendor":"WINDMILL 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":"vitadrops-energy-b12-jellybeans-90ct","spu":"99674","note":"","need_variant_image":false,"fake_sales":"0","display_fake_sales":false,"independent_seo":false,"available":false,"price_min":"13.99","price_max":"13.99","price":"13.99","compare_at_price":"13.99","compare_at_price_min":"13.99","compare_at_price_max":"13.99","url":"\/products\/vitadrops-energy-b12-jellybeans-90ct","sales":"11","image":{"src":"\/\/img.staticdj.com\/25f99f6b5e57d49d7a83918c5927197f.jpeg","alt":"","path":"25f99f6b5e57d49d7a83918c5927197f.jpeg","width":2000,"height":2000},"variants":[{"id":"c6edb2d7-6a62-4f0c-8af0-9daf0a18529f","title":"","weight_unit":"lb","inventory_quantity":"0","sku":"99674","barcode":"035046132057","position":1,"option1":"","option2":"","option3":"","note":"","image":null,"weight":"0.00","compare_at_price":"13.99","price":"13.99","available":false,"url":"\/products\/vitadrops-energy-b12-jellybeans-90ct?variant=c6edb2d7-6a62-4f0c-8af0-9daf0a18529f","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\/25f99f6b5e57d49d7a83918c5927197f.jpeg","alt":"","path":"25f99f6b5e57d49d7a83918c5927197f.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/9f46f37114bd754e600afa87aa35be9f.jpeg","alt":"","path":"9f46f37114bd754e600afa87aa35be9f.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/a065a0d9c9a8c218125607980a167a52.jpeg","alt":"","path":"a065a0d9c9a8c218125607980a167a52.jpeg","width":2000,"height":2000},{"src":"\/\/img.staticdj.com\/323709802f36796bbb052ab00244a0a6.jpeg","alt":"","path":"323709802f36796bbb052ab00244a0a6.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":15,"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)
15
Sort by