|
@@ -1,1112 +0,0 @@
|
|
|
-<template>
|
|
|
- <el-container>
|
|
|
- <el-header>
|
|
|
- <el-tabs class="sa-tabs bg-#fff sa-m-t-10 z-999" v-model="activeTab" @tab-change="handleTabChange">
|
|
|
- <el-tab-pane name="basic">
|
|
|
- <template #label>
|
|
|
- <div class="sa-flex" :class="basicFormErrors ? 'is-error' : ''">
|
|
|
- 基本信息
|
|
|
- <el-icon v-if="basicFormErrors" class="sa-m-l-5">
|
|
|
- <WarningFilled />
|
|
|
- </el-icon>
|
|
|
- <el-icon v-if="basicSaved" class="sa-m-l-5 text-success">
|
|
|
- <CircleCheckFilled />
|
|
|
- </el-icon>
|
|
|
- </div>
|
|
|
- </template>
|
|
|
- </el-tab-pane>
|
|
|
- <el-tab-pane name="attributes" :disabled="!goodsId && !isEdit">
|
|
|
- <template #label>
|
|
|
- <div class="sa-flex" :class="attributesFormErrors ? 'is-error' : ''">
|
|
|
- 商品属性
|
|
|
- <el-icon v-if="attributesFormErrors" class="sa-m-l-5">
|
|
|
- <WarningFilled />
|
|
|
- </el-icon>
|
|
|
- <el-icon v-if="attributesSaved" class="sa-m-l-5 text-success">
|
|
|
- <CircleCheckFilled />
|
|
|
- </el-icon>
|
|
|
- <span v-if="!goodsId && !isEdit" class="tab-disabled-tip">(需先保存基本信息)</span>
|
|
|
- </div>
|
|
|
- </template>
|
|
|
- </el-tab-pane>
|
|
|
- </el-tabs>
|
|
|
- </el-header>
|
|
|
-
|
|
|
- <el-main class="sa-p-t-30">
|
|
|
- <!-- 基本信息Tab -->
|
|
|
- <div v-show="activeTab === 'basic'">
|
|
|
- <el-form ref="basicFormRef" :model="basicFormData" :rules="basicRules" label-width="120px">
|
|
|
- <el-row :gutter="40">
|
|
|
- <!-- 左侧表单 -->
|
|
|
- <el-col :span="14">
|
|
|
- <el-form-item label="商品分类" prop="cateId" required>
|
|
|
- <el-select v-model="basicFormData.cateId" placeholder="请选择商品分类" clearable style="width: 100%">
|
|
|
- <el-option v-for="category in categoryOptions" :key="category.id" :label="category.name"
|
|
|
- :value="category.id" />
|
|
|
- </el-select>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品名称" prop="storeName" required>
|
|
|
- <el-input v-model="basicFormData.storeName" placeholder="请填写商品名称(限100字符)" maxlength="100"
|
|
|
- show-word-limit />
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="副标题" prop="keyword">
|
|
|
- <el-input v-model="basicFormData.keyword" placeholder="请填写副标题(限50字符)" maxlength="50" show-word-limit />
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品品牌" prop="itemBrand" required>
|
|
|
- <el-input v-model="basicFormData.itemBrand" placeholder="请填写商品品牌" />
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品介绍" prop="storeInfo">
|
|
|
- <el-input v-model="basicFormData.storeInfo" type="textarea" :rows="4" placeholder="请填写商品介绍(限500字符)"
|
|
|
- maxlength="500" show-word-limit />
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="运费模板" prop="tempId">
|
|
|
- <div class="mt-1px">包邮</div>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品货号" prop="itemNumber" required>
|
|
|
- <el-input v-model="basicFormData.itemNumber" placeholder="请填写商品货号" />
|
|
|
- <div class="form-tip ml-10px">如果您不输入商品货号,系统将自动生成一个唯一的货号</div>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品售价" prop="price" required>
|
|
|
- <el-select v-model="basicFormData.price" placeholder="请选择商品售价" clearable>
|
|
|
- <el-option :value="300" label="300৳" />
|
|
|
- <el-option :value="500" label="500৳" />
|
|
|
- <el-option :value="1000" label="1000৳" />
|
|
|
- <el-option :value="2000" label="2000৳" />
|
|
|
- <el-option :value="3000" label="3000৳" />
|
|
|
- </el-select>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="市场价" prop="otPrice">
|
|
|
- <el-select v-model="basicFormData.otPrice" placeholder="请选择市场价" clearable>
|
|
|
- <el-option :value="300" label="300৳" />
|
|
|
- <el-option :value="500" label="500৳" />
|
|
|
- <el-option :value="1000" label="1000৳" />
|
|
|
- <el-option :value="2000" label="2000৳" />
|
|
|
- <el-option :value="3000" label="3000৳" />
|
|
|
- </el-select>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品库存" prop="stock" required>
|
|
|
- <el-input v-model="basicFormData.stock" placeholder="请输入商品库存" type="number" min="0" />
|
|
|
- <div class="form-tip ml-10px">该设置只对单品有效,当商品存在多规格货品时为不可编辑状态,库存数值取决于货品数量</div>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="库存预警值" prop="stockThreshold">
|
|
|
- <el-input v-model="basicFormData.stockThreshold" placeholder="请输入库存预警值" type="number" min="0" />
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品状态" prop="isShow" required>
|
|
|
- <el-radio-group v-model="basicFormData.isShow">
|
|
|
- <el-radio :label="1">上架</el-radio>
|
|
|
- <el-radio :label="0">下架</el-radio>
|
|
|
- </el-radio-group>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="商品供应商" prop="itemSupplier" required>
|
|
|
- <el-input v-model="basicFormData.itemSupplier" placeholder="请输入商品供应商" />
|
|
|
- </el-form-item>
|
|
|
- </el-col>
|
|
|
-
|
|
|
- <!-- 右侧图片上传 -->
|
|
|
- <el-col :span="10">
|
|
|
- <el-form-item label="商品主图" prop="image" required>
|
|
|
- <sa-upload-image v-model="basicFormData.image" :max-count="5" :accept="['jpg', 'jpeg', 'png']"
|
|
|
- :max-size="5" :direct-upload="true" :size="100" placeholder="上传商品主图" />
|
|
|
- <div class="form-tip">作用于商城列表、分享图片;建议尺寸:750*750 px</div>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="轮播图" prop="sliderImage">
|
|
|
- <sa-upload-image v-model="basicFormData.sliderImage" :max-count="5" :accept="['jpg', 'jpeg', 'png']"
|
|
|
- :max-size="5" :direct-upload="true" :size="100" placeholder="上传轮播图" />
|
|
|
- <div class="form-tip">作用于商品详情顶部轮播显示,轮播图可以拖拽调整顺序</div>
|
|
|
- </el-form-item>
|
|
|
-
|
|
|
- <el-form-item label="详情图" prop="flatPattern" required>
|
|
|
- <sa-upload-image v-model="basicFormData.flatPattern" :max-count="10" :accept="['jpg', 'jpeg', 'png']"
|
|
|
- :max-size="5" :direct-upload="true" :size="100" placeholder="上传详情图" />
|
|
|
- <div class="form-tip">详情图片,用于商品详情页展示</div>
|
|
|
- </el-form-item>
|
|
|
- </el-col>
|
|
|
- </el-row>
|
|
|
- </el-form>
|
|
|
- </div>
|
|
|
-
|
|
|
- <!-- 商品属性Tab -->
|
|
|
- <div v-show="activeTab === 'attributes'">
|
|
|
- <!-- 如果没有商品ID,显示提示 -->
|
|
|
- <div v-if="!goodsId && !isEdit" class="tab-placeholder">
|
|
|
- <el-empty description="请先保存基本信息后再编辑商品属性">
|
|
|
- <el-button type="primary" @click="activeTab = 'basic'"> 去保存基本信息 </el-button>
|
|
|
- </el-empty>
|
|
|
- </div>
|
|
|
-
|
|
|
- <!-- 有商品ID时显示属性表单 -->
|
|
|
- <div>
|
|
|
- <el-form ref="attributesFormRef" :model="attributesFormData" :rules="attributesRules" label-width="120px">
|
|
|
- <!-- 多规格设置 -->
|
|
|
- <el-card class="spec-card">
|
|
|
- <template #header>
|
|
|
- <div class="card-header">
|
|
|
- <span>商品规格设置</span>
|
|
|
- </div>
|
|
|
- </template>
|
|
|
-
|
|
|
- <!-- 操作 -->
|
|
|
- <div class="sku-wrap">
|
|
|
- <div class="sku" v-for="(s, k) in attributesFormData.skus" :key="k">
|
|
|
- <div class="sku-key sa-flex sa-row-between">
|
|
|
- <div class="sa-flex">
|
|
|
- <div class="sa-m-r-16">规格名称</div>
|
|
|
- <el-input v-model="s.name" placeholder="请输入规格名称" class="sku-key-input"
|
|
|
- @input="buildSkuPriceTable"></el-input>
|
|
|
- </div>
|
|
|
- <el-icon @click="deleteMainSku(k)" class="sku-key-icon">
|
|
|
- <CircleCloseFilled />
|
|
|
- </el-icon>
|
|
|
- </div>
|
|
|
- <div class="sku-value sa-flex sa-flex-wrap">
|
|
|
- <div class="sku-value-title sa-m-r-16 sa-m-b-16 sa-flex"> 规格值 </div>
|
|
|
- <div v-for="(sc, c) in s.children" :key="c" class="sku-value-box sa-m-b-16">
|
|
|
- <el-input v-model="sc.name" placeholder="请输入规格值" class="sku-value-input"
|
|
|
- @input="buildSkuPriceTable"></el-input>
|
|
|
- <el-icon @click="deleteChildrenSku(k, c)" class="sku-value-icon">
|
|
|
- <CircleCloseFilled />
|
|
|
- </el-icon>
|
|
|
- </div>
|
|
|
- <div @click="addChildrenSku(k)" class="sku-value-add sa-m-r-24 sa-m-b-16 sa-flex cursor-pointer">
|
|
|
- 添加规格值
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="sku-tools sa-flex">
|
|
|
- <el-button type="primary" class="add" @click="addMainSku">+ 添加规格</el-button>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-
|
|
|
- <!-- 批量设置 -->
|
|
|
- <div class="sa-m-t-20" v-if="attributesFormData.sku_prices.length > 0">
|
|
|
- <el-form-item label="批量设置" label-width="80px">
|
|
|
- <div class="sku sa-m-r-20" v-for="(item, index) in attributesFormData.skus" :key="index">
|
|
|
- <el-select v-model="item.batchId" placeholder="请选择规格" class="sa-w-150" clearable>
|
|
|
- <template v-for="(citem, cindex) in item.children">
|
|
|
- <el-option :key="cindex" :label="citem.name" :value="citem.temp_id"
|
|
|
- v-if="citem.temp_id && citem.name"></el-option>
|
|
|
- </template>
|
|
|
- </el-select>
|
|
|
- </div>
|
|
|
- <div class="warning-title" style="margin-left: 8px">
|
|
|
- 未选择规格默认为全选批量设置
|
|
|
- </div>
|
|
|
- </el-form-item>
|
|
|
- <div class="sa-flex sa-flex-wrap">
|
|
|
- <el-select v-model="allEditObj.price" placeholder="请选择售价(৳)" class="sa-w-200 sa-m-r-10 sa-m-b-10"
|
|
|
- clearable>
|
|
|
- <el-option :value="300" label="300৳" />
|
|
|
- <el-option :value="500" label="500৳" />
|
|
|
- <el-option :value="1000" label="1000৳" />
|
|
|
- <el-option :value="2000" label="2000৳" />
|
|
|
- <el-option :value="3000" label="3000৳" />
|
|
|
- </el-select>
|
|
|
- <el-input v-model="allEditObj.stock" placeholder="请输入库存(件)" class="sa-w-200 sa-m-r-10 sa-m-b-10">
|
|
|
- <template #prepend>库存(件)</template>
|
|
|
- </el-input>
|
|
|
- <el-input v-model="allEditObj.stock_warning" placeholder="请输入库存预警值(件)"
|
|
|
- class="sa-w-200 sa-m-r-10 sa-m-b-10">
|
|
|
- <template #prepend>库存预警值(件)</template>
|
|
|
- </el-input>
|
|
|
- <el-button type="primary" @click="batchEdit" class="sa-m-b-10">批量设置</el-button>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-
|
|
|
- <!-- 表格 -->
|
|
|
- <div class="sku-table-wrap sa-m-b-20">
|
|
|
- <table class="sku-table" rules="all">
|
|
|
- <thead>
|
|
|
- <tr>
|
|
|
- <template v-for="(item, i) in attributesFormData.skus" :key="i">
|
|
|
- <th v-if="item.children.length">{{ item.name }}</th>
|
|
|
- </template>
|
|
|
- <th>图片</th>
|
|
|
- <th><span class="required">*</span>销售价格(৳)</th>
|
|
|
- <th><span class="required">*</span>商品库存</th>
|
|
|
- <th>库存预警值</th>
|
|
|
- <th>SKU编码</th>
|
|
|
- <th>操作</th>
|
|
|
- </tr>
|
|
|
- </thead>
|
|
|
- <tbody>
|
|
|
- <tr v-for="(item, i) in attributesFormData.sku_prices" :key="i">
|
|
|
- <template v-for="(v, j) in item.goods_sku_text" :key="j">
|
|
|
- <td>
|
|
|
- <span class="th-center">{{ v }}</span>
|
|
|
- </td>
|
|
|
- </template>
|
|
|
- <td class="image">
|
|
|
- <sa-upload-image v-model="item.imageList" :max-count="1" :accept="['jpg', 'jpeg', 'png']"
|
|
|
- :max-size="5" :direct-upload="true" :size="30" :show-tip="false" :compact="true"
|
|
|
- placeholder="" />
|
|
|
- </td>
|
|
|
- <td>
|
|
|
- <el-select v-model="item.price" placeholder="选择价格" size="small"
|
|
|
- :class="{ 'is-error': !item.price || item.price <= 0 }">
|
|
|
- <el-option :value="300" label="300" />
|
|
|
- <el-option :value="500" label="500" />
|
|
|
- <el-option :value="1000" label="1000" />
|
|
|
- <el-option :value="2000" label="2000" />
|
|
|
- <el-option :value="3000" label="3000" />
|
|
|
- </el-select>
|
|
|
- </td>
|
|
|
- <td class="stock">
|
|
|
- <el-input v-model="item.stock" placeholder="请输入库存" size="small" type="number" :step="1" :min="0"
|
|
|
- :class="{ 'is-error': !item.stock || item.stock < 0 }"></el-input>
|
|
|
- </td>
|
|
|
- <td class="stock_warning">
|
|
|
- <el-input v-model="item.stock_warning" placeholder="请输入预警值" size="small" type="number" :step="1"
|
|
|
- :min="0"></el-input>
|
|
|
- </td>
|
|
|
- <td class="sn">
|
|
|
- <el-input v-model="item.sn" placeholder="请输入SKU编码" size="small"></el-input>
|
|
|
- </td>
|
|
|
- <td>
|
|
|
- <el-button type="danger" size="small" text @click="deleteSkuPrice(i)">删除</el-button>
|
|
|
- </td>
|
|
|
- </tr>
|
|
|
- </tbody>
|
|
|
- </table>
|
|
|
- </div>
|
|
|
- </el-card>
|
|
|
- </el-form>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </el-main>
|
|
|
-
|
|
|
- <!-- 统一操作按钮 -->
|
|
|
- <el-footer class="sa-footer--submit">
|
|
|
- <el-button @click="closeDialog" size="large">关闭</el-button>
|
|
|
-
|
|
|
- <!-- 基本信息Tab的按钮 -->
|
|
|
- <template v-if="activeTab === 'basic'">
|
|
|
- <el-button type="primary" @click="saveBasicInfo" :loading="savingStates.basic" size="large">
|
|
|
- 保存基本信息
|
|
|
- </el-button>
|
|
|
- </template>
|
|
|
-
|
|
|
- <!-- 商品属性Tab的按钮 -->
|
|
|
- <template v-else-if="activeTab === 'attributes'">
|
|
|
- <el-button type="primary" @click="saveAttributes" :loading="savingStates.attributes" size="large">
|
|
|
- 保存商品属性
|
|
|
- </el-button>
|
|
|
- </template>
|
|
|
-
|
|
|
- <!-- 保存全部按钮(始终显示) -->
|
|
|
- <el-button type="primary" plain @click="saveAll" :loading="savingStates.all" size="large">
|
|
|
- 保存全部
|
|
|
- </el-button>
|
|
|
- </el-footer>
|
|
|
- </el-container>
|
|
|
-</template>
|
|
|
-
|
|
|
-<script setup>
|
|
|
-import { onMounted, reactive, ref, computed, nextTick } from 'vue';
|
|
|
-import { WarningFilled, CircleCheckFilled, CircleCloseFilled } from '@element-plus/icons-vue';
|
|
|
-import { api } from '../goods.service';
|
|
|
-const emit = defineEmits(['modalCallBack']);
|
|
|
-const props = defineProps({
|
|
|
- modal: {
|
|
|
- type: Object,
|
|
|
- required: true,
|
|
|
- },
|
|
|
-});
|
|
|
-
|
|
|
-// 从modal参数中获取类型和ID
|
|
|
-const type = computed(() => props.modal?.params?.type || 'add');
|
|
|
-const goodsIdFromProps = computed(() => props.modal?.params?.id || null);
|
|
|
-
|
|
|
-// 响应式数据
|
|
|
-const activeTab = ref('basic');
|
|
|
-const goodsId = ref(goodsIdFromProps.value);
|
|
|
-const isEdit = computed(() => type.value === 'edit');
|
|
|
-
|
|
|
-// 保存状态
|
|
|
-const savingStates = reactive({
|
|
|
- basic: false,
|
|
|
- attributes: false,
|
|
|
- all: false,
|
|
|
-});
|
|
|
-
|
|
|
-// 保存成功状态
|
|
|
-const basicSaved = ref(false);
|
|
|
-const attributesSaved = ref(false);
|
|
|
-
|
|
|
-// 表单错误状态
|
|
|
-const basicFormErrors = ref(false);
|
|
|
-const attributesFormErrors = ref(false);
|
|
|
-
|
|
|
-// 表单引用
|
|
|
-const basicFormRef = ref(null);
|
|
|
-const attributesFormRef = ref(null);
|
|
|
-
|
|
|
-// 分类选项
|
|
|
-const categoryOptions = ref([]);
|
|
|
-
|
|
|
-// 基本信息表单数据
|
|
|
-const basicFormData = reactive({
|
|
|
- id: '',
|
|
|
- cateId: '',
|
|
|
- storeName: '',
|
|
|
- keyword: '',
|
|
|
- itemBrand: '',
|
|
|
- storeInfo: '',
|
|
|
- tempId: 1,
|
|
|
- itemNumber: '',
|
|
|
- price: '',
|
|
|
- otPrice: '',
|
|
|
- stock: '',
|
|
|
- stockThreshold: '',
|
|
|
- isShow: 1,
|
|
|
- itemSupplier: '',
|
|
|
- sort: 0,
|
|
|
- isHot: 0,
|
|
|
- isNew: 0,
|
|
|
- isBest: 0,
|
|
|
- isGood: 0,
|
|
|
- isBenefit: 0,
|
|
|
- isPostage: 1,
|
|
|
- cost: '',
|
|
|
- vipPrice: '',
|
|
|
- image: [],
|
|
|
- sliderImage: [],
|
|
|
- flatPattern: [],
|
|
|
-});
|
|
|
-
|
|
|
-// 商品属性表单数据
|
|
|
-const attributesFormData = reactive({
|
|
|
- specType: 1, // 规格 0单 1多,默认多规格
|
|
|
- skus: [
|
|
|
- {
|
|
|
- id: 0,
|
|
|
- temp_id: 1,
|
|
|
- name: '',
|
|
|
- batchId: '',
|
|
|
- pid: 0,
|
|
|
- children: [],
|
|
|
- },
|
|
|
- ],
|
|
|
- sku_prices: [],
|
|
|
-});
|
|
|
-
|
|
|
-// 基本信息验证规则
|
|
|
-const basicRules = {
|
|
|
- cateId: [{ required: true, message: '请选择商品分类', trigger: 'change' }],
|
|
|
- storeName: [{ required: true, message: '请填写商品名称', trigger: 'blur' }],
|
|
|
- itemBrand: [{ required: true, message: '请填写商品品牌', trigger: 'blur' }],
|
|
|
- itemNumber: [{ required: true, message: '请填写商品货号', trigger: 'blur' }],
|
|
|
- price: [{ required: true, message: '请填写商品售价', trigger: 'blur' }],
|
|
|
- stock: [{ required: true, message: '请填写商品库存', trigger: 'blur' }],
|
|
|
- isShow: [{ required: true, message: '请选择商品状态', trigger: 'change' }],
|
|
|
- itemSupplier: [{ required: true, message: '请填写商品供应商', trigger: 'blur' }],
|
|
|
- image: [{ required: true, message: '请上传商品主图', trigger: 'change' }],
|
|
|
- flatPattern: [{ required: true, message: '请上传商品详情图', trigger: 'change' }],
|
|
|
-};
|
|
|
-
|
|
|
-// 商品属性验证规则
|
|
|
-const attributesRules = {
|
|
|
- specType: [{ required: true, message: '请选择规格类型', trigger: 'change' }],
|
|
|
-};
|
|
|
-
|
|
|
-// 图片数组转换为逗号分隔字符串的函数
|
|
|
-const convertImagesToString = (imageArray) => {
|
|
|
- return Array.isArray(imageArray) ? imageArray.join(',') : '';
|
|
|
-};
|
|
|
-
|
|
|
-// 图片字符串转换为数组的函数
|
|
|
-const convertStringToImages = (imageString) => {
|
|
|
- if (!imageString) return [];
|
|
|
- return imageString.split(',').filter((img) => img.trim());
|
|
|
-};
|
|
|
-
|
|
|
-// Tab切换处理
|
|
|
-const handleTabChange = (tabName) => {
|
|
|
- if (tabName === 'attributes' && !goodsId.value && !isEdit.value) {
|
|
|
- nextTick(() => {
|
|
|
- ElMessage.info('提示:保存商品属性需要先保存基本信息');
|
|
|
- });
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 保存基本信息
|
|
|
-const saveBasicInfo = async () => {
|
|
|
- savingStates.basic = true;
|
|
|
- basicFormErrors.value = false;
|
|
|
-
|
|
|
- const valid = await basicFormRef.value?.validate().catch(() => false);
|
|
|
- if (!valid) {
|
|
|
- basicFormErrors.value = true;
|
|
|
- savingStates.basic = false;
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- const submitData = {
|
|
|
- ...basicFormData,
|
|
|
- image: convertImagesToString(basicFormData.image),
|
|
|
- sliderImage: convertImagesToString(basicFormData.sliderImage),
|
|
|
- flatPattern: convertImagesToString(basicFormData.flatPattern),
|
|
|
- };
|
|
|
-
|
|
|
- const { code, data } = isEdit.value
|
|
|
- ? await api.goods.edit(goodsId.value, submitData)
|
|
|
- : await api.goods.add(submitData);
|
|
|
-
|
|
|
- if (code === '200') {
|
|
|
- if (!isEdit.value) {
|
|
|
- goodsId.value = data.id;
|
|
|
- basicFormData.id = data.id;
|
|
|
- }
|
|
|
- basicSaved.value = true;
|
|
|
- savingStates.basic = false;
|
|
|
- return true;
|
|
|
- }
|
|
|
-
|
|
|
- basicFormErrors.value = true;
|
|
|
- savingStates.basic = false;
|
|
|
- return false;
|
|
|
-};
|
|
|
-
|
|
|
-// 保存商品属性(内部方法)
|
|
|
-const saveAttributesInternal = async () => {
|
|
|
- try {
|
|
|
- // 验证属性表单
|
|
|
- const valid = await attributesFormRef.value?.validate().catch(() => false);
|
|
|
- if (!valid) {
|
|
|
- attributesFormErrors.value = true;
|
|
|
- ElMessage.error('请完善商品属性');
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- // 验证SKU
|
|
|
- if (!validateSku()) {
|
|
|
- attributesFormErrors.value = true;
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- // 准备属性数据 - 转换为后端需要的格式
|
|
|
- const submitData = {
|
|
|
- goodsId: goodsId.value,
|
|
|
- specType: attributesFormData.specType,
|
|
|
- attrValue: generateAttrValueData(),
|
|
|
- attr: generateAttrData(),
|
|
|
- };
|
|
|
-
|
|
|
- // 这里调用属性保存接口(待实现)
|
|
|
- const { code, data } = await api.rule.add(submitData);
|
|
|
- console.log(code, data);
|
|
|
-
|
|
|
- // 临时模拟成功
|
|
|
- attributesSaved.value = true;
|
|
|
- ElMessage.success(t('message.goodsAttributeSaveSuccess'));
|
|
|
- return true;
|
|
|
- } catch (error) {
|
|
|
- attributesFormErrors.value = true;
|
|
|
- ElMessage.error('保存失败:' + error.message);
|
|
|
- return false;
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 保存商品属性(对外方法)
|
|
|
-const saveAttributes = async () => {
|
|
|
- try {
|
|
|
- savingStates.attributes = true;
|
|
|
- attributesFormErrors.value = false;
|
|
|
-
|
|
|
- // 检查依赖
|
|
|
- if (!goodsId.value && !isEdit.value) {
|
|
|
- const result = await ElMessageBox.confirm(
|
|
|
- '保存商品属性需要先保存基本信息,是否现在保存基本信息?',
|
|
|
- '提示',
|
|
|
- {
|
|
|
- confirmButtonText: '保存基本信息并继续',
|
|
|
- cancelButtonText: '取消',
|
|
|
- type: 'warning',
|
|
|
- },
|
|
|
- );
|
|
|
-
|
|
|
- if (result === 'confirm') {
|
|
|
- // 先保存基本信息
|
|
|
- const basicSaved = await saveBasicInfo();
|
|
|
-
|
|
|
- if (basicSaved) {
|
|
|
- // 基本信息保存成功后,保存属性
|
|
|
- await saveAttributesInternal();
|
|
|
- }
|
|
|
- }
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- // 直接保存属性
|
|
|
- await saveAttributesInternal();
|
|
|
- } finally {
|
|
|
- savingStates.attributes = false;
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 保存全部
|
|
|
-const saveAll = async () => {
|
|
|
- try {
|
|
|
- savingStates.all = true;
|
|
|
-
|
|
|
- // 1. 如果没有商品ID,先保存基本信息
|
|
|
- if (!goodsId.value && !isEdit.value) {
|
|
|
- const basicSaved = await saveBasicInfo();
|
|
|
- if (!basicSaved) return;
|
|
|
- }
|
|
|
-
|
|
|
- // 2. 如果属性有修改,保存属性
|
|
|
- await saveAttributesInternal();
|
|
|
-
|
|
|
- ElMessage.success(t('message.saveSuccess'));
|
|
|
- emit('modalCallBack', { event: 'confirm' });
|
|
|
- } catch (error) {
|
|
|
- ElMessage.error(t('message.saveFailed') + ':' + error.message);
|
|
|
- } finally {
|
|
|
- savingStates.all = false;
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 关闭对话框
|
|
|
-const closeDialog = () => {
|
|
|
- emit('modalCallBack', { event: 'close' });
|
|
|
-};
|
|
|
-
|
|
|
-// SKU相关方法
|
|
|
-const countId = ref(2);
|
|
|
-const childrenModal = [];
|
|
|
-const isResetSku = ref(0);
|
|
|
-
|
|
|
-// 批量操作相关变量
|
|
|
-const allEditObj = ref({
|
|
|
- price: 0,
|
|
|
- stock: 0,
|
|
|
- stock_warning: 0,
|
|
|
-});
|
|
|
-
|
|
|
-// 添加主规格
|
|
|
-const addMainSku = () => {
|
|
|
- attributesFormData.skus.push({
|
|
|
- id: 0,
|
|
|
- temp_id: countId.value++,
|
|
|
- name: '',
|
|
|
- batchId: '',
|
|
|
- pid: 0,
|
|
|
- children: [],
|
|
|
- });
|
|
|
- buildSkuPriceTable();
|
|
|
-};
|
|
|
-
|
|
|
-// 删除主规格
|
|
|
-const deleteMainSku = (k) => {
|
|
|
- let data = attributesFormData.skus[k];
|
|
|
-
|
|
|
- // 删除主规格
|
|
|
- attributesFormData.skus.splice(k, 1);
|
|
|
-
|
|
|
- // 如果当前删除的主规格存在子规格,则清空 skuPrice
|
|
|
- if (data.children.length > 0) {
|
|
|
- attributesFormData.sku_prices = [];
|
|
|
- isResetSku.value = 1;
|
|
|
- }
|
|
|
- buildSkuPriceTable();
|
|
|
-};
|
|
|
-
|
|
|
-// 添加子规格
|
|
|
-const addChildrenSku = (k) => {
|
|
|
- let isExist = false;
|
|
|
- attributesFormData.skus[k].children.forEach((e) => {
|
|
|
- if (e.name == childrenModal[k] && e.name != '') {
|
|
|
- isExist = true;
|
|
|
- }
|
|
|
- });
|
|
|
- if (isExist) {
|
|
|
- ElMessage.warning('子规格已存在');
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- attributesFormData.skus[k].children.push({
|
|
|
- id: 0,
|
|
|
- temp_id: countId.value++,
|
|
|
- name: childrenModal[k] || '',
|
|
|
- pid: attributesFormData.skus[k].id,
|
|
|
- });
|
|
|
- childrenModal[k] = '';
|
|
|
-
|
|
|
- // 如果是添加的第一个子规格,清空 skuPrice
|
|
|
- if (attributesFormData.skus[k].children.length == 1) {
|
|
|
- attributesFormData.sku_prices = [];
|
|
|
- isResetSku.value = 1;
|
|
|
- }
|
|
|
- buildSkuPriceTable();
|
|
|
-};
|
|
|
-
|
|
|
-// 删除子规格
|
|
|
-const deleteChildrenSku = (k, i) => {
|
|
|
- let data = attributesFormData.skus[k].children[i];
|
|
|
- attributesFormData.skus[k].children.splice(i, 1);
|
|
|
-
|
|
|
- // 查询 sku_prices 中包含被删除的子规格的项,然后移除
|
|
|
- let deleteArr = [];
|
|
|
- attributesFormData.sku_prices.forEach((item, index) => {
|
|
|
- item.goods_sku_text.forEach((e) => {
|
|
|
- if (e == data.name) {
|
|
|
- deleteArr.push(index);
|
|
|
- }
|
|
|
- });
|
|
|
- });
|
|
|
- deleteArr.sort(function (a, b) {
|
|
|
- return b - a;
|
|
|
- });
|
|
|
- // 移除有相关子规格的项
|
|
|
- deleteArr.forEach((idx) => {
|
|
|
- attributesFormData.sku_prices.splice(idx, 1);
|
|
|
- });
|
|
|
-
|
|
|
- // 当前规格项,所有子规格都被删除,清空 sku_prices
|
|
|
- if (attributesFormData.skus[k].children.length <= 0) {
|
|
|
- attributesFormData.sku_prices = [];
|
|
|
- isResetSku.value = 1;
|
|
|
- }
|
|
|
- buildSkuPriceTable();
|
|
|
-};
|
|
|
-
|
|
|
-// 组成新的规格
|
|
|
-const buildSkuPriceTable = () => {
|
|
|
- let arr = [];
|
|
|
- // 遍历sku子规格生成新数组,然后执行递归笛卡尔积
|
|
|
- attributesFormData.skus.forEach((s1) => {
|
|
|
- let children = s1.children;
|
|
|
- let childrenIdArray = [];
|
|
|
- if (children.length > 0) {
|
|
|
- children.forEach((s2) => {
|
|
|
- childrenIdArray.push(s2.temp_id);
|
|
|
- });
|
|
|
- // 如果 children 子规格数量为 0,则不渲染当前规格
|
|
|
- arr.push(childrenIdArray);
|
|
|
- }
|
|
|
- });
|
|
|
- recursionSku(arr, 0, []);
|
|
|
-};
|
|
|
-
|
|
|
-// 递归找笛卡尔规格集合
|
|
|
-const recursionSku = (arr, k, temp) => {
|
|
|
- if (k == arr.length && k != 0) {
|
|
|
- let tempDetail = [];
|
|
|
- let tempDetailIds = [];
|
|
|
- temp.forEach((item) => {
|
|
|
- for (let sku of attributesFormData.skus) {
|
|
|
- for (let child of sku.children) {
|
|
|
- if (item == child.temp_id) {
|
|
|
- tempDetail.push(child.name);
|
|
|
- tempDetailIds.push(child.temp_id);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- });
|
|
|
- let flag = false; // 默认添加新的
|
|
|
- for (let i = 0; i < attributesFormData.sku_prices.length; i++) {
|
|
|
- if (
|
|
|
- attributesFormData.sku_prices[i].goods_sku_temp_ids.join(',') == tempDetailIds.join(',')
|
|
|
- ) {
|
|
|
- flag = i;
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if (flag === false) {
|
|
|
- attributesFormData.sku_prices.push({
|
|
|
- id: 0,
|
|
|
- temp_id: attributesFormData.sku_prices.length + 1,
|
|
|
- goods_sku_ids: '',
|
|
|
- goods_id: 0,
|
|
|
- image: '',
|
|
|
- imageList: [],
|
|
|
- price: 0,
|
|
|
- stock: 0,
|
|
|
- stock_warning: 0,
|
|
|
- sn: '',
|
|
|
- goods_sku_text: tempDetail,
|
|
|
- goods_sku_temp_ids: tempDetailIds,
|
|
|
- });
|
|
|
- } else {
|
|
|
- attributesFormData.sku_prices[flag].goods_sku_text = tempDetail;
|
|
|
- attributesFormData.sku_prices[flag].goods_sku_temp_ids = tempDetailIds;
|
|
|
- }
|
|
|
- return;
|
|
|
- }
|
|
|
- if (arr.length) {
|
|
|
- for (let i = 0; i < arr[k].length; i++) {
|
|
|
- temp[k] = arr[k][i];
|
|
|
- recursionSku(arr, k + 1, temp);
|
|
|
- }
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 批量操作
|
|
|
-const batchEdit = () => {
|
|
|
- const batchIds = attributesFormData.skus
|
|
|
- .map((item) => item.batchId)
|
|
|
- .filter((item) => Boolean(item));
|
|
|
- attributesFormData.sku_prices.forEach((item) => {
|
|
|
- if (
|
|
|
- batchIds.length ? batchIds.every((citem) => item.goods_sku_temp_ids.includes(citem)) : true
|
|
|
- ) {
|
|
|
- const { price, stock, stock_warning } = allEditObj.value;
|
|
|
- if (price) item.price = price;
|
|
|
- if (stock) item.stock = stock;
|
|
|
- if (stock_warning) item.stock_warning = stock_warning;
|
|
|
- }
|
|
|
- });
|
|
|
-
|
|
|
- // 清空输入框
|
|
|
- allEditObj.value = {
|
|
|
- price: 0,
|
|
|
- stock: 0,
|
|
|
- stock_warning: 0,
|
|
|
- };
|
|
|
-
|
|
|
- // 清空选择的规格
|
|
|
- attributesFormData.skus.forEach((item) => {
|
|
|
- item.batchId = '';
|
|
|
- });
|
|
|
-
|
|
|
- ElMessage.success('批量设置成功');
|
|
|
-};
|
|
|
-
|
|
|
-// 删除规格组合
|
|
|
-const deleteSkuPrice = (index) => {
|
|
|
- attributesFormData.sku_prices.splice(index, 1);
|
|
|
- ElMessage.success(t('message.deleteSuccess'));
|
|
|
-};
|
|
|
-
|
|
|
-// SKU校验
|
|
|
-const validateSku = () => {
|
|
|
- if (attributesFormData.sku_prices.length === 0) {
|
|
|
- ElMessage.error('请先添加商品规格');
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- for (let i = 0; i < attributesFormData.sku_prices.length; i++) {
|
|
|
- const item = attributesFormData.sku_prices[i];
|
|
|
- if (!item.price || item.price <= 0) {
|
|
|
- ElMessage.error(`第${i + 1}个规格的销售价格不能为空且必须大于0`);
|
|
|
- return false;
|
|
|
- }
|
|
|
- if (item.stock === null || item.stock === undefined || item.stock < 0) {
|
|
|
- ElMessage.error(`第${i + 1}个规格的商品库存不能为空且不能小于0`);
|
|
|
- return false;
|
|
|
- }
|
|
|
- }
|
|
|
- return true;
|
|
|
-};
|
|
|
-
|
|
|
-// 生成后端需要的 attrValue 数据格式
|
|
|
-const generateAttrValueData = () => {
|
|
|
- return attributesFormData.sku_prices.map((item) => {
|
|
|
- // 构建规格属性对象,如 {"颜色": "红色", "尺寸": "S"}
|
|
|
- const attrObj = {};
|
|
|
- const attrValueObj = {};
|
|
|
-
|
|
|
- // 根据 goods_sku_text 和对应的规格名称构建属性对象
|
|
|
- item.goods_sku_text.forEach((value, index) => {
|
|
|
- const specName = attributesFormData.skus[index]?.name;
|
|
|
- if (specName) {
|
|
|
- attrObj[specName] = value;
|
|
|
- attrValueObj[specName] = value;
|
|
|
- }
|
|
|
- });
|
|
|
-
|
|
|
- return {
|
|
|
- image: item.imageList && item.imageList.length > 0 ? item.imageList[0] : '',
|
|
|
- price: item.price || '0',
|
|
|
- stock: item.stock || 0,
|
|
|
- barCode: item.sn || '',
|
|
|
- stock_warning: item.stock_warning || 0,
|
|
|
- attrValue: JSON.stringify(attrValueObj),
|
|
|
- ...attrObj, // 展开规格属性,如 "颜色": "红色", "尺寸": "S"
|
|
|
- id: 0,
|
|
|
- productId: 0,
|
|
|
- };
|
|
|
- });
|
|
|
-};
|
|
|
-
|
|
|
-// 生成后端需要的 attr 数据格式
|
|
|
-const generateAttrData = () => {
|
|
|
- return attributesFormData.skus
|
|
|
- .filter((sku) => sku.name && sku.children.length > 0)
|
|
|
- .map((sku) => ({
|
|
|
- attrName: sku.name,
|
|
|
- attrValues: sku.children.map((child) => child.name).join(','),
|
|
|
- }));
|
|
|
-};
|
|
|
-
|
|
|
-// 获取分类数据
|
|
|
-const getCategoryData = async () => {
|
|
|
- const { code, data } = await api.category.list({ size: 100 });
|
|
|
- code === '200' &&
|
|
|
- (categoryOptions.value = data.list.map((cat) => ({
|
|
|
- label: cat.name,
|
|
|
- value: cat.id,
|
|
|
- id: cat.id,
|
|
|
- name: cat.name,
|
|
|
- })));
|
|
|
-};
|
|
|
-
|
|
|
-// 加载商品详情
|
|
|
-const loadGoodsDetail = async () => {
|
|
|
- if (!goodsId.value) return;
|
|
|
-
|
|
|
- const { code, data } = await api.goods.detail(goodsId.value);
|
|
|
- if (code === '200') {
|
|
|
- // 转换图片字段
|
|
|
- data.image = convertStringToImages(data.image);
|
|
|
- data.sliderImage = convertStringToImages(data.sliderImage);
|
|
|
- data.flatPattern = convertStringToImages(data.flatPattern);
|
|
|
-
|
|
|
- Object.assign(basicFormData, data);
|
|
|
- basicSaved.value = true;
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 初始化
|
|
|
-const init = async () => {
|
|
|
- await getCategoryData();
|
|
|
-
|
|
|
- if (isEdit.value && goodsId.value) {
|
|
|
- await loadGoodsDetail();
|
|
|
- }
|
|
|
-};
|
|
|
-
|
|
|
-// 组件挂载
|
|
|
-onMounted(() => {
|
|
|
- init();
|
|
|
-});
|
|
|
-</script>
|
|
|
-
|
|
|
-<style scoped lang="scss">
|
|
|
-.goods-edit {
|
|
|
- height: 100%;
|
|
|
-
|
|
|
- .el-header {
|
|
|
- height: auto;
|
|
|
- padding: 0;
|
|
|
- }
|
|
|
-
|
|
|
- .el-main {
|
|
|
- padding: 20px;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-.tab-placeholder {
|
|
|
- display: flex;
|
|
|
- justify-content: center;
|
|
|
- align-items: center;
|
|
|
- min-height: 300px;
|
|
|
-}
|
|
|
-
|
|
|
-.tab-disabled-tip {
|
|
|
- font-size: 12px;
|
|
|
- color: var(--el-color-info);
|
|
|
- margin-left: 4px;
|
|
|
-}
|
|
|
-
|
|
|
-.is-error {
|
|
|
- color: var(--el-color-danger);
|
|
|
-}
|
|
|
-
|
|
|
-.text-success {
|
|
|
- color: var(--el-color-success);
|
|
|
-}
|
|
|
-
|
|
|
-.spec-card {
|
|
|
- .card-header {
|
|
|
- display: flex;
|
|
|
- justify-content: space-between;
|
|
|
- align-items: center;
|
|
|
- font-weight: 500;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-.sku-wrap {
|
|
|
- width: 100%;
|
|
|
- border: 1px solid #d9d9d9;
|
|
|
- padding: 8px;
|
|
|
- box-sizing: border-box;
|
|
|
-
|
|
|
- .sku {
|
|
|
- width: 100%;
|
|
|
- min-height: 100px;
|
|
|
-
|
|
|
- .sku-key {
|
|
|
- width: 100%;
|
|
|
- height: 40px;
|
|
|
- color: var(--sa-subtitle);
|
|
|
- padding: 0 16px;
|
|
|
- background: var(--sa-table-header-bg);
|
|
|
- font-size: 14px;
|
|
|
-
|
|
|
- .sku-key-input {
|
|
|
- width: 120px;
|
|
|
- }
|
|
|
-
|
|
|
- .sku-key-icon {
|
|
|
- color: var(--el-color-primary);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- .sku-value {
|
|
|
- padding: 12px 0 0 30px;
|
|
|
- font-size: 14px;
|
|
|
- color: var(--sa-subtitle);
|
|
|
-
|
|
|
- .sku-value-title {
|
|
|
- height: 32px;
|
|
|
- }
|
|
|
-
|
|
|
- .sku-value-box {
|
|
|
- position: relative;
|
|
|
- margin-right: 24px;
|
|
|
-
|
|
|
- .sku-value-input {
|
|
|
- width: 104px;
|
|
|
- }
|
|
|
-
|
|
|
- .sku-value-icon {
|
|
|
- position: absolute;
|
|
|
- right: -8px;
|
|
|
- top: -8px;
|
|
|
- width: 16px;
|
|
|
- height: 16px;
|
|
|
- color: var(--el-color-primary);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- .sku-value-add {
|
|
|
- width: 104px;
|
|
|
- height: 32px;
|
|
|
- font-size: 14px;
|
|
|
- color: var(--el-color-primary);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- .sku-tools {
|
|
|
- width: 100%;
|
|
|
- height: 40px;
|
|
|
- color: #434343;
|
|
|
- padding-left: 16px;
|
|
|
- background: var(--sa-table-header-bg);
|
|
|
- font-size: 12px;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-.sku-table-wrap {
|
|
|
- width: 100%;
|
|
|
- overflow: auto;
|
|
|
- margin-top: 16px;
|
|
|
-
|
|
|
- .sku-table {
|
|
|
- width: 100%;
|
|
|
- border: 1px solid var(--sa-border);
|
|
|
-
|
|
|
- tbody {
|
|
|
- font-size: 12px;
|
|
|
- }
|
|
|
-
|
|
|
- th {
|
|
|
- font-size: 12px;
|
|
|
- color: var(--subtitle);
|
|
|
- height: 32px;
|
|
|
- line-height: 1;
|
|
|
- padding-left: 12px;
|
|
|
- box-sizing: border-box;
|
|
|
- text-align: left;
|
|
|
- }
|
|
|
-
|
|
|
- td {
|
|
|
- min-width: 88px;
|
|
|
- padding: 0 10px;
|
|
|
- height: 40px;
|
|
|
- box-sizing: border-box;
|
|
|
-
|
|
|
- &.image {
|
|
|
- min-width: 48px;
|
|
|
- }
|
|
|
-
|
|
|
- &.stock {
|
|
|
- min-width: 138px;
|
|
|
- }
|
|
|
-
|
|
|
- &.stock_warning {
|
|
|
- min-width: 168px;
|
|
|
- }
|
|
|
-
|
|
|
- &.sn {
|
|
|
- min-width: 116px;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-:deep(.el-tabs__header) {
|
|
|
- margin: 0;
|
|
|
-}
|
|
|
-
|
|
|
-:deep(.el-tabs__nav-wrap::after) {
|
|
|
- height: 1px;
|
|
|
-}
|
|
|
-
|
|
|
-:deep(.el-tabs__item) {
|
|
|
- padding: 0 20px;
|
|
|
- font-size: 14px;
|
|
|
-}
|
|
|
-
|
|
|
-:deep(.el-tabs__nav) {
|
|
|
- border: none;
|
|
|
-}
|
|
|
-
|
|
|
-.form-tip {
|
|
|
- font-size: 12px;
|
|
|
- color: var(--el-color-info);
|
|
|
- margin-top: 4px;
|
|
|
- line-height: 1.4;
|
|
|
-}
|
|
|
-
|
|
|
-.mt-1px {
|
|
|
- margin-top: 1px;
|
|
|
-}
|
|
|
-
|
|
|
-/* 必填项样式 */
|
|
|
-.required {
|
|
|
- color: #f56c6c;
|
|
|
- margin-right: 4px;
|
|
|
-}
|
|
|
-
|
|
|
-/* 错误状态样式 */
|
|
|
-.is-error .el-input__wrapper {
|
|
|
- border-color: #f56c6c !important;
|
|
|
- box-shadow: 0 0 0 1px #f56c6c inset !important;
|
|
|
-}
|
|
|
-
|
|
|
-.warning-title {
|
|
|
- font-size: 12px;
|
|
|
- color: #909399;
|
|
|
-}
|
|
|
-
|
|
|
-.th-center {
|
|
|
- text-align: center;
|
|
|
-}
|
|
|
-</style>
|