[Improvement] Sku: support i18n (#439)

This commit is contained in:
neverland 2017-12-15 15:38:32 +08:00 committed by GitHub
parent 2db9b12816
commit f43b496333
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 846 additions and 507 deletions

View File

@ -1,236 +1,517 @@
/* eslint-disable */ /* eslint-disable */
var _global = { export default {
"kdt_id": 55, 'zh-CN': {
"user_id": 4674509, kdt_id: 55,
"offline_id": 0, user_id: 4674509,
"activity_alias": "", offline_id: 0,
"sku": { activity_alias: '',
"tree": [{ sku: {
"k": "\u989c\u8272", tree: [
"k_id": "1", {
"v": [{ k: '颜色',
"id": "30349", k_id: '1',
"name": "\u5929\u84dd\u8272", v: [
"imgUrl": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/02\/21\/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg" {
}], id: '30349',
"k_s": "s1", name: '天蓝色',
"count": 2 imgUrl:
}, { 'https://img.yzcdn.cn/upload_files/2017/02/21/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg'
"k": "\u5c3a\u5bf8", }
"k_id": "2", ],
"v": [{ k_s: 's1',
"id": "1193", count: 2
"name": "1" },
}, { {
"id": "1194", k: '尺寸',
"name": "2" k_id: '2',
}], v: [
"k_s": "s2", {
"count": 2 id: '1193',
}], name: '1'
"list": [{ },
"id": 2259, {
"price": 100, id: '1194',
"discount": 100, name: '2'
"code": "", }
"s1": "1215", ],
"s2": "1193", k_s: 's2',
"s3": "0", count: 2
"s4": "0", }
"s5": "0", ],
"extend": null, list: [
"kdt_id": 55, {
"discount_price": 0, id: 2259,
"stock_num": 110, price: 100,
"stock_mode": 0, discount: 100,
"is_sell": null, code: '',
"combin_sku": false, s1: '1215',
"goods_id": 946755 s2: '1193',
}, { s3: '0',
"id": 2260, s4: '0',
"price": 100, s5: '0',
"discount": 100, extend: null,
"code": "", kdt_id: 55,
"s1": "1215", discount_price: 0,
"s2": "1194", stock_num: 110,
"s3": "0", stock_mode: 0,
"s4": "0", is_sell: null,
"s5": "0", combin_sku: false,
"extend": null, goods_id: 946755
"kdt_id": 55, },
"discount_price": 0, {
"stock_num": 0, id: 2260,
"stock_mode": 0, price: 100,
"is_sell": null, discount: 100,
"combin_sku": false, code: '',
"goods_id": 946755 s1: '1215',
}, { s2: '1194',
"id": 2257, s3: '0',
"price": 100, s4: '0',
"discount": 100, s5: '0',
"code": "", extend: null,
"s1": "30349", kdt_id: 55,
"s2": "1193", discount_price: 0,
"s3": "0", stock_num: 0,
"s4": "0", stock_mode: 0,
"s5": "0", is_sell: null,
"extend": null, combin_sku: false,
"kdt_id": 55, goods_id: 946755
"discount_price": 0, },
"stock_num": 111, {
"stock_mode": 0, id: 2257,
"is_sell": null, price: 100,
"combin_sku": false, discount: 100,
"goods_id": 946755 code: '',
}, { s1: '30349',
"id": 2258, s2: '1193',
"price": 100, s3: '0',
"discount": 100, s4: '0',
"code": "", s5: '0',
"s1": "30349", extend: null,
"s2": "1194", kdt_id: 55,
"s3": "0", discount_price: 0,
"s4": "0", stock_num: 111,
"s5": "0", stock_mode: 0,
"extend": null, is_sell: null,
"kdt_id": 55, combin_sku: false,
"discount_price": 0, goods_id: 946755
"stock_num": 6, },
"stock_mode": 0, {
"is_sell": null, id: 2258,
"combin_sku": false, price: 100,
"goods_id": 946755 discount: 100,
}], code: '',
"price": "1.00", s1: '30349',
"stock_num": 227, s2: '1194',
"collection_id": 2261, s3: '0',
"collection_price": 0, s4: '0',
"none_sku": false, s5: '0',
"sold_num": 0, extend: null,
"min_price": "1.00", kdt_id: 55,
"max_price": "1.00", discount_price: 0,
"messages": [{ stock_num: 6,
"datetime": "0", stock_mode: 0,
"disable_multiple": false, is_sell: null,
"disable": false, combin_sku: false,
"multiple": "0", goods_id: 946755
"name": "\u7559\u8a001", }
"disable_required": false, ],
"disable_edit_name": false, price: '1.00',
"type": "text", stock_num: 227,
"disable_delete": false, collection_id: 2261,
"disable_type": false, collection_price: 0,
"required": "1" none_sku: false,
}, { sold_num: 0,
"datetime": "0", min_price: '1.00',
"disable_multiple": false, max_price: '1.00',
"disable": false, messages: [
"multiple": 0, {
"name": "\u7559\u8a002", datetime: '0',
"disable_required": false, disable_multiple: false,
"disable_edit_name": false, disable: false,
"type": "id_no", multiple: '0',
"disable_delete": false, name: '留言1',
"disable_type": false, disable_required: false,
"required": 0 disable_edit_name: false,
}, { type: 'text',
"datetime": "0", disable_delete: false,
"disable_multiple": false, disable_type: false,
"disable": false, required: '1'
"multiple": 0, },
"name": "\u7559\u8a003", {
"disable_required": false, datetime: '0',
"disable_edit_name": false, disable_multiple: false,
"type": "image", disable: false,
"disable_delete": false, multiple: 0,
"disable_type": false, name: '留言2',
"required": 0 disable_required: false,
}, { disable_edit_name: false,
"datetime": "0", type: 'id_no',
"disable_multiple": false, disable_delete: false,
"disable": false, disable_type: false,
"multiple": 1, required: 0
"name": "\u7559\u8a004", },
"disable_required": false, {
"disable_edit_name": false, datetime: '0',
"type": "text", disable_multiple: false,
"disable_delete": false, disable: false,
"disable_type": false, multiple: 0,
"required": 0 name: '留言3',
}, { disable_required: false,
"datetime": "0", disable_edit_name: false,
"disable_multiple": false, type: 'image',
"disable": false, disable_delete: false,
"name": "\u6570\u5b57", disable_type: false,
"multiple": 0, required: 0
"disable_required": false, },
"disable_edit_name": false, {
"type": "tel", datetime: '0',
"disable_delete": false, disable_multiple: false,
"disable_type": false, disable: false,
"required": 0 multiple: 1,
}, { name: '留言4',
"datetime": "0", disable_required: false,
"disable_multiple": false, disable_edit_name: false,
"disable": false, type: 'text',
"name": "\u90ae\u4ef6", disable_delete: false,
"multiple": 0, disable_type: false,
"disable_required": false, required: 0
"disable_edit_name": false, },
"type": "email", {
"disable_delete": false, datetime: '0',
"disable_type": false, disable_multiple: false,
"required": 0 disable: false,
}, { name: '数字',
"datetime": "0", multiple: 0,
"disable_multiple": false, disable_required: false,
"disable": false, disable_edit_name: false,
"name": "\u65e5\u671f", type: 'tel',
"multiple": 0, disable_delete: false,
"disable_required": false, disable_type: false,
"disable_edit_name": false, required: 0
"type": "date", },
"disable_delete": false, {
"disable_type": false, datetime: '0',
"required": 0 disable_multiple: false,
}, { disable: false,
"datetime": "0", name: '邮件',
"disable_multiple": false, multiple: 0,
"disable": false, disable_required: false,
"name": "\u65f6\u95f4\u542b\u65e5\u671f", disable_edit_name: false,
"multiple": 0, type: 'email',
"disable_required": false, disable_delete: false,
"disable_edit_name": false, disable_type: false,
"type": "time", required: 0
"disable_delete": false, },
"disable_type": false, {
"required": 0 datetime: '0',
}, { disable_multiple: false,
"datetime": "0", disable: false,
"disable_multiple": false, name: '日期',
"disable": false, multiple: 0,
"name": "\u65f6\u95f4", disable_required: false,
"multiple": 0, disable_edit_name: false,
"disable_required": false, type: 'date',
"disable_edit_name": false, disable_delete: false,
"type": "time", disable_type: false,
"disable_delete": false, required: 0
"disable_type": false, },
"required": 0 {
}], datetime: '0',
"hide_stock": false disable_multiple: false,
disable: false,
name: '时间含日期',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'time',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: '时间',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'time',
disable_delete: false,
disable_type: false,
required: 0
}
],
hide_stock: false
}, },
"goods_id": "946755", goods_id: '946755',
"alias": "2oml0r0n5vytj", alias: '2oml0r0n5vytj',
"quota": 15, quota: 15,
"is_virtual": "0", is_virtual: '0',
"quota_used": 0, quota_used: 0,
"goods_info": { goods_info: {
"title": "测试商品", title: '测试商品',
"picture": ["https:\/\/img.yzcdn.cn\/upload_files\/2017\/03\/16\/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2\/2\/w\/100\/h\/100\/q\/75\/format\/jpg"], picture: 'https://img.yzcdn.cn/upload_files/2017/03/16/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2/2/w/100/h/100/q/75/format/jpg',
"price": 1, price: 1,
"origin": "" origin: ''
} }
},
'en-US': {
kdt_id: 55,
user_id: 4674509,
offline_id: 0,
activity_alias: '',
sku: {
tree: [
{
k: 'Color',
k_id: '1',
v: [
{
id: '30349',
name: 'Blue',
imgUrl:
'https://img.yzcdn.cn/upload_files/2017/02/21/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg'
}
],
k_s: 's1',
count: 2
},
{
k: 'Size',
k_id: '2',
v: [
{
id: '1193',
name: '1'
},
{
id: '1194',
name: '2'
}
],
k_s: 's2',
count: 2
}
],
list: [
{
id: 2259,
price: 100,
discount: 100,
code: '',
s1: '1215',
s2: '1193',
s3: '0',
s4: '0',
s5: '0',
extend: null,
kdt_id: 55,
discount_price: 0,
stock_num: 110,
stock_mode: 0,
is_sell: null,
combin_sku: false,
goods_id: 946755
},
{
id: 2260,
price: 100,
discount: 100,
code: '',
s1: '1215',
s2: '1194',
s3: '0',
s4: '0',
s5: '0',
extend: null,
kdt_id: 55,
discount_price: 0,
stock_num: 0,
stock_mode: 0,
is_sell: null,
combin_sku: false,
goods_id: 946755
},
{
id: 2257,
price: 100,
discount: 100,
code: '',
s1: '30349',
s2: '1193',
s3: '0',
s4: '0',
s5: '0',
extend: null,
kdt_id: 55,
discount_price: 0,
stock_num: 111,
stock_mode: 0,
is_sell: null,
combin_sku: false,
goods_id: 946755
},
{
id: 2258,
price: 100,
discount: 100,
code: '',
s1: '30349',
s2: '1194',
s3: '0',
s4: '0',
s5: '0',
extend: null,
kdt_id: 55,
discount_price: 0,
stock_num: 6,
stock_mode: 0,
is_sell: null,
combin_sku: false,
goods_id: 946755
}
],
price: '1.00',
stock_num: 227,
collection_id: 2261,
collection_price: 0,
none_sku: false,
sold_num: 0,
min_price: '1.00',
max_price: '1.00',
messages: [
{
datetime: '0',
disable_multiple: false,
disable: false,
multiple: '0',
name: 'Messsage 1',
disable_required: false,
disable_edit_name: false,
type: 'text',
disable_delete: false,
disable_type: false,
required: '1'
},
{
datetime: '0',
disable_multiple: false,
disable: false,
multiple: 0,
name: 'Messsage 2',
disable_required: false,
disable_edit_name: false,
type: 'id_no',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
multiple: 0,
name: 'Messsage 3',
disable_required: false,
disable_edit_name: false,
type: 'image',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
multiple: 1,
name: 'Messsage 4',
disable_required: false,
disable_edit_name: false,
type: 'text',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: 'Number',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'tel',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: 'Email',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'email',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: 'Date',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'date',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: 'Datetime',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'time',
disable_delete: false,
disable_type: false,
required: 0
},
{
datetime: '0',
disable_multiple: false,
disable: false,
name: 'Time',
multiple: 0,
disable_required: false,
disable_edit_name: false,
type: 'time',
disable_delete: false,
disable_type: false,
required: 0
}
],
hide_stock: false
},
goods_id: '946755',
alias: '2oml0r0n5vytj',
quota: 15,
is_virtual: '0',
quota_used: 0,
goods_info: {
title: 'Goods Name',
picture: 'https://img.yzcdn.cn/upload_files/2017/03/16/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2/2/w/100/h/100/q/75/format/jpg',
price: 1,
origin: ''
}
}
}; };
export default _global;

