[improvement] Coupon: jsx (#2456)

This commit is contained in:
neverland 2019-01-07 15:50:51 +08:00 committed by GitHub
parent c6e1aaf5da
commit 7f853f694a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 471 additions and 479 deletions

View File

@ -1,21 +1,9 @@
<template>
<cell
:class="b()"
:title="title || $t('title')"
:value="value"
:border="border"
:is-link="editable"
:value-class="valueClass"
@click="$emit('click')"
/>
</template>
import { use } from '../utils';
import Cell from '../cell';
<script>
import create from '../utils/create';
export default create({
name: 'coupon-cell',
const [sfc, bem, t] = use('coupon-cell');
export default sfc({
model: {
prop: 'chosenCoupon'
},
@ -31,13 +19,13 @@ export default create({
type: Boolean,
default: true
},
chosenCoupon: {
type: Number,
default: -1
},
editable: {
type: Boolean,
default: true
},
chosenCoupon: {
type: Number,
default: -1
}
},
@ -49,12 +37,27 @@ export default create({
const value = coupon.denominations || coupon.value;
return `-${this.currency}${(value / 100).toFixed(2)}`;
}
return coupons.length === 0 ? this.$t('tips') : this.$t('count', coupons.length);
return coupons.length === 0 ? t('tips') : t('count', coupons.length);
},
valueClass() {
return this.coupons[this.chosenCoupon] ? 'van-coupon-cell--selected' : '';
}
},
render(h) {
return (
<Cell
class={bem()}
title={this.title || t('title')}
value={this.value}
border={this.border}
is-link={this.editable}
value-class={this.valueClass}
onClick={() => {
this.$emit('click');
}}
/>
);
}
});
</script>

View File

@ -0,0 +1,7 @@
@import '../style/var';
.van-coupon-cell {
&--selected {
color: @text-color;
}
}

View File

@ -1,81 +0,0 @@
<template>
<div :class="b({ disabled })">
<div :class="b('content')">
<div :class="b('head')">
<h2 v-html="faceAmount" />
<p v-text="conditionMessage" />
</div>
<div :class="b('body')">
<h2 v-text="data.name" />
<p v-text="validPeriod" />
<checkbox
v-if="chosen"
:value="true"
:class="b('corner')"
/>
</div>
</div>
<p
v-if="disabled && data.reason"
v-text="data.reason"
:class="b('reason')"
/>
</div>
</template>
<script>
import create from '../utils/create';
import Checkbox from '../checkbox';
function padZero(num) {
return (num < 10 ? '0' : '') + num;
}
function getDate(timeStamp) {
const date = new Date(timeStamp * 1000);
return `${date.getFullYear()}.${padZero(date.getMonth() + 1)}.${padZero(date.getDate())}`;
}
function formatDiscount(discount) {
return (discount / 10).toFixed(discount % 10 === 0 ? 0 : 1);
}
function formatAmount(amount) {
return (amount / 100).toFixed(amount % 100 === 0 ? 0 : amount % 10 === 0 ? 1 : 2);
}
export default create({
name: 'coupon-item',
props: {
data: Object,
chosen: Boolean,
disabled: Boolean,
currency: String
},
components: {
Checkbox
},
computed: {
validPeriod() {
return `${this.$t('valid')}${getDate(this.data.startAt)} - ${getDate(this.data.endAt)}`;
},
faceAmount() {
return this.data.denominations !== 0
? `<span>${this.currency}</span> ${formatAmount(this.data.denominations)}`
: this.data.discount !== 0
? this.$t('discount', formatDiscount(this.data.discount))
: '';
},
conditionMessage() {
let condition = this.data.originCondition;
condition = condition % 100 === 0 ? Math.round(condition / 100) : (condition / 100).toFixed(2);
return this.data.originCondition === 0 ? this.$t('unlimited') : this.$t('condition', condition);
}
}
});
</script>

View File

@ -0,0 +1,207 @@
import { use } from '../utils';
import Tab from '../tab';
import Tabs from '../tabs';
import Field from '../field';
import Button from '../button';
import Coupon from '../coupon';
const [sfc, bem, t] = use('coupon-list');
const EMPTY_IMAGE = 'https://img.yzcdn.cn/v2/image/wap/trade/new_order/empty@2x.png';
export default sfc({
model: {
prop: 'code'
},
props: {
code: String,
coupons: Array,
disabledCoupons: Array,
closeButtonText: String,
inputPlaceholder: String,
exchangeButtonText: String,
exchangeButtonLoading: Boolean,
exchangeButtonDisabled: Boolean,
exchangeMinLength: {
type: Number,
default: 1
},
chosenCoupon: {
type: Number,
default: -1
},
displayedCouponIndex: {
type: Number,
default: -1
},
showExchangeBar: {
type: Boolean,
default: true
},
showCloseButton: {
type: Boolean,
default: true
},
currency: {
type: String,
default: '¥'
}
},
data() {
return {
tab: 0,
winHeight: window.innerHeight,
currentCode: this.code || ''
};
},
computed: {
buttonDisabled() {
return (
!this.exchangeButtonLoading &&
(this.exchangeButtonDisabled ||
!this.currentCode ||
this.currentCode.length < this.exchangeMinLength)
);
},
title() {
return `${t('enable')} (${this.coupons.length})`;
},
disabledTitle() {
return `${t('disabled')} (${this.disabledCoupons.length})`;
},
listStyle() {
return {
height: this.winHeight - (this.showExchangeBar ? 140 : 94) + 'px'
};
}
},
watch: {
code(code) {
this.currentCode = code;
},
currentCode(code) {
this.$emit('input', code);
},
displayedCouponIndex(val) {
this.scrollToShowCoupon(val);
}
},
mounted() {
this.scrollToShowCoupon(this.displayedCouponIndex);
},
methods: {
onClickExchangeButton() {
this.$emit('exchange', this.currentCode);
// auto clear currentCode when not use v-model
if (!this.code) {
this.currentCode = '';
}
},
// scroll to show specific coupon
scrollToShowCoupon(index) {
if (index === -1) {
return;
}
this.$nextTick(() => {
const { card, list } = this.$refs;
/* istanbul ignore next */
if (list && card && card[index]) {
list.scrollTop = card[index].$el.offsetTop - 100;
}
});
}
},
render(h) {
const ExchangeBar = this.showExchangeBar && (
<Field
v-model={this.currentCode}
clearable
border={false}
class={bem('field')}
placeholder={this.inputPlaceholder || t('placeholder')}
maxlength="20"
>
<Button
slot="button"
size="small"
type="danger"
class={bem('exchange')}
text={this.exchangeButtonText || t('exchange')}
loading={this.exchangeButtonLoading}
disabled={this.buttonDisabled}
onClick={this.onClickExchangeButton}
/>
</Field>
);
const Empty = (
<div class={bem('empty')}>
<img src={EMPTY_IMAGE} />
<p>{t('empty')}</p>
</div>
);
const onChange = index => () => this.$emit('change', index);
const CouponTab = (
<Tab title={this.title}>
<div class={bem('list')} style={this.listStyle}>
{this.coupons.map((coupon, index) => (
<Coupon
ref="card"
key={coupon.id}
coupon={coupon}
currency={this.currency}
chosen={index === this.chosenCoupon}
nativeOnClick={onChange(index)}
/>
))}
{!this.coupons.length && Empty}
</div>
</Tab>
);
const DisabledCouponTab = (
<Tab title={this.disabledTitle}>
<div class={bem('list')} style={this.listStyle}>
{this.disabledCoupons.map(coupon => (
<Coupon disabled key={coupon.id} coupon={coupon} currency={this.currency} />
))}
{!this.disabledCoupons.length && Empty}
</div>
</Tab>
);
return (
<div class={bem()}>
{ExchangeBar}
<Tabs v-model={this.tab} class={bem('tab')} line-width={120}>
{CouponTab}
{DisabledCouponTab}
</Tabs>
<Button
v-show={this.showCloseButton}
size="large"
class={bem('close')}
text={this.closeButtonText || t('close')}
onClick={onChange(-1)}
/>
</div>
);
}
});

View File

@ -1,148 +1,47 @@
@import '../style/var';
@import '../style/mixins/ellipsis';
.van-coupon {
&-cell--selected {
color: @text-color;
.van-coupon-list {
height: 100%;
position: relative;
background-color: @background-color;
&__field {
padding: 7px 15px;
}
&-list {
height: 100%;
position: relative;
background-color: @background-color;
&__field {
padding: 7px 15px;
}
&__exchange {
height: 32px;
line-height: 30px;
}
&__list {
overflow-y: auto;
padding: 15px 0;
box-sizing: border-box;
-webkit-overflow-scrolling: touch;
}
&__close {
left: 0;
bottom: 0;
position: absolute;
font-weight: 500;
}
&__empty {
padding-top: 100px;
text-align: center;
p {
color: @gray-dark;
margin: 15px 0;
font-size: 14px;
line-height: 20px;
}
img {
width: 80px;
height: 84px;
}
}
&__exchange {
height: 32px;
line-height: 30px;
}
&-item {
overflow: hidden;
border-radius: 4px;
margin: 0 15px 15px;
background-color: @white;
box-shadow: 0 0 4px rgba(0, 0, 0, 0.1);
&__list {
overflow-y: auto;
padding: 15px 0;
box-sizing: border-box;
-webkit-overflow-scrolling: touch;
}
&:active {
background-color: @active-color;
}
&__close {
left: 0;
bottom: 0;
position: absolute;
font-weight: 500;
}
&__content {
display: flex;
height: 100px;
padding: 24px 0 0 15px;
box-sizing: border-box;
}
p,
h2 {
margin: 0;
.ellipsis();
}
h2 {
height: 34px;
font-weight: 500;
line-height: 34px;
}
&__empty {
padding-top: 100px;
text-align: center;
p {
font-size: 12px;
line-height: 16px;
color: @gray-dark;
margin: 15px 0;
font-size: 14px;
line-height: 20px;
}
&__head {
min-width: 90px;
h2 {
color: @red;
font-size: 24px;
span {
font-size: 50%;
}
}
}
&__body {
flex: 1;
position: relative;
border-radius: 0 4px 4px 0;
h2 {
font-size: 16px;
}
}
&__corner {
top: 16px;
right: 15px;
position: absolute;
.van-icon {
border-color: @red;
background-color: @red;
}
}
&__reason {
padding: 7px 15px;
border-top: 1px dashed @border-color;
background-color: @background-color-light;
}
&--disabled {
&:active {
background-color: @white;
}
.van-coupon-item__content {
height: 90px;
}
p,
h2,
span {
color: @gray-dark;
}
img {
width: 80px;
height: 84px;
}
}
}

View File

@ -1,218 +0,0 @@
<template>
<div :class="b()">
<field
v-if="showExchangeBar"
v-model="currentCode"
clearable
:border="false"
:class="b('field')"
:placeholder="inputPlaceholder || $t('placeholder')"
:maxlength="20"
>
<van-button
slot="button"
size="small"
type="danger"
:class="b('exchange')"
:text="exchangeButtonText || $t('exchange')"
:loading="exchangeButtonLoading"
:disabled="buttonDisabled"
@click="onClickExchangeButton"
/>
</field>
<tabs
v-model="tab"
:class="b('tab')"
:line-width="120"
>
<tab :title="title">
<div
:class="b('list')"
:style="listStyle"
>
<coupon-item
ref="card"
v-for="(item, index) in coupons"
:key="item.id || item.name"
:data="item"
:currency="currency"
:chosen="index === chosenCoupon"
@click.native="$emit('change', index)"
/>
<div
v-if="!coupons.length"
:class="b('empty')"
>
<img src="https://img.yzcdn.cn/v2/image/wap/trade/new_order/empty@2x.png">
<p v-text="$t('empty')" />
</div>
</div>
</tab>
<tab :title="disabledTitle">
<div
:class="b('list')"
:style="listStyle"
>
<coupon-item
disabled
v-for="item in disabledCoupons"
:key="item.id || item.name"
:data="item"
:currency="currency"
/>
<div
v-if="!disabledCoupons.length"
:class="b('empty')"
>
<img src="https://img.yzcdn.cn/v2/image/wap/trade/new_order/empty@2x.png">
<p v-text="$t('empty')" />
</div>
</div>
</tab>
</tabs>
<van-button
v-show="showCloseButton"
size="large"
:class="b('close')"
:text="closeButtonText || $t('close')"
@click="$emit('change', -1)"
/>
</div>
</template>
<script>
import create from '../utils/create';
import CouponItem from './Item';
import Field from '../field';
import VanButton from '../button';
import Tab from '../tab';
import Tabs from '../tabs';
export default create({
name: 'coupon-list',
components: {
Tab,
Tabs,
Field,
VanButton,
CouponItem
},
model: {
prop: 'code'
},
props: {
code: String,
coupons: Array,
disabledCoupons: Array,
closeButtonText: String,
inputPlaceholder: String,
exchangeButtonText: String,
exchangeButtonLoading: Boolean,
exchangeButtonDisabled: Boolean,
exchangeMinLength: {
type: Number,
default: 1
},
chosenCoupon: {
type: Number,
default: -1
},
displayedCouponIndex: {
type: Number,
default: -1
},
showExchangeBar: {
type: Boolean,
default: true
},
showCloseButton: {
type: Boolean,
default: true
},
currency: {
type: String,
default: '¥'
}
},
data() {
return {
tab: 0,
winHeight: window.innerHeight,
currentCode: this.code || ''
};
},
computed: {
buttonDisabled() {
return (
!this.exchangeButtonLoading &&
(this.exchangeButtonDisabled || !this.currentCode ||
this.currentCode.length < this.exchangeMinLength)
);
},
title() {
return `${this.$t('enable')} (${this.coupons.length})`;
},
disabledTitle() {
return `${this.$t('disabled')} (${this.disabledCoupons.length})`;
},
listStyle() {
return {
height: this.winHeight - (this.showExchangeBar ? 140 : 94) + 'px'
};
}
},
watch: {
code(code) {
this.currentCode = code;
},
currentCode(code) {
this.$emit('input', code);
},
displayedCouponIndex(val) {
this.scrollToShowCoupon(val);
}
},
mounted() {
this.scrollToShowCoupon(this.displayedCouponIndex);
},
methods: {
onClickExchangeButton() {
this.$emit('exchange', this.currentCode);
// auto clear currentCode when not use v-model
if (!this.code) {
this.currentCode = '';
}
},
// scroll to show specific coupon
scrollToShowCoupon(index) {
if (index === -1) {
return;
}
this.$nextTick(() => {
const { card, list } = this.$refs;
/* istanbul ignore next */
if (list && card && card[index]) {
list.scrollTop = card[index].$el.offsetTop - 100;
}
});
}
}
});
</script>

View File

@ -1,4 +1,4 @@
import CouponList from '../../coupon-list';
import CouponList from '..';
import CouponCell from '../../coupon-cell';
import { mount } from '../../../test/utils';

74
packages/coupon/index.js Normal file
View File

@ -0,0 +1,74 @@
import { use } from '../utils';
import Checkbox from '../checkbox';
const [sfc, bem, t] = use('coupon');
function padZero(num) {
return (num < 10 ? '0' : '') + num;
}
function getDate(timeStamp) {
const date = new Date(timeStamp * 1000);
return `${date.getFullYear()}.${padZero(date.getMonth() + 1)}.${padZero(date.getDate())}`;
}
function formatDiscount(discount) {
return (discount / 10).toFixed(discount % 10 === 0 ? 0 : 1);
}
function formatAmount(amount) {
return (amount / 100).toFixed(amount % 100 === 0 ? 0 : amount % 10 === 0 ? 1 : 2);
}
export default sfc({
props: {
coupon: Object,
chosen: Boolean,
disabled: Boolean,
currency: {
type: String,
default: '¥'
}
},
computed: {
validPeriod() {
return `${t('valid')}${getDate(this.coupon.startAt)} - ${getDate(this.coupon.endAt)}`;
},
faceAmount() {
return this.coupon.denominations
? `<span>${this.currency}</span> ${formatAmount(this.coupon.denominations)}`
: this.coupon.discount
? t('discount', formatDiscount(this.coupon.discount))
: '';
},
conditionMessage() {
let condition = this.coupon.originCondition;
condition = condition % 100 === 0 ? Math.round(condition / 100) : (condition / 100).toFixed(2);
return condition === 0 ? t('unlimited') : t('condition', condition);
}
},
render(h) {
const { coupon, disabled } = this;
return (
<div class={bem({ disabled: this.disabled })}>
<div class={bem('content')}>
<div class={bem('head')}>
<h2 domPropsInnerHTML={this.faceAmount} />
<p>{this.conditionMessage}</p>
</div>
<div class={bem('body')}>
<h2>{coupon.name}</h2>
<p>{this.validPeriod}</p>
{this.chosen && <Checkbox value={true} class={bem('corner')} />}
</div>
</div>
{(disabled && coupon.reason) && <p class={bem('reason')}>{coupon.reason}</p>}
</div>
);
}
});

View File

@ -0,0 +1,96 @@
@import '../style/var';
@import '../style/mixins/ellipsis';
.van-coupon {
overflow: hidden;
border-radius: 4px;
margin: 0 15px 15px;
background-color: @white;
box-shadow: 0 0 4px rgba(0, 0, 0, 0.1);
&:active {
background-color: @active-color;
}
&__content {
display: flex;
height: 100px;
padding: 24px 0 0 15px;
box-sizing: border-box;
}
p,
h2 {
margin: 0;
.ellipsis();
}
h2 {
height: 34px;
font-weight: 500;
line-height: 34px;
}
p {
font-size: 12px;
line-height: 16px;
color: @gray-dark;
}
&__head {
min-width: 90px;
h2 {
color: @red;
font-size: 24px;
span {
font-size: 50%;
}
}
}
&__body {
flex: 1;
position: relative;
border-radius: 0 4px 4px 0;
h2 {
font-size: 16px;
}
}
&__corner {
top: 16px;
right: 15px;
position: absolute;
.van-icon {
border-color: @red;
background-color: @red;
}
}
&__reason {
padding: 7px 15px;
border-top: 1px dashed @border-color;
background-color: @background-color-light;
}
&--disabled {
&:active {
background-color: @white;
}
.van-coupon-item__content {
height: 90px;
}
p,
h2,
span {
color: @gray-dark;
}
}
}

View File

@ -19,6 +19,7 @@ import CollapseItem from './collapse-item';
import ContactCard from './contact-card';
import ContactEdit from './contact-edit';
import ContactList from './contact-list';
import Coupon from './coupon';
import CouponCell from './coupon-cell';
import CouponList from './coupon-list';
import DatetimePicker from './datetime-picker';
@ -93,6 +94,7 @@ const components = [
ContactCard,
ContactEdit,
ContactList,
Coupon,
CouponCell,
CouponList,
DatetimePicker,
@ -178,6 +180,7 @@ export {
ContactCard,
ContactEdit,
ContactList,
Coupon,
CouponCell,
CouponList,
DatetimePicker,

View File

@ -5,7 +5,7 @@
/* base */
@import './style/base';
// /* common components */
/* common components */
@import './col/index';
@import './row/index';
@import './badge/index';
@ -63,6 +63,8 @@
@import './contact-card/index';
@import './contact-list/index';
@import './contact-edit/index';
@import './coupon/index';
@import './coupon-cell/index';
@import './coupon-list/index';
@import './goods-action/index';
@import './goods-action-big-btn/index';

View File

@ -28,6 +28,12 @@ export default {
vanSubmitBar: {
label: 'Total'
},
vanCoupon: {
valid: 'Valid',
unlimited: 'Unlimited',
discount: discount => `${discount * 10}% off`,
condition: condition => `At least ${condition}`
},
vanCouponCell: {
title: 'Coupon',
tips: 'Select coupon',
@ -41,12 +47,6 @@ export default {
disabled: 'Unavailable',
placeholder: 'Coupon code'
},
vanCouponItem: {
valid: 'Valid',
unlimited: 'Unlimited',
discount: discount => `${discount * 10}% off`,
condition: condition => `At least ${condition}`
},
vanAddressEdit: {
area: 'Area',
postal: 'Postal',

View File

@ -28,6 +28,12 @@ export default {
vanSubmitBar: {
label: '合计:'
},
vanCoupon: {
valid: '有效期',
unlimited: '无使用门槛',
discount: discount => `${discount}`,
condition: (condition) => `${condition}元可用`
},
vanCouponCell: {
title: '优惠券',
tips: '使用优惠',
@ -41,12 +47,6 @@ export default {
disabled: '不可使用优惠券',
placeholder: '请输入优惠码'
},
vanCouponItem: {
valid: '有效期',
unlimited: '无使用门槛',
discount: discount => `${discount}`,
condition: (condition) => `${condition}元可用`
},
vanAddressEdit: {
area: '地区',
postal: '邮政编码',

View File

@ -28,6 +28,12 @@ export default {
vanSubmitBar: {
label: '合計:'
},
vanCoupon: {
valid: '有效期',
unlimited: '無使用門檻',
discount: discount => `${discount}`,
condition: (condition) => `滿${condition}元可用`
},
vanCouponCell: {
title: '優惠券',
tips: '使用優惠',
@ -41,12 +47,6 @@ export default {
disabled: '不可使用優惠券',
placeholder: '請輸入優惠碼'
},
vanCouponItem: {
valid: '有效期',
unlimited: '無使用門檻',
discount: discount => `${discount}`,
condition: (condition) => `滿${condition}元可用`
},
vanAddressEdit: {
area: '地區',
postal: '郵政編碼',

View File

@ -28,6 +28,12 @@ export default {
vanSubmitBar: {
label: '合計:'
},
vanCoupon: {
valid: '有效期限',
unlimited: '無使用門檻',
discount: discount => `${discount}`,
condition: (condition) => `滿${condition}元可用`
},
vanCouponCell: {
title: '優惠券',
tips: '使用優惠',
@ -41,12 +47,6 @@ export default {
disabled: '不可使用優惠券',
placeholder: '請輸入優惠代碼'
},
vanCouponItem: {
valid: '有效期限',
unlimited: '無使用門檻',
discount: discount => `${discount}`,
condition: (condition) => `滿${condition}元可用`
},
vanAddressEdit: {
area: '地區',
postal: '郵遞區號',