/* * Name: skuTable * Author: cshaptx4869 * Project: https://github.com/cshaptx4869/skuTable */ layui.define(['jquery', 'form', 'upload', 'layer', 'sortable'], function (exports) { "use strict"; var $ = layui.jquery, form = layui.form, upload = layui.upload, layer = layui.layer, sortable = layui.sortable, MOD_NAME = 'skuTable'; //工具类 class Util { static config = { shade: [0.02, '#000'], time: 2000 }; static msg = { // 成功消息 success: function (msg, callback = null) { return layer.msg(msg, { icon: 1, shade: Util.config.shade, scrollbar: false, time: Util.config.time, shadeClose: true }, callback); }, // 失败消息 error: function (msg, callback = null) { return layer.msg(msg, { icon: 2, shade: Util.config.shade, scrollbar: false, time: Util.config.time, shadeClose: true }, callback); }, // 警告消息框 alert: function (msg, callback = null) { return layer.alert(msg, {end: callback, scrollbar: false}); }, // 对话框 confirm: function (msg, ok, no) { var index = layer.confirm(msg, {title: '操作确认', btn: ['确认', '取消']}, function () { typeof ok === 'function' && ok.call(this); }, function () { typeof no === 'function' && no.call(this); Util.msg.close(index); }); return index; }, // 消息提示 tips: function (msg, callback = null) { return layer.msg(msg, { time: Util.config.time, shade: Util.config.shade, end: callback, shadeClose: true }); }, // 加载中提示 loading: function (msg, callback = null) { return msg ? layer.msg(msg, { icon: 16, scrollbar: false, shade: Util.config.shade, time: 0, end: callback }) : layer.load(2, {time: 0, scrollbar: false, shade: Util.config.shade, end: callback}); }, // 输入框 prompt: function (option, callback = null) { return layer.prompt(option, callback); }, // 关闭消息框 close: function (index) { return layer.close(index); } }; static request = { post: function (option, ok, no, ex) { return Util.request.ajax('post', option, ok, no, ex); }, get: function (option, ok, no, ex) { return Util.request.ajax('get', option, ok, no, ex); }, ajax: function (type, option, ok, no, ex) { type = type || 'get'; option.url = option.url || ''; option.data = option.data || {}; option.statusName = option.statusName || 'code'; option.statusCode = option.statusCode || 200; ok = ok || function (res) { }; no = no || function (res) { var msg = res.msg == undefined ? '返回数据格式有误' : res.msg; Util.msg.error(msg); return false; }; ex = ex || function (res) { }; if (option.url == '') { Util.msg.error('请求地址不能为空'); return false; } //var index = Util.msg.loading('加载中'); $.ajax({ url: option.url, type: type, contentType: "application/x-www-form-urlencoded; charset=UTF-8", dataType: "json", data: option.data, timeout: 60000, success: function (res) { //Util.msg.close(index); /* if (res[option.statusName] == option.statusCode) { return ok(res); } else { return no(res); }*/ return ok(res); }, error: function (xhr, textstatus, thrown) { Util.msg.error('Status:' + xhr.status + ',' + xhr.statusText + ',请稍后再试!', function () { ex(xhr); }); return false; } }); } }; static tool = { uuid: function uuid(randomLength = 8) { return Number(Math.random().toString().substr(2, randomLength) + Date.now()).toString(36) } } } class SkuTable { options = { show : 1,//显示的选项,1是全显示,2是只显示单规格,3是只显示多规格 isAttributeValue: 1, //规格类型 0统一规格 1多规格 isAttributeElemId: 'fairy-is-attribute', //规格类型容器id specTableElemId: 'fairy-spec-table', //规格表容器id skuTableElemId: 'fairy-sku-table', //SKU表容器id rowspan: false, //是否开启SKU行合并, sortable: false, //规格拖拽排序 skuIcon: '', uploadUrl: '', requestSuccessCode: 1, //请求成功返回状态码值 specDataDelete: false, //开启规格删除 productId: '', //商品id 配合specDataUrl和skuDataUrl使用 specData: [], //规格数据 specDataUrl: '', //优先级大于specData skuData: {}, //SKU数据 skuDataUrl: '', //优先级大于skuDataUrl skuNameType: 0, skuNameDelimiter: '-', //统一规格配置项 singleSkuTableConfig: { thead: [ {title: '销售价(元)', icon: 'layui-icon-cols'}, {title: '市场价(元)', icon: 'layui-icon-cols'}, {title: '成本价(元)', icon: 'layui-icon-cols'}, {title: '库存', icon: 'layui-icon-cols'}, {title: '状态', icon: ''}, ], tbody: [ {type: 'input', field: 'price', value: '', verify: 'required|number', reqtext: '销售价不能为空'}, {type: 'input', field: 'market_price', value: '0', verify: 'required|number', reqtext: '市场价不能为空'}, {type: 'input', field: 'cost_price', value: '0', verify: 'required|number', reqtext: '成本价不能为空'}, {type: 'input', field: 'stock', value: '0', verify: 'required|number', reqtext: '库存不能为空'}, {type: 'select', field: 'status', option: [{key: '启用', value: '1'}, {key: '禁用', value: '0'}], verify: 'required', reqtext: '状态不能为空'}, ] }, //多规格配置项 multipleSkuTableConfig: { thead: [ {title: '图片', icon: ''}, {title: '销售价(元)', icon: 'layui-icon-cols'}, {title: '市场价(元)', icon: 'layui-icon-cols'}, {title: '成本价(元)', icon: 'layui-icon-cols'}, {title: '库存', icon: 'layui-icon-cols'}, {title: '状态', icon: ''}, ], tbody: [ {type: 'image', field: 'picture', value: '', verify: '', reqtext: ''}, {type: 'input', field: 'price', value: '', verify: 'required|number', reqtext: '销售价不能为空'}, {type: 'input', field: 'market_price', value: '0', verify: 'required|number', reqtext: '市场价不能为空'}, {type: 'input', field: 'cost_price', value: '0', verify: 'required|number', reqtext: '成本价不能为空'}, {type: 'input', field: 'stock', value: '0', verify: 'required|number', reqtext: '库存不能为空'}, { type: 'select', field: 'status', option: [{key: '启用', value: '1'}, {key: '禁用', value: '0'}], verify: '', reqtext: '' }, ] } }; constructor(options) { this.options = $.extend(this.options, options); if (this.options.skuDataUrl && this.options.productId) { Util.request.get({ url: this.options.skuDataUrl, data: { product_id: this.options.productId }, statusCode: this.options.requestSuccessCode }, (res) => { if (!res.data) { res.data = []; } this.options.skuData = res.data; this.css(); this.render(); this.listen(); }); } else { this.css(); this.render(); this.listen(); } } css() { $('head').append(`` ); } listen() { var that = this; /** * 监听规格类型选择 */ form.on('radio(fairy-is-attribute)', function (data) { that.options.isAttributeValue = data.value; that.render(); }); /** * 监听所选规格值的变化 */ form.on('checkbox(fairy-spec-filter)', function (data) { var specData = []; $.each($(`#${that.options.specTableElemId} tbody tr`), function () { var child = []; $.each($(this).find('input[type=checkbox]'), function () { child.push({id: $(this).val(), title: $(this).attr('title'), checked: $(this).is(':checked') ? '1' : '0'}); }); var specItem = { id: $(this).find('td').eq(0).data('spec-id'), title: $(this).find('td').eq(0).text(), child: child }; specData.push(specItem); }); that.options.specData = specData; //that.options.skuData = $.extend(that.options.skuData, that.getFormSkuData()); that.resetRender(that.options.skuTableElemId); that.renderMultipleSkuTable(); }); /** * 监听批量赋值 */ $(document).on('click', `#${this.options.skuTableElemId} thead tr th i`, function () { var thisI = this; Util.msg.prompt({title: $(thisI).parent().text().trim() + '批量赋值'}, function (value, index, elem) { $.each($(`#${that.options.skuTableElemId} tbody tr`), function () { var index = that.options.rowspan ? $(thisI).parent().index() - ($(`#${that.options.skuTableElemId} thead th.fairy-spec-name`).length - $(this).children('td.fairy-spec-value').length) : $(thisI).parent().index(); $(this).find('td').eq(index).children('input').val(value); }); Util.msg.close(index); }); }); /** * 监听添加规格 */ $(document).on('click', `#${this.options.specTableElemId} .fairy-spec-create`, function () { layer.prompt({title: '添加规格'}, function (value, index, elem) { var specTitleArr = []; $.each(that.options.specData, function (k, v) { specTitleArr.push(v.title) }) if (specTitleArr.includes(value)) { Util.msg.error('规格名已存在'); } else { that.options.specData.push({id: Util.tool.uuid(), title: value, child: []}); that.resetRender(that.options.specTableElemId); that.renderSpecTable(); } Util.msg.close(index); }); }); /** * 监听添加规格值 */ $(document).on('click', `#${this.options.specTableElemId} .fairy-spec-value-create`, function () { var specId = $(this).parent('td').prev().data('spec-id'); layer.prompt({title: '添加规格值'}, function (value, index, elem) { that.options.specData.forEach(function (v, i) { if (v.id == specId) { v.child.push({id: Util.tool.uuid(), title: value, checked: '0'}); } }); that.resetRender(that.options.specTableElemId); that.renderSpecTable(); Util.msg.close(index); }); }); /** * 监听删除规格/规格值 */ $(document).on('click', `#${this.options.specTableElemId} i.layui-icon-delete`, function () { if (typeof $(this).attr('data-spec-index') !== "undefined") { that.options.specData.splice($(this).data('spec-index'), 1); that.resetRender([that.options.specTableElemId, that.options.skuTableElemId]); that.renderSpecTable(); that.renderMultipleSkuTable(); } else if (typeof $(this).attr('data-spec-value-index') !== "undefined") { var [i, ii] = $(this).data('spec-value-index').split('-'); that.options.specData[i].child.splice(ii, 1); that.resetRender([that.options.specTableElemId, that.options.skuTableElemId]); that.renderSpecTable(); that.renderMultipleSkuTable(); } }); /** * 监听规格表是否开启删除 */ form.on('checkbox(fairy-spec-delete-filter)', function (data) { that.options.specDataDelete = data.elem.checked; if (data.elem.checked) { $(`#${that.options.specTableElemId} tbody tr i.layui-icon-delete`).removeClass('layui-hide'); } else { $(`#${that.options.specTableElemId} tbody tr i.layui-icon-delete`).addClass('layui-hide') } }); /** * 图片移入放大/移出恢复 */ var imgLayerIndex = null; $(document).on('mouseenter', '.fairy-sku-img', function () { imgLayerIndex = layer.tips('', this, { tips: [2, 'rgba(41,41,41,.5)'], time: 0 }); }).on('mouseleave', '.fairy-sku-img', function () { layer.close(imgLayerIndex); }) } /** * 渲染 */ render() { this.resetRender(); this.renderIsAttribute(this.options.isAttributeValue); if (this.options.isAttributeValue == '2') { if (this.options.specDataUrl && this.options.productId) { Util.request.get({ url: this.options.specDataUrl, productId: this.options.productId, statusCode: this.options.requestSuccessCode }, (res) => { if (!res.data) { res.data = []; } this.options.specData = res.data; this.renderSpecTable(); this.renderMultipleSkuTable(); }); } else { this.renderSpecTable(); this.renderMultipleSkuTable(); } } else { this.renderSingleSkuTable(); } } /** * 重新渲染 * @param targets */ resetRender(targets) { if (typeof targets === 'string') { $(`#${targets}`).parents('.dever_sku').replaceWith(`
`); } else if ($.isArray(targets) && targets.length) { targets.forEach((item) => { $(`#${item}`).parents('.dever_sku').replaceWith(``); }) } else { $(`#${this.options.isAttributeElemId}`).parents('.dever_sku').replaceWith(``); $(`#${this.options.specTableElemId}`).parents('.dever_sku').replaceWith(``); $(`#${this.options.skuTableElemId}`).parents('.dever_sku').replaceWith(``); } } /** * 渲染规格类型 * @param checkedValue */ renderIsAttribute(checkedValue) { var html = ''; if (this.options.show == 1 || this.options.show == 2) { html += ``; } if (this.options.show == 1 || this.options.show == 3) { html += ``; } this.renderFormItem('规格类型', html, this.options.isAttributeElemId); } renderSingleSkuTable() { var that = this, table = `${item.title} | `; }); table += '|
---|---|
'; table += `'; table += ' | '; break; case "input": default: table += ''; table += ``; table += ' | '; break; } }); table += '
规格名 | 规格值 |
---|---|
${item.title} | `; table += '';
if (item.child) {
$.each(item.child, function (key, value) {
table += ` `;
});
}
table += ' 添加规格值 '
table += ' | ';
table += '
';
table += ``;
table += ` 添加规格 `;
table += ' |
' + t + ' | '; }).join(''); this.options.multipleSkuTableConfig.thead.forEach(function (item) { theadTr += '' + item.title + (item.icon ? ' ' : '') + ' | '; }); theadTr += '||||
---|---|---|---|---|---|
' + t + ' | '; } else if (skuRowspanArr[i] === 1) { return '' + t + ' | '; } else { return ''; } } else { return '' + t + ' | '; } }).join(''); that.options.multipleSkuTableConfig.tbody.forEach(function (c) { switch (c.type) { case "image": var skuValue = that.getSkuData(that.options.skuData, item, c); tr += ''; break; case "select": var skuValue = that.getSkuData(that.options.skuData, item, c); tr += ' | '; break; case "input": default: var skuValue = that.getSkuData(that.options.skuData, item, c); tr += ' | '; break; } }); tr += ' |
请先选择规格值 |