View File

@ -4,19 +4,19 @@
<div class="sku-container"> <div class="sku-container">
<van-sku <van-sku
v-model="showBase" v-model="showBase"
:sku="sku" :sku="$t('sku').sku"
:goods="goods" :goods="$t('sku').goods_info"
:goodsId="goodsId" :goodsId="$t('sku').goods_id"
:hideStock="sku.hide_stock" :hideStock="$t('sku').sku.hide_stock"
:quota="quota" :quota="$t('sku').quota"
:quotaUsed="quotaUsed" :quotaUsed="$t('sku').quota_used"
:resetStepperOnHide="resetStepperOnHide" :resetStepperOnHide="true"
:disableStepperInput="disableStepperInput" :disableStepperInput="true"
@buy-clicked="handleBuyClicked" @buy-clicked="handleBuyClicked"
@add-cart="handleAddCartClicked" @add-cart="handleAddCartClicked"
> >
</van-sku> </van-sku>
<van-button type="primary" @click="showBase = true" block>基础用法</van-button> <van-button type="primary" @click="showBase = true" block>{{ $t('basicUsage') }}</van-button>
</div> </div>
</demo-block> </demo-block>
@ -24,31 +24,28 @@
<div class="sku-container"> <div class="sku-container">
<van-sku <van-sku
v-model="showCustomAction" v-model="showCustomAction"
stepperTitle="我要买" :stepperTitle="$t('stepperTitle')"
:sku="sku" :sku="$t('sku').sku"
:goods="goods" :goods="$t('sku').goods_info"
:goodsId="goodsId" :goodsId="$t('sku').goods_id"
:hideStock="sku.hide_stock" :hideStock="$t('sku').sku.hide_stock"
:showAddCartBtn="true" :showAddCartBtn="true"
:quota="quota" :quota="$t('sku').quota"
:quotaUsed="quotaUsed" :quotaUsed="$t('sku').quota_used"
:resetStepperOnHide="true" :resetStepperOnHide="true"
:initialSku="initialSku" :initialSku="initialSku"
@buy-clicked="handleBuyClicked" @buy-clicked="handleBuyClicked"
@add-cart="handleAddCartClicked" @add-cart="handleAddCartClicked"
> >
<!-- 隐藏sku messages -->
<template slot="sku-messages"></template> <template slot="sku-messages"></template>
<!-- 自定义sku actions -->
<template slot="sku-actions" slot-scope="props"> <template slot="sku-actions" slot-scope="props">
<div class="van-sku-actions"> <div class="van-sku-actions">
<van-button bottomAction @click="handlePointClicked">积分兑换</van-button> <van-button bottomAction @click="handlePointClicked">{{ $t('button1') }}</van-button>
<!-- 直接触发sku内部事件通过内部事件执行handleBuyClicked回调 --> <van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">{{ $t('button2') }}</van-button>
<van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">买买买</van-button>
</div> </div>
</template> </template>
</van-sku> </van-sku>
<van-button type="primary" @click="showCustomAction = true" block>自定义sku actions</van-button> <van-button type="primary" @click="showCustomAction = true" block>{{ $t('advancedUsage') }}</van-button>
</div> </div>
</demo-block> </demo-block>
</demo-section> </demo-section>
@ -57,16 +54,19 @@
<script> <script>
import data from '../mock/sku'; import data from '../mock/sku';
const goods = data.goods_info;
goods.picture = goods.picture[0];
export default { export default {
i18n: { i18n: {
'zh-CN': { 'zh-CN': {
sku: data['zh-CN'],
stepperTitle: '我要买',
button1: '积分兑换',
button2: '买买买'
}, },
'en-US': { 'en-US': {
sku: data['en-US'],
stepperTitle: 'Stepper title',
button1: 'Button',
button2: 'Button'
} }
}, },
@ -74,13 +74,6 @@ export default {
return { return {
showBase: false, showBase: false,
showCustomAction: false, showCustomAction: false,
sku: data.sku,
goods: goods,
goodsId: data.goods_id,
quota: data.quota,
quotaUsed: data.quota_used,
disableStepperInput: true,
resetStepperOnHide: true,
initialSku: { initialSku: {
s1: '30349', s1: '30349',
s2: '1193' s2: '1193'

View File

@ -26,12 +26,12 @@ Vue.use(Sku);
/> />
``` ```
#### 自定义 sku slot 区块 #### Advanced Usage
```html ```html
<van-sku <van-sku
v-model="showCustomAction" v-model="showCustomAction"
stepperTitle="我要买" stepperTitle="Stepper title"
:sku="sku" :sku="sku"
:goods="goods" :goods="goods"
:goodsId="goodsId" :goodsId="goodsId"
@ -44,14 +44,14 @@ Vue.use(Sku);
@buy-clicked="handleBuyClicked" @buy-clicked="handleBuyClicked"
@add-cart="handleAddCartClicked" @add-cart="handleAddCartClicked"
> >
<!-- 隐藏sku messages --> <!-- hide sku messages -->
<template slot="sku-messages"></template> <template slot="sku-messages"></template>
<!-- 自定义sku actions --> <!-- custom sku actions -->
<template slot="sku-actions" slot-scope="props"> <template slot="sku-actions" slot-scope="props">
<div class="van-sku-actions"> <div class="van-sku-actions">
<van-button bottomAction @click="handlePointClicked">积分兑换</van-button> <van-button bottomAction @click="handlePointClicked">Button</van-button>
<!-- 直接触发sku内部事件通过内部事件执行handleBuyClicked回调 --> <!-- trigger sku inner event -->
<van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">买买买</van-button> <van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">Button</van-button>
</div> </div>
</template> </template>
</van-sku> </van-sku>
@ -59,113 +59,116 @@ Vue.use(Sku);
### API ### API
| Attribute | Description | Type | Default | 必须 | | Attribute | Description | Type | Default | Accepted Values |
|-----------|-----------|-----------|-------------|-------------| |-----------|-----------|-----------|-------------|-------------|
| v-model | 是否显示sku | Boolean | false | 是 | | v-model | Whether to show sku | `Boolean` | `false` | - |
| sku | 商品sku数据 | Object | - | 是 | | sku | Sku data | `Object` | - | - |
| goods | 商品信息 | Object | - | 是 | | goods | Goods info | `Object` | - | - |
| goodsId | 商品id | String/Number | - | 是 | | goodsId | Goods id | `String | Number` | - | - |
| hideStock | 是否显示商品剩余库存 | Boolean | false | 否 | | hideStock | Whether to hide stock | `Boolean` | `false` | - |
| showAddCartBtn | 是否显示加入购物车按钮 | Boolean | true | 否 | | showAddCartBtn | Whether to show cart button | `Boolean` | `true` | - |
| quota | 限购数(0表示不限购) | Number | 0 | 否 | | quota | Quota (0 as no limit) | `Number` | `0` | - |
| quotaUsed | 已经购买过的数量 | Number | 0 | 否 | | quotaUsed | Used quota | `Number` | `0` | - |
| resetStepperOnHide | 窗口隐藏时重置选择的商品数量 | Boolean | false | 否 | | resetStepperOnHide | Whether to reset stepper when hide | `Boolean` | `false` | - |
| disableStepperInput | 是否禁用sku中stepper的input框 | Boolean | false | 否 | | disableStepperInput | Whether to disable stepper input | `Boolean` | `false` | - |
| stepperTitle | 数量选择组件左侧文案 | String | '购买数量' | 否 | | stepperTitle | Quantity title | `String` | `Quantity` | - |
| add-cart | 点击添加购物车回调 | Function(skuData: Object) | - | 否 |
| buy-clicked | 点击购买回调 | Function(skuData: Object) | - | 否 |
### slots ### Event
sku组件默认划分好了若干区块这些区块都定义成了slot可以按需进行替换。区块顺序见下表
| Event | Description | Attribute |
|-----------|-----------|-----------|
| add-cart | Triggered when click cart button | data: Object |
| buy-clicked | Triggered when click buy button | data: Object |
### Slot
| Name | Description | | Name | Description |
|-----------|-----------| |-----------|-----------|
| sku-header | 商品信息展示区,包含商品图片、名称、价格等信息 | | sku-header | Custom header |
| sku-group | 商品sku展示区 | | sku-group | Custom sku |
| extra-sku-group | 额外商品sku展示区一般用不到 | | extra-sku-group | Extra custom content |
| sku-stepper | 商品数量选择区 | | sku-stepper | Custom stepper |
| sku-messages | 商品留言区 | | sku-messages | Custom messages |
| sku-actions | 操作按钮区 | | sku-actions | Custom button actions |
#### Data Structure #### Data Structure
#### sku对象结构 #### Sku Data Structure
```javascript ```javascript
"sku": { sku: {
// 所有sku规格类目与其值的从属关系比如商品有颜色和尺码两大类规格颜色下面又有红色和蓝色两个规格值。 tree: [
// 可以理解为一个商品可以有多个规格类目,一个规格类目下可以有多个规格值。 {
"tree": [{ k: 'Color',
"k": "颜色", // skuKeyName规格类目名称 v: [
"v": [{ {
"id": "30349", // skuValueId规格值id id: '30349',
"name": "红色", // skuValueName规格值名称 name: 'Red',
"imgUrl": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/02\/21\/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg" // 规格类目图片,只有第一个规格类目可以定义图片 imgUrl: 'https://img.yzcdn.cn/1.jpg'
}, { },
"id": "1215", {
"name": "蓝色", id: '1215',
"imgUrl": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/03\/16\/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg" name: 'Blue',
}], imgUrl: 'https://img.yzcdn.cn/2.jpg'
"k_s": "s1" // skuKeyStrsku组合列表下方list中当前类目对应的key值value值会是从属于当前类目的一个规格值id }
}, ...], ],
// 所有sku的组合列表比如红色、M码为一个sku组合红色、S码为另一个组合 k_s: 's1'
"list": [{ }
"id": 2259, // skuId下单时后端需要 ],
"price": 100, // 价格(单位分) list: [
"s1": "1215", // 规格类目k_s为s1的对应规格值id {
"s2": "1193", // 规格类目k_s为s2的对应规格值id id: 2259,
"s3": "0", // 最多包含3个规格值为0表示不存在该规格 price: 100,
"stock_num": 110 // 当前sku组合对应的库存 s1: '1215',
}, ...], s2: '1193',
"price": "1.00", // 默认价格(单位元)后端单位暂时有点不统一 s3: '0',
"stock_num": 227, // 商品总库存 stock_num: 110
"collection_id": 2261, // 无规格商品skuId取collection_id否则取所选sku组合对应的id }
"none_sku": false, // 是否无规格商品 ],
"messages": [{ // 商品留言 price: '1.00',
"datetime": "0", // 留言类型为time时是否含日期。“1”表示包含 stock_num: 227,
"multiple": "0", // 留言类型为text时是否多行文本。“1”表示多行 collection_id: 2261,
"name": "留言", // 留言名称 none_sku: false,
"type": "text", // 留言类型可选id_no身份证, text, tel, date, time, email messages: [
"required": "1" // 是否必填 “1”表示必填 {
}, ...], datetime: '0',
"hide_stock": false // 是否隐藏剩余库存 multiple: '0',
}, name: 'Message',
type: 'text',
required: '1'
}
],
hide_stock: false
}
``` ```
#### goods对象结构 #### Goods Data Structure
```javascript ```javascript
"goods": { goods: {
// 商品标题 title: 'Title',
"title": "测试商品", picture: 'https://img.yzcdn.cn/1.jpg'
// 默认商品sku缩略图 }
"picture": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/03\/16\/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2\/2\/w\/100\/h\/100\/q\/75\/format\/webp"
},
``` ```
#### 添加购物车和点击购买回调函数接收的skuData对象结构 #### Event Params Data Structure
```javascript ```javascript
skuData: { skuData: {
// 商品id goodsId: '946755',
goodsId:"946755",
// 留言信息
messages: { messages: {
message_0:"12", message_0: '12',
message_1:"", message_1: ''
... // 有几个留言就有几条
}, },
// 另一种格式的留言key不同
cartMessages: { cartMessages: {
'留言1': 'xxxx', 'Message 1': 'xxxx'
... // key是message的name
}, },
// 选择的商品数量 selectedNum: 1,
selectedNum:1,
// 选择的sku组合
selectedSkuComb: { selectedSkuComb: {
id:2257, id: 2257,
price:100, price: 100,
s1:"30349", s1: '30349',
s2:"1193", s2: '1193',
s3:"0", s3: '0',
stock_num:111 stock_num: 111
} }
} }
``` ```

View File

@ -26,7 +26,7 @@ Vue.use(Sku);
/> />
``` ```
#### 自定义 sku slot 区块 #### 高级用法
```html ```html
<van-sku <van-sku
@ -44,13 +44,13 @@ Vue.use(Sku);
@buy-clicked="handleBuyClicked" @buy-clicked="handleBuyClicked"
@add-cart="handleAddCartClicked" @add-cart="handleAddCartClicked"
> >
<!-- 隐藏sku messages --> <!-- 隐藏 sku messages -->
<template slot="sku-messages"></template> <template slot="sku-messages"></template>
<!-- 自定义sku actions --> <!-- 自定义 sku actions -->
<template slot="sku-actions" slot-scope="props"> <template slot="sku-actions" slot-scope="props">
<div class="van-sku-actions"> <div class="van-sku-actions">
<van-button bottomAction @click="handlePointClicked">积分兑换</van-button> <van-button bottomAction @click="handlePointClicked">积分兑换</van-button>
<!-- 直接触发sku内部事件通过内部事件执行handleBuyClicked回调 --> <!-- 直接触发 sku 内部事件,通过内部事件执行 handleBuyClicked 回调 -->
<van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">买买买</van-button> <van-button type="primary" bottomAction @click="props.skuEventBus.$emit('sku:buy')">买买买</van-button>
</div> </div>
</template> </template>
@ -60,26 +60,32 @@ Vue.use(Sku);
### API ### API
| 参数 | 说明 | 类型 | 默认值 | 必须 | | 参数 | 说明 | 类型 | 默认值 | 可选值 |
|-----------|-----------|-----------|-------------|-------------| |-----------|-----------|-----------|-------------|-------------|
| v-model | 是否显示sku | Boolean | false | 是 | | v-model | 是否显示sku | `Boolean` | `false` | - |
| sku | 商品sku数据 | Object | - | 是 | | sku | 商品sku数据 | `Object` | - | - |
| goods | 商品信息 | Object | - | 是 | | goods | 商品信息 | `Object` | - | - |
| goodsId | 商品id | String/Number | - | 是 | | goodsId | 商品id | `String | Number` | - | - |
| hideStock | 是否显示商品剩余库存 | Boolean | false | 否 | | hideStock | 是否显示商品剩余库存 | `Boolean` | `false` | - |
| showAddCartBtn | 是否显示加入购物车按钮 | Boolean | true | 否 | | showAddCartBtn | 是否显示加入购物车按钮 | `Boolean` | `true` | - |
| quota | 限购数(0表示不限购) | Number | 0 | 否 | | quota | 限购数(0表示不限购) | `Number` | `0` | - |
| quotaUsed | 已经购买过的数量 | Number | 0 | 否 | | quotaUsed | 已经购买过的数量 | `Number` | `0` | - |
| resetStepperOnHide | 窗口隐藏时重置选择的商品数量 | Boolean | false | 否 | | resetStepperOnHide | 窗口隐藏时重置选择的商品数量 | `Boolean` | `false` | - |
| disableStepperInput | 是否禁用sku中stepper的input框 | Boolean | false | 否 | | disableStepperInput | 是否禁用sku中stepper的input框 | `Boolean` | `false` | - |
| stepperTitle | 数量选择组件左侧文案 | String | '购买数量' | 否 | | stepperTitle | 数量选择组件左侧文案 | `String` | `购买数量` | - |
| add-cart | 点击添加购物车回调 | Function(skuData: Object) | - | 否 |
| buy-clicked | 点击购买回调 | Function(skuData: Object) | - | 否 |
### slots ### Event
sku组件默认划分好了若干区块这些区块都定义成了slot可以按需进行替换。区块顺序见下表
| Name | 说明 | | 事件名 | 说明 | 参数 |
|-----------|-----------|-----------|
| add-cart | 点击添加购物车回调 | skuData: Object |
| buy-clicked | 点击购买回调 | skuData: Object |
### Slot
Sku 组件默认划分好了若干区块,这些区块都定义成了 slot可以按需进行替换。区块顺序见下表
| 名称 | 说明 |
|-----------|-----------| |-----------|-----------|
| sku-header | 商品信息展示区,包含商品图片、名称、价格等信息 | | sku-header | 商品信息展示区,包含商品图片、名称、价格等信息 |
| sku-group | 商品sku展示区 | | sku-group | 商品sku展示区 |
@ -91,82 +97,90 @@ sku组件默认划分好了若干区块这些区块都定义成了slot
### 数据结构 ### 数据结构
#### sku对象结构 #### sku对象结构
```javascript ```javascript
"sku": { sku: {
// 所有sku规格类目与其值的从属关系比如商品有颜色和尺码两大类规格颜色下面又有红色和蓝色两个规格值。 // 所有sku规格类目与其值的从属关系比如商品有颜色和尺码两大类规格颜色下面又有红色和蓝色两个规格值。
// 可以理解为一个商品可以有多个规格类目,一个规格类目下可以有多个规格值。 // 可以理解为一个商品可以有多个规格类目,一个规格类目下可以有多个规格值。
"tree": [{ tree: [
"k": "颜色", // skuKeyName规格类目名称 {
"v": [{ k: '颜色', // skuKeyName规格类目名称
"id": "30349", // skuValueId规格值id v: [
"name": "红色", // skuValueName规格值名称 {
"imgUrl": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/02\/21\/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg" // 规格类目图片,只有第一个规格类目可以定义图片 id: '30349', // skuValueId规格值 id
}, { name: '红色', // skuValueName规格值名称
"id": "1215", imgUrl: 'https://img.yzcdn.cn/1.jpg' // 规格类目图片,只有第一个规格类目可以定义图片
"name": "蓝色", },
"imgUrl": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/03\/16\/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg" {
}], id: '1215',
"k_s": "s1" // skuKeyStrsku组合列表下方list中当前类目对应的key值value值会是从属于当前类目的一个规格值id name: '蓝色',
}, ...], imgUrl: 'https://img.yzcdn.cn/2.jpg'
// 所有sku的组合列表比如红色、M码为一个sku组合红色、S码为另一个组合 }
"list": [{ ],
"id": 2259, // skuId下单时后端需要 k_s: 's1' // skuKeyStrsku 组合列表(下方 list中当前类目对应的 key 值value 值会是从属于当前类目的一个规格值 id
"price": 100, // 价格(单位分) }
"s1": "1215", // 规格类目k_s为s1的对应规格值id ],
"s2": "1193", // 规格类目k_s为s2的对应规格值id // 所有 sku 的组合列表比如红色、M 码为一个 sku 组合红色、S 码为另一个组合
"s3": "0", // 最多包含3个规格值为0表示不存在该规格 list: [
"stock_num": 110 // 当前sku组合对应的库存 {
}, ...], id: 2259, // skuId下单时后端需要
"price": "1.00", // 默认价格(单位元)后端单位暂时有点不统一 price: 100, // 价格(单位分)
"stock_num": 227, // 商品总库存 s1: '1215', // 规格类目 k_s 为 s1 的对应规格值 id
"collection_id": 2261, // 无规格商品skuId取collection_id否则取所选sku组合对应的id s2: '1193', // 规格类目 k_s 为 s2 的对应规格值 id
"none_sku": false, // 是否无规格商品 s3: '0', // 最多包含3个规格值为0表示不存在该规格
"messages": [{ // 商品留言 stock_num: 110 // 当前 sku 组合对应的库存
"datetime": "0", // 留言类型为time时是否含日期。“1”表示包含 }
"multiple": "0", // 留言类型为text时是否多行文本。“1”表示多行 ],
"name": "留言", // 留言名称 price: '1.00', // 默认价格(单位元)
"type": "text", // 留言类型可选id_no身份证, text, tel, date, time, email stock_num: 227, // 商品总库存
"required": "1" // 是否必填 “1”表示必填 collection_id: 2261, // 无规格商品 skuId 取 collection_id否则取所选 sku 组合对应的 id
}, ...], none_sku: false, // 是否无规格商品
"hide_stock": false // 是否隐藏剩余库存 messages: [
}, {
// 商品留言
datetime: '0', // 留言类型为 time 时,是否含日期。'1' 表示包含
multiple: '0', // 留言类型为 text 时,是否多行文本。'1' 表示多行
name: '留言', // 留言名称
type: 'text', // 留言类型,可选: id_no身份证, text, tel, date, time, email
required: '1' // 是否必填 '1' 表示必填
}
],
hide_stock: false // 是否隐藏剩余库存
}
``` ```
#### goods对象结构 #### goods 对象结构
```javascript ```javascript
"goods": { goods: {
// 商品标题 // 商品标题
"title": "测试商品", title: '测试商品',
// 默认商品sku缩略图 // 默认商品 sku 缩略图
"picture": "https:\/\/img.yzcdn.cn\/upload_files\/2017\/03\/16\/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2\/2\/w\/100\/h\/100\/q\/75\/format\/webp" picture: 'https://img.yzcdn.cn/1.jpg'
}, }
``` ```
#### 添加购物车和点击购买回调函数接收的skuData对象结构 #### 添加购物车和点击购买回调函数接收的 skuData 对象结构
```javascript ```javascript
skuData: { skuData: {
// 商品id // 商品 id
goodsId:"946755", goodsId: '946755',
// 留言信息 // 留言信息
messages: { messages: {
message_0:"12", message_0: '12',
message_1:"", message_1: ''
... // 有几个留言就有几条
}, },
// 另一种格式的留言key不同 // 另一种格式的留言key 不同
cartMessages: { cartMessages: {
'留言1': 'xxxx', '留言1': 'xxxx'
... // key是message的name
}, },
// 选择的商品数量 // 选择的商品数量
selectedNum:1, selectedNum: 1,
// 选择的sku组合 // 选择的 sku 组合
selectedSkuComb: { selectedSkuComb: {
id:2257, id: 2257,
price:100, price: 100,
s1:"30349", s1: '30349',
s2:"1193", s2: '1193',
s3:"0", s3: '0',
stock_num:111 stock_num: 111
} }
} }
``` ```

View File

@ -93,5 +93,38 @@ export default {
vanAddressList: { vanAddressList: {
address: 'Address', address: 'Address',
add: 'Add new address' add: 'Add new address'
},
vanSku: {
unavailable: 'The product is no longer available for purchase',
spec: 'Please select the full specification',
least: 'Choose at least one',
quota: quota => `Buy up to ${quota} items`,
inventory: 'Inventory shortage',
purchase: count => `You have purchased ${count} items`
},
vanSkuActions: {
cart: 'Add to cart',
buy: 'Buy'
},
vanSkuMessages: {
fill: 'Please fill',
number: 'Please fill in the correct number format message',
email: 'Please fill in the correct email message',
idcard: 'Please fill in the correct ID number message',
overlimit: 'not more than 200 words',
placeholder: {
'id_no': 'Idcard Number',
text: 'Text',
tel: 'Number',
email: 'Email',
date: 'Date',
time: 'Time',
textarea: 'Text'
}
},
vanSkuStepper: {
title: 'Quantity',
remain: count => `Remain ${count} items`,
quota: quota => `Buy up to ${quota} items`
} }
}; };

View File

@ -97,5 +97,38 @@ export default {
vanAddressList: { vanAddressList: {
address: '收货地址', address: '收货地址',
add: '新增收货地址' add: '新增收货地址'
},
vanSku: {
unavailable: '商品已经无法购买啦',
spec: '请选择完整的规格',
least: '至少选择一件',
quota: quota => `限购${quota}`,
inventory: '库存不足',
purchase: count => `您已购买${count}`
},
vanSkuActions: {
cart: '加入购物车',
buy: '立即购买'
},
vanSkuMessages: {
fill: '请填写',
number: '请填写正确的数字格式留言',
email: '请填写正确的邮箱',
'id_no': '请填写正确的身份证号码',
overlimit: '写的太多了不要超过200字',
placeholder: {
'id_no': '输入18位身份证号码',
text: '输入文本',
tel: '输入数字',
email: '输入邮箱',
date: '点击选择日期',
time: '点击选择时间',
textarea: '点击填写段落文本'
}
},
vanSkuStepper: {
title: '购买数量',
remain: count => `剩余${count}`,
quota: quota => `每人限购${quota}`
} }
}; };

View File

@ -1,14 +1,15 @@
<template> <template>
<div class="van-sku-actions"> <div class="van-sku-actions">
<van-button v-if="showAddCartBtn" bottomAction @click="onAddCartClicked">加入购物车</van-button> <van-button v-if="showAddCartBtn" bottomAction @click="onAddCartClicked">{{ $t('cart') }}</van-button>
<van-button type="primary" bottomAction @click="onBuyClicked">{{ buyText }}</van-button> <van-button type="primary" bottomAction @click="onBuyClicked">{{ buyText || $t('buy') }}</van-button>
</div> </div>
</template> </template>
<script> <script>
import VanButton from '../../button'; import VanButton from '../../button';
import { create } from '../../utils';
export default { export default create({
name: 'van-sku-actions', name: 'van-sku-actions',
components: { components: {
@ -16,12 +17,9 @@ export default {
}, },
props: { props: {
buyText: String,
skuEventBus: Object, skuEventBus: Object,
showAddCartBtn: Boolean, showAddCartBtn: Boolean
buyText: {
type: String,
default: '立即购买'
}
}, },
methods: { methods: {
@ -32,5 +30,5 @@ export default {
this.skuEventBus.$emit('sku:buy'); this.skuEventBus.$emit('sku:buy');
} }
} }
}; });
</script> </script>

View File

@ -12,7 +12,9 @@
</template> </template>
<script> <script>
export default { import { create } from '../../utils';
export default create({
name: 'van-sku-header', name: 'van-sku-header',
props: { props: {
@ -62,5 +64,5 @@ export default {
} }
} }
} }
}; });
</script> </script>

