diff --git a/packages/coupon-cell/index.vue b/packages/coupon-cell/index.js
similarity index 58%
rename from packages/coupon-cell/index.vue
rename to packages/coupon-cell/index.js
index 64fd71699..f9e38ac9b 100644
--- a/packages/coupon-cell/index.vue
+++ b/packages/coupon-cell/index.js
@@ -1,21 +1,9 @@
-
- |
-
+import { use } from '../utils';
+import Cell from '../cell';
-
diff --git a/packages/coupon-cell/index.less b/packages/coupon-cell/index.less
new file mode 100644
index 000000000..10430b38d
--- /dev/null
+++ b/packages/coupon-cell/index.less
@@ -0,0 +1,7 @@
+@import '../style/var';
+
+.van-coupon-cell {
+ &--selected {
+ color: @text-color;
+ }
+}
diff --git a/packages/coupon-list/Item.vue b/packages/coupon-list/Item.vue
deleted file mode 100644
index 183b07078..000000000
--- a/packages/coupon-list/Item.vue
+++ /dev/null
@@ -1,81 +0,0 @@
-
-
-
-
-
diff --git a/packages/coupon-list/index.js b/packages/coupon-list/index.js
new file mode 100644
index 000000000..a38e9f445
--- /dev/null
+++ b/packages/coupon-list/index.js
@@ -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 && (
+
+
+
+ );
+
+ const Empty = (
+
+

+
{t('empty')}
+
+ );
+
+ const onChange = index => () => this.$emit('change', index);
+
+ const CouponTab = (
+
+
+ {this.coupons.map((coupon, index) => (
+
+ ))}
+ {!this.coupons.length && Empty}
+
+
+ );
+
+ const DisabledCouponTab = (
+
+
+ {this.disabledCoupons.map(coupon => (
+
+ ))}
+ {!this.disabledCoupons.length && Empty}
+
+
+ );
+
+ return (
+
+ {ExchangeBar}
+
+ {CouponTab}
+ {DisabledCouponTab}
+
+
+
+ );
+ }
+});
diff --git a/packages/coupon-list/index.less b/packages/coupon-list/index.less
index c97e37891..9e03f2f62 100644
--- a/packages/coupon-list/index.less
+++ b/packages/coupon-list/index.less
@@ -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;
}
}
}
diff --git a/packages/coupon-list/index.vue b/packages/coupon-list/index.vue
deleted file mode 100644
index 9402ab56d..000000000
--- a/packages/coupon-list/index.vue
+++ /dev/null
@@ -1,218 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-

-
-
-
-
-
-
-
-
-

-
-
-
-
-
-
-
-
-
-
diff --git a/packages/coupon-list/test/index.spec.js b/packages/coupon-list/test/index.spec.js
index c5438a4e9..4a8792862 100644
--- a/packages/coupon-list/test/index.spec.js
+++ b/packages/coupon-list/test/index.spec.js
@@ -1,4 +1,4 @@
-import CouponList from '../../coupon-list';
+import CouponList from '..';
import CouponCell from '../../coupon-cell';
import { mount } from '../../../test/utils';
diff --git a/packages/coupon/index.js b/packages/coupon/index.js
new file mode 100644
index 000000000..59a1f73ea
--- /dev/null
+++ b/packages/coupon/index.js
@@ -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
+ ? `${this.currency} ${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 (
+
+
+
+
+
{this.conditionMessage}
+
+
+
{coupon.name}
+
{this.validPeriod}
+ {this.chosen &&
}
+
+
+ {(disabled && coupon.reason) &&
{coupon.reason}
}
+
+ );
+ }
+});
diff --git a/packages/coupon/index.less b/packages/coupon/index.less
new file mode 100644
index 000000000..216e1f458
--- /dev/null
+++ b/packages/coupon/index.less
@@ -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;
+ }
+ }
+}
diff --git a/packages/index.js b/packages/index.js
index 1a645c868..e40789709 100644
--- a/packages/index.js
+++ b/packages/index.js
@@ -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,
diff --git a/packages/index.less b/packages/index.less
index a8f6db59e..d9727e87d 100644
--- a/packages/index.less
+++ b/packages/index.less
@@ -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';
diff --git a/packages/locale/lang/en-US.js b/packages/locale/lang/en-US.js
index 32cdb3e5a..7a4b2be50 100644
--- a/packages/locale/lang/en-US.js
+++ b/packages/locale/lang/en-US.js
@@ -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',
diff --git a/packages/locale/lang/zh-CN.js b/packages/locale/lang/zh-CN.js
index b2c3c03d6..39167617a 100644
--- a/packages/locale/lang/zh-CN.js
+++ b/packages/locale/lang/zh-CN.js
@@ -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: '邮政编码',
diff --git a/packages/locale/lang/zh-HK.js b/packages/locale/lang/zh-HK.js
index 88d18146c..515b2f0e4 100644
--- a/packages/locale/lang/zh-HK.js
+++ b/packages/locale/lang/zh-HK.js
@@ -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: '郵政編碼',
diff --git a/packages/locale/lang/zh-TW.js b/packages/locale/lang/zh-TW.js
index 71fe92857..bd78cbb18 100644
--- a/packages/locale/lang/zh-TW.js
+++ b/packages/locale/lang/zh-TW.js
@@ -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: '郵遞區號',