|
@@ -1,32 +1,259 @@
|
|
|
<template>
|
|
|
- <div class="specifications-container">
|
|
|
+ <div class="specifications-container flex flex-col">
|
|
|
+ <div class="s-header flex flex-row flex-row-aic" v-show="goods_stock.length">
|
|
|
+ <span>商品规格</span>
|
|
|
+ <div class="icons">
|
|
|
+ <van-icon @click="handleRemoveAll" name="delete-o" :size="18" color="#828181" />
|
|
|
+ <van-icon @click="closeable = !closeable" name="setting-o" :size="closeable ? 20 : 18"
|
|
|
+ :color="closeable ? '#414141' : '#828181'" />
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="s-context flex flex-row" v-show="goods_stock.length">
|
|
|
+ <template v-for="(standar, idx) in goods_stock">
|
|
|
+ <van-tag :closeable="closeable" size="large" type="primary" @close="handleClose(standar, idx)"
|
|
|
+ @click="handleEdit(standar, idx)" :key="idx">
|
|
|
+ {{ standar.name }}
|
|
|
+ </van-tag>
|
|
|
+ </template>
|
|
|
+ </div>
|
|
|
+ <div class="s-btn" v-show="goods_stock.length">
|
|
|
+ <van-icon name="plus" :size="14" />
|
|
|
+ <span @click="visibilityDialog = true">追加新规格</span>
|
|
|
+ </div>
|
|
|
+
|
|
|
<div class="main">
|
|
|
+ <!-- 内容 -->
|
|
|
+ <div class="standars-box" v-if="goods_stock.length">
|
|
|
+ <!-- NOTE: goods_stock -->
|
|
|
+ <div class="row" v-for="(item, idx) in goods_stock" :key="idx">
|
|
|
+ <div class="row__header flex flex-row flex-row-aic">
|
|
|
+ <span>
|
|
|
+ {{ item.name }}
|
|
|
+ </span>
|
|
|
+ <!-- <van-icon v-show="closeable" @click="handleClose(item, idx)" name="delete-o" :size="18" color="#b8b8b8" /> -->
|
|
|
+ </div>
|
|
|
+ <div class="row__main">
|
|
|
+ <c-input title="价格" :maxlength="9" :showWordLimit="false" input-type="number" v-model="item.price" />
|
|
|
+ <c-input title="数量" :maxlength="5" :showWordLimit="false" input-type="digit" v-model="item.stock" />
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ </div>
|
|
|
+
|
|
|
<!-- 规格示例 -->
|
|
|
- <div class="addbtn">
|
|
|
+ <div class="addbtn" v-else @click="visibilityDialog = true">
|
|
|
<van-icon name="plus" :size="18" />
|
|
|
添加规格
|
|
|
</div>
|
|
|
- <!-- 内容 -->
|
|
|
</div>
|
|
|
|
|
|
+ <van-dialog v-model="visibilityDialog" title="添加/修改规格" show-cancel-button confirm-button-color="#007aff"
|
|
|
+ @cancel="handleCancelEvent" @confirm="handleConfirmEdit" :before-close="handleBeforeCloseDialog">
|
|
|
+ <van-field v-model.trim="templateValue" center input-align="center" placeholder="蓝色,165cm" />
|
|
|
+ </van-dialog>
|
|
|
+
|
|
|
<!-- btn -->
|
|
|
- <div class="btn-container">
|
|
|
- <div class="btn-span">提交</div>
|
|
|
+ <div class="btn-container" v-if="goods_stock.length">
|
|
|
+ <div class="btn-span" @click="handleSubmitStandars">提交</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
</template>
|
|
|
<script>
|
|
|
+import CInput from './components/CInput.vue'
|
|
|
+import vueBus from '@/utils/vueBus'
|
|
|
+
|
|
|
/**
|
|
|
* @description 商品规格页面
|
|
|
*/
|
|
|
export default {
|
|
|
name: 'GoodsSpeci',
|
|
|
+ components: {
|
|
|
+ CInput
|
|
|
+ },
|
|
|
data: () => ({
|
|
|
- standars: []
|
|
|
+ templateValue: '',
|
|
|
+ visibilityDialog: false,
|
|
|
+ closeable: false,
|
|
|
+ editIdx: -1,
|
|
|
+ goods_stock: [], // {name, price, stock}
|
|
|
}),
|
|
|
created() {
|
|
|
- console.log('%c standars >>>', 'background: blue; color: #fff', this.$route.query.standars);
|
|
|
-
|
|
|
+ // NOTE: 携带规格更新
|
|
|
+ this.__init__()
|
|
|
},
|
|
|
+ methods: {
|
|
|
+ __init__() {
|
|
|
+ // NOTE: 携带规格更新
|
|
|
+ const { goods_stock } = this.$route.query
|
|
|
+ const stocks = goods_stock ? JSON.parse(goods_stock) : []
|
|
|
+
|
|
|
+ if (stocks.length) {
|
|
|
+ this.goods_stock = [...stocks].reverse()
|
|
|
+ }
|
|
|
+ },
|
|
|
+ // Remove all standars
|
|
|
+ handleRemoveAll() {
|
|
|
+ this.$dialog.confirm({
|
|
|
+ message: '是否删除全部规格',
|
|
|
+ confirmButtonColor: '#007aff',
|
|
|
+ }).then(() => {
|
|
|
+ this.goods_stock = []
|
|
|
+ }).catch(() => { })
|
|
|
+ },
|
|
|
+
|
|
|
+ // Close signal
|
|
|
+ handleClose(standar, idx) {
|
|
|
+ this.$dialog.confirm({
|
|
|
+ message: `是否删除规格 ${standar.name}`,
|
|
|
+ confirmButtonColor: '#007aff',
|
|
|
+ }).then(() => {
|
|
|
+ this.goods_stock.splice(idx, 1)
|
|
|
+ }).catch(() => { })
|
|
|
+ },
|
|
|
+
|
|
|
+ // Edit signal
|
|
|
+ handleEdit(standar, idx) {
|
|
|
+ this.templateValue = standar
|
|
|
+ this.visibilityDialog = true
|
|
|
+ this.editIdx = idx
|
|
|
+ },
|
|
|
+
|
|
|
+ // Close dialog on before-close
|
|
|
+ handleBeforeCloseDialog(action, done) {
|
|
|
+ if (action === 'confirm' && !this.templateValue) return done(false)
|
|
|
+ const isRepeat = this.goods_stock.some(stock => stock.name === this.templateValue)
|
|
|
+ if (isRepeat) return done(false)
|
|
|
+ return done()
|
|
|
+ },
|
|
|
+
|
|
|
+ handleCancelEvent() {
|
|
|
+ this.visibilityDialog = false
|
|
|
+ this.templateValue = ''
|
|
|
+ this.editIdx = -1
|
|
|
+ },
|
|
|
+ // Confirm edit
|
|
|
+ handleConfirmEdit() {
|
|
|
+ const val = this.templateValue
|
|
|
+ if (!val) return this.$toast('规格名称为空')
|
|
|
+ const isRepeat = this.goods_stock.some(stock => stock.name === val)
|
|
|
+ if (isRepeat) return this.$toast('规格名称重复!')
|
|
|
+
|
|
|
+ if (this.editIdx >= 0) {
|
|
|
+ this.goods_stock.splice(this.editIdx, 1, {
|
|
|
+ ...this.goods_stock[this.editIdx],
|
|
|
+ name: val
|
|
|
+ })
|
|
|
+ } else {
|
|
|
+ this.goods_stock.unshift({
|
|
|
+ name: val,
|
|
|
+ price: '',
|
|
|
+ stock: '',
|
|
|
+ })
|
|
|
+ }
|
|
|
+ this.$nextTick(() => {
|
|
|
+ this.visibilityDialog = false
|
|
|
+ this.templateValue = ''
|
|
|
+ this.editIdx = -1
|
|
|
+ })
|
|
|
+ },
|
|
|
+
|
|
|
+ // submit data
|
|
|
+ handleSubmitStandars() {
|
|
|
+ const arrs = this.goods_stock
|
|
|
+ let isExist = false
|
|
|
+ // NOTE: 判断内容是否存在
|
|
|
+ for (let i = 0; i < arrs.length; i++) {
|
|
|
+ const { price, stock, name } = arrs[i];
|
|
|
+ if (!price || !stock) {
|
|
|
+ isExist = true
|
|
|
+ this.$toast(`规则${name}的${!price ? '价格' : '数量'}为空,请填写`)
|
|
|
+ break
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (isExist) return
|
|
|
+
|
|
|
+ vueBus.$emit('listenStrandarsEvent', [...arrs].reverse())
|
|
|
+
|
|
|
+ this.$nextTick(() => {
|
|
|
+ this.$router.go(-1)
|
|
|
+ })
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+</script>
|
|
|
+
|
|
|
+<style lang="less" scoped>
|
|
|
+.specifications-container {
|
|
|
+ height: 100vh;
|
|
|
+ justify-content: space-between;
|
|
|
+
|
|
|
+ .s-header {
|
|
|
+ justify-content: space-between;
|
|
|
+ background-color: #fff;
|
|
|
+ font-size: 18px;
|
|
|
+ padding: 10px;
|
|
|
+ border-bottom: 1px solid #eee;
|
|
|
+
|
|
|
+ .icons {
|
|
|
+ letter-spacing: 8px;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .s-btn {
|
|
|
+ padding: 12px 0;
|
|
|
+ text-align: center;
|
|
|
+ font-size: 14px;
|
|
|
+ color: #3290C4;
|
|
|
+ }
|
|
|
+
|
|
|
+ .s-context {
|
|
|
+ flex-wrap: wrap;
|
|
|
+ gap: 12px;
|
|
|
+ padding: 18px 10px;
|
|
|
+ background-color: #fff;
|
|
|
+ }
|
|
|
+
|
|
|
+ .main {
|
|
|
+ height: 0;
|
|
|
+ flex: 1;
|
|
|
+ overflow: auto;
|
|
|
+
|
|
|
+ .addbtn {
|
|
|
+ width: 195px;
|
|
|
+ height: 42px;
|
|
|
+ line-height: 42px;
|
|
|
+ text-align: center;
|
|
|
+ background: #FFFFFF;
|
|
|
+ border-radius: 21px;
|
|
|
+ border: 1px solid #3290C4;
|
|
|
+ font-size: 16px;
|
|
|
+ font-weight: 500;
|
|
|
+ color: #3290C4;
|
|
|
+ margin: 80px auto;
|
|
|
+ }
|
|
|
+
|
|
|
+ .row {
|
|
|
+ &__header {
|
|
|
+ justify-content: space-between;
|
|
|
+ padding: 0 10px;
|
|
|
+ font-size: 14px;
|
|
|
+ font-family: PingFangSC, PingFang SC;
|
|
|
+ font-weight: 400;
|
|
|
+ color: #727273;
|
|
|
+ line-height: 32px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &__main {
|
|
|
+ .layout-container {
|
|
|
+ border-bottom: 1px solid #eee;
|
|
|
+
|
|
|
+ &:last-child {
|
|
|
+ border-bottom: initial;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
-</script>
|
|
|
+</style>
|