View File

@ -6,7 +6,7 @@
:key="`${goodsId}-${index}`" :key="`${goodsId}-${index}`"
:required="message.required == '1'" :required="message.required == '1'"
:label="message.name" :label="message.name"
:placeholder="placeholderMap.textarea" :placeholder="getPlaceholder('textarea')"
type="textarea" type="textarea"
v-model="messageValues[index]"> v-model="messageValues[index]">
</field> </field>
@ -14,7 +14,7 @@
:key="`${goodsId}-${index}`" :key="`${goodsId}-${index}`"
:required="message.required == '1'" :required="message.required == '1'"
:label="message.name" :label="message.name"
:placeholder="placeholderMap[message.type]" :placeholder="getPlaceholder(message.type)"
:type="getType(message)" :type="getType(message)"
v-model="messageValues[index]"> v-model="messageValues[index]">
</field> </field>
@ -23,13 +23,13 @@
</template> </template>
<script> <script>
import { create } from '../../utils';
import Field from '../../field'; import Field from '../../field';
import CellGroup from '../../cell-group'; import CellGroup from '../../cell-group';
import validateEmail from '../../utils/validate/email'; import validateEmail from '../../utils/validate/email';
import validateNumber from '../../utils/validate/number'; import validateNumber from '../../utils/validate/number';
import { DEFAULT_PLACEHOLDER_MAP } from '../constants';
export default { export default create({
name: 'van-sku-messages', name: 'van-sku-messages',
components: { components: {
@ -43,12 +43,6 @@ export default {
goodsId: [Number, String] goodsId: [Number, String]
}, },
data() {
return {
placeholderMap: Object.assign({}, DEFAULT_PLACEHOLDER_MAP, this.messagePlaceholderMap)
};
},
computed: { computed: {
internalMessages() { internalMessages() {
if (Object.prototype.toString.call(this.messages) === '[object Array]') { if (Object.prototype.toString.call(this.messages) === '[object Array]') {
@ -56,6 +50,7 @@ export default {
} }
return []; return [];
}, },
messageValues() { messageValues() {
const messageValues = []; const messageValues = [];
this.internalMessages.forEach((message, index) => { this.internalMessages.forEach((message, index) => {
@ -71,6 +66,7 @@ export default {
if (type === 'id_no') return 'text'; if (type === 'id_no') return 'text';
return datetime > 0 ? 'datetime-local' : type; return datetime > 0 ? 'datetime-local' : type;
}, },
getMessages() { getMessages() {
const messages = {}; const messages = {};
@ -83,6 +79,7 @@ export default {
return messages; return messages;
}, },
getCartMessages() { getCartMessages() {
const messages = {}; const messages = {};
@ -96,6 +93,11 @@ export default {
return messages; return messages;
}, },
getPlaceholder(key) {
return this.messagePlaceholderMap[key] || this.$t(`placeholder.${key}`);
},
validateMessages() { validateMessages() {
const values = this.messageValues; const values = this.messageValues;
@ -109,27 +111,26 @@ export default {
if (message.type === 'image') { if (message.type === 'image') {
continue; continue;
} else { } else {
return `请填写${message.name}`; return this.$t('fill') + message.name;
} }
} }
} else { } else {
if (message.type === 'tel' && !validateNumber(value)) { if (message.type === 'tel' && !validateNumber(value)) {
return '请填写正确的数字格式留言'; return this.$t('number');
} }
if (message.type === 'email' && !validateEmail(value)) { if (message.type === 'email' && !validateEmail(value)) {
return '请填写正确的邮箱'; return this.$t('email');
} }
if (message.type === 'id_no' && (value.length < 15 || value.length > 18)) { if (message.type === 'id_no' && (value.length < 15 || value.length > 18)) {
return '请填写正确的身份证号码'; return this.$t('id_no');
} }
} }
if (value.length > 200) { if (value.length > 200) {
return `${message.name} 写的太多了<br/>不要超过200字`; return `${message.name} ${this.$t('overlimit')}`;
} }
} }
} }
} }
});
};
</script> </script>

View File

@ -1,7 +1,7 @@
<template> <template>
<div class="van-sku-stepper-stock"> <div class="van-sku-stepper-stock">
<div class="van-sku-stepper-container"> <div class="van-sku-stepper-container">
<div class="van-sku__stepper-title">{{ stepperTitle }}</div> <div class="van-sku__stepper-title">{{ stepperTitle || $t('title') }}</div>
<stepper <stepper
class="van-sku__stepper" class="van-sku__stepper"
v-model="currentNum" v-model="currentNum"
@ -11,18 +11,19 @@
@overlimit="handleOverLimit"> @overlimit="handleOverLimit">
</stepper> </stepper>
</div> </div>
<div v-if="!hideStock" class="van-sku__stock">剩余{{ stock }}</div> <div v-if="!hideStock" class="van-sku__stock">{{ $t('remain', stock) }}</div>
<div v-if="quota > 0" class="van-sku__quota">每人限购{{ quota }}</div> <div v-if="quota > 0" class="van-sku__quota">{{ $t('quota', quota) }}</div>
</div> </div>
</template> </template>
<script> <script>
import { create } from '../../utils';
import Stepper from '../../stepper'; import Stepper from '../../stepper';
import { LIMIT_TYPE, DEFAULT_BUY_TEXT } from '../constants'; import { LIMIT_TYPE } from '../constants';
const { QUOTA_LIMIT, STOCK_LIMIT } = LIMIT_TYPE; const { QUOTA_LIMIT, STOCK_LIMIT } = LIMIT_TYPE;
export default { export default create({
name: 'van-sku-stepper', name: 'van-sku-stepper',
components: { components: {
@ -35,6 +36,7 @@ export default {
selectedSku: Object, selectedSku: Object,
selectedSkuComb: Object, selectedSkuComb: Object,
selectedNum: Number, selectedNum: Number,
stepperTitle: String,
quota: Number, quota: Number,
quotaUsed: Number, quotaUsed: Number,
hideStock: { hideStock: {
@ -44,10 +46,6 @@ export default {
disableStepperInput: { disableStepperInput: {
type: Boolean, type: Boolean,
default: false default: false
},
stepperTitle: {
type: String,
default: DEFAULT_BUY_TEXT
} }
}, },
@ -107,5 +105,5 @@ export default {
}); });
} }
} }
}; });
</script> </script>

View File

@ -2,17 +2,3 @@ export const LIMIT_TYPE = {
QUOTA_LIMIT: 0, QUOTA_LIMIT: 0,
STOCK_LIMIT: 1 STOCK_LIMIT: 1
}; };
export const DEFAULT_BUY_TEXT = '立即购买';
export const DEFAULT_STEPPER_TITLE = '购买数量';
export const DEFAULT_PLACEHOLDER_MAP = {
'id_no': '输入18位身份证号码',
text: '输入文本',
tel: '输入数字',
email: '输入邮箱',
date: '点击选择日期',
time: '点击选择时间',
textarea: '点击填写段落文本'
};

View File

@ -81,7 +81,7 @@ import SkuStepper from '../components/SkuStepper';
import SkuMessages from '../components/SkuMessages'; import SkuMessages from '../components/SkuMessages';
import SkuActions from '../components/SkuActions'; import SkuActions from '../components/SkuActions';
import { isAllSelected, getSkuComb, getSelectedSkuValues } from '../utils/skuHelper'; import { isAllSelected, getSkuComb, getSelectedSkuValues } from '../utils/skuHelper';
import { LIMIT_TYPE, DEFAULT_STEPPER_TITLE } from '../constants'; import { LIMIT_TYPE } from '../constants';
import { create } from '../../utils'; import { create } from '../../utils';
const { QUOTA_LIMIT } = LIMIT_TYPE; const { QUOTA_LIMIT } = LIMIT_TYPE;
@ -121,10 +121,7 @@ export default create({
default: true default: true
}, },
buyText: String, buyText: String,
stepperTitle: { stepperTitle: String,
type: String,
default: DEFAULT_STEPPER_TITLE
},
bodyOffsetTop: { bodyOffsetTop: {
type: Number, type: Number,
default: 200 default: 200
@ -260,7 +257,7 @@ export default create({
}, },
validateSku() { validateSku() {
if (this.selectedNum === 0) { if (this.selectedNum === 0) {
return '商品已经无法购买啦'; return this.$t('unavailable');
} }
if (this.isSkuCombSelected) { if (this.isSkuCombSelected) {
@ -268,7 +265,7 @@ export default create({
// sku // sku
return error; return error;
} else { } else {
return '请选择完整的规格'; return this.$t('spec');
} }
}, },
handleCloseClicked() { handleCloseClicked() {
@ -291,14 +288,14 @@ export default create({
}, },
handleOverLimit({ action, limitType, quota, quotaUsed }) { handleOverLimit({ action, limitType, quota, quotaUsed }) {
if (action === 'minus') { if (action === 'minus') {
Toast('至少选择一件'); Toast(this.$t('least'));
} else if (action === 'plus') { } else if (action === 'plus') {
if (limitType === QUOTA_LIMIT) { if (limitType === QUOTA_LIMIT) {
let msg = `限购${quota}`; let msg = this.$t('quota', quota);
if (quotaUsed > 0) msg += `您已购买${quotaUsed}`; if (quotaUsed > 0) msg += `${this.$t('purchase', quotaUsed)}`;
Toast(msg); Toast(msg);
} else { } else {
Toast('库存不足'); Toast(this.$t('inventory'));
} }
} }
}, },

View File

@ -216,7 +216,7 @@ describe('Sku', (done) => {
buyBtn.trigger('click'); buyBtn.trigger('click');
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
expect(toastText.textContent).to.equal('留言4 写的太多了<br/>不要超过200字'); expect(toastText.textContent).to.equal('留言4 写的太多了不要超过200字');
textarea.element.value = ''; textarea.element.value = '';
// 测试数字留言 // 测试数字留言