feat(Coupon): support for checkbox usage (#12744)

This commit is contained in:
凌览 2024-04-06 21:32:45 +08:00 committed by GitHub
parent a830c230af
commit 3a524e831b
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 196 additions and 27 deletions

View File

@ -53,6 +53,7 @@ export function initDemoLocale() {
disabled: '禁用状态', disabled: '禁用状态',
uneditable: '不可编辑', uneditable: '不可编辑',
basicUsage: '基础用法', basicUsage: '基础用法',
checkboxUsage: '多选用法',
usingUrl: '使用图片 URL', usingUrl: '使用图片 URL',
advancedUsage: '高级用法', advancedUsage: '高级用法',
loadingStatus: '加载状态', loadingStatus: '加载状态',
@ -80,6 +81,7 @@ export function initDemoLocale() {
disabled: 'Disabled', disabled: 'Disabled',
uneditable: 'Uneditable', uneditable: 'Uneditable',
basicUsage: 'Basic Usage', basicUsage: 'Basic Usage',
checkboxUsage: 'Checkbox Usage',
usingUrl: 'Using URL', usingUrl: 'Using URL',
advancedUsage: 'Advanced Usage', advancedUsage: 'Advanced Usage',
loadingStatus: 'Loading', loadingStatus: 'Loading',

View File

@ -6,7 +6,6 @@ import {
truthProp, truthProp,
makeArrayProp, makeArrayProp,
makeStringProp, makeStringProp,
makeNumericProp,
createNamespace, createNamespace,
} from '../utils'; } from '../utils';
@ -24,27 +23,47 @@ export const couponCellProps = {
editable: truthProp, editable: truthProp,
coupons: makeArrayProp<CouponInfo>(), coupons: makeArrayProp<CouponInfo>(),
currency: makeStringProp('¥'), currency: makeStringProp('¥'),
chosenCoupon: makeNumericProp(-1), chosenCoupon: {
type: [Number, Array],
default: -1,
},
}; };
export type CouponCellProps = ExtractPropTypes<typeof couponCellProps>; export type CouponCellProps = ExtractPropTypes<typeof couponCellProps>;
function formatValue({ coupons, chosenCoupon, currency }: CouponCellProps) { function formatValue({ coupons, chosenCoupon, currency }: CouponCellProps) {
const coupon = coupons[+chosenCoupon]; const getValue = (coupon: CouponInfo) => {
if (coupon) {
let value = 0; let value = 0;
const { value: couponValue, denominations } = coupon;
if (isDef(coupon.value)) { if (isDef(coupon.value)) {
({ value } = coupon); value = couponValue;
} else if (isDef(coupon.denominations)) { } else if (isDef(coupon.denominations)) {
value = coupon.denominations; value = denominations as number;
} }
return value;
};
return `-${currency} ${(value / 100).toFixed(2)}`; let value = 0,
isExist = false;
if (Array.isArray(chosenCoupon)) {
(chosenCoupon as CouponInfo[]).forEach((i) => {
const coupon = coupons[+i];
if (coupon) {
isExist = true;
value += getValue(coupon);
}
});
} else {
const coupon = coupons[+chosenCoupon];
if (coupon) {
isExist = true;
value = getValue(coupon);
}
} }
if (!isExist) {
return coupons.length === 0 ? t('noCoupon') : t('count', coupons.length); return coupons.length === 0 ? t('noCoupon') : t('count', coupons.length);
}
return `-${currency} ${(value / 100).toFixed(2)}`;
} }
export default defineComponent({ export default defineComponent({
@ -54,7 +73,9 @@ export default defineComponent({
setup(props) { setup(props) {
return () => { return () => {
const selected = props.coupons[+props.chosenCoupon]; const selected = Array.isArray(props.chosenCoupon)
? props.chosenCoupon.length
: props.coupons[+props.chosenCoupon];
return ( return (
<Cell <Cell
class={bem()} class={bem()}

View File

@ -4,6 +4,7 @@ import {
computed, computed,
nextTick, nextTick,
onMounted, onMounted,
unref,
defineComponent, defineComponent,
type ExtractPropTypes, type ExtractPropTypes,
} from 'vue'; } from 'vue';
@ -37,7 +38,7 @@ export const couponListProps = {
currency: makeStringProp('¥'), currency: makeStringProp('¥'),
showCount: truthProp, showCount: truthProp,
emptyImage: String, emptyImage: String,
chosenCoupon: makeNumberProp(-1), chosenCoupon: [Number, Array],
enabledTitle: String, enabledTitle: String,
disabledTitle: String, disabledTitle: String,
disabledCoupons: makeArrayProp<CouponInfo>(), disabledCoupons: makeArrayProp<CouponInfo>(),
@ -137,6 +138,19 @@ export default defineComponent({
const count = props.showCount ? ` (${coupons.length})` : ''; const count = props.showCount ? ` (${coupons.length})` : '';
const title = (props.enabledTitle || t('enable')) + count; const title = (props.enabledTitle || t('enable')) + count;
const getChosenCoupon = (
chosenCoupon: Array<any> = [],
value: number = 0,
): Array<any> => {
const unrefChosenCoupon = unref(chosenCoupon);
const index = unrefChosenCoupon.indexOf(value);
if (index === -1) {
return [...unrefChosenCoupon, value];
}
unrefChosenCoupon.splice(index, 1);
return [...unrefChosenCoupon];
};
return ( return (
<Tab title={title}> <Tab title={title}>
<div <div
@ -148,9 +162,20 @@ export default defineComponent({
key={coupon.id} key={coupon.id}
ref={setCouponRefs(index)} ref={setCouponRefs(index)}
coupon={coupon} coupon={coupon}
chosen={index === props.chosenCoupon} chosen={
Array.isArray(props.chosenCoupon)
? props.chosenCoupon.includes(index)
: index === props.chosenCoupon
}
currency={props.currency} currency={props.currency}
onClick={() => emit('change', index)} onClick={() =>
emit(
'change',
Array.isArray(props.chosenCoupon)
? getChosenCoupon(props.chosenCoupon, index)
: index,
)
}
/> />
))} ))}
{!coupons.length && renderEmpty()} {!coupons.length && renderEmpty()}
@ -210,15 +235,21 @@ export default defineComponent({
{renderDisabledTab()} {renderDisabledTab()}
</Tabs> </Tabs>
<div class={bem('bottom')}> <div class={bem('bottom')}>
<Button {slots['list-button'] ? (
v-show={props.showCloseButton} slots['list-button']()
round ) : (
block <Button
type="primary" v-show={props.showCloseButton}
class={bem('close')} round
text={props.closeButtonText || t('close')} block
onClick={() => emit('change', -1)} type="primary"
/> class={bem('close')}
text={props.closeButtonText || t('close')}
onClick={() =>
emit('change', Array.isArray(props.chosenCoupon) ? [] : -1)
}
/>
)}
</div> </div>
</div> </div>
); );

View File

@ -104,7 +104,7 @@ export default {
| Attribute | Description | Type | Default | | Attribute | Description | Type | Default |
| --- | --- | --- | --- | | --- | --- | --- | --- |
| v-model | Current exchange code | _string_ | - | | v-model | Current exchange code | _string_ | - |
| chosen-coupon | Index of chosen coupon | _number_ | `-1` | | chosen-coupon | Index of chosen coupon,support multiple selection(type `[]`) | _number\|number[]_ | `-1` |
| coupons | Coupon list | _CouponInfo[]_ | `[]` | | coupons | Coupon list | _CouponInfo[]_ | `[]` |
| disabled-coupons | Disabled coupon list | _CouponInfo[]_ | `[]` | | disabled-coupons | Disabled coupon list | _CouponInfo[]_ | `[]` |
| enabled-title | Title of coupon list | _string_ | `Available` | | enabled-title | Title of coupon list | _string_ | `Available` |
@ -133,6 +133,7 @@ export default {
| -------------------- | ------------------------------- | | -------------------- | ------------------------------- |
| list-footer | Coupon list bottom | | list-footer | Coupon list bottom |
| disabled-list-footer | Unavailable coupons list bottom | | disabled-list-footer | Unavailable coupons list bottom |
| list-button | Customize the bottom button |
### Data Structure of CouponInfo ### Data Structure of CouponInfo

View File

@ -104,7 +104,7 @@ export default {
| 参数 | 说明 | 类型 | 默认值 | | 参数 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- | | --- | --- | --- | --- |
| v-model:code | 当前输入的兑换码 | _string_ | - | | v-model:code | 当前输入的兑换码 | _string_ | - |
| chosen-coupon | 当前选中优惠券的索引 | _number_ | `-1` | | chosen-coupon | 当前选中优惠券的索引,支持多选(类型为`[]` | _number\|number[]_ | `-1` |
| coupons | 可用优惠券列表 | _CouponInfo[]_ | `[]` | | coupons | 可用优惠券列表 | _CouponInfo[]_ | `[]` |
| disabled-coupons | 不可用优惠券列表 | _CouponInfo[]_ | `[]` | | disabled-coupons | 不可用优惠券列表 | _CouponInfo[]_ | `[]` |
| enabled-title | 可用优惠券列表标题 | _string_ | `可使用优惠券` | | enabled-title | 可用优惠券列表标题 | _string_ | `可使用优惠券` |
@ -135,6 +135,7 @@ export default {
| -------------------- | -------------------- | | -------------------- | -------------------- |
| list-footer | 优惠券列表底部 | | list-footer | 优惠券列表底部 |
| disabled-list-footer | 不可用优惠券列表底部 | | disabled-list-footer | 不可用优惠券列表底部 |
| list-button | 自定义底部按钮 |
### CouponInfo 数据结构 ### CouponInfo 数据结构

View File

@ -1,8 +1,9 @@
<script setup lang="ts"> <script setup lang="ts">
import VanCouponCell from '../../coupon-cell'; import VanCouponCell from '../../coupon-cell';
import VanPopup from '../../popup'; import VanPopup from '../../popup';
import VanButton from '../../button';
import VanCouponList from '..'; import VanCouponList from '..';
import { ref, computed } from 'vue'; import { ref, computed, unref } from 'vue';
import { useTranslate } from '../../../docs/site'; import { useTranslate } from '../../../docs/site';
import { CouponInfo } from '../../coupon'; import { CouponInfo } from '../../coupon';
import { showToast } from '../../toast'; import { showToast } from '../../toast';
@ -30,7 +31,10 @@ const getRandomId = (max = 999999) =>
String(Math.floor(Math.random() * max) + 1); String(Math.floor(Math.random() * max) + 1);
const showList = ref(false); const showList = ref(false);
const showListArray = ref(false);
const chosenCoupon = ref(-1); const chosenCoupon = ref(-1);
const chosenCouponArray = ref([]);
const chosenCouponArrayResult = ref([]);
const exchangedCoupons = ref<CouponInfo[]>([]); const exchangedCoupons = ref<CouponInfo[]>([]);
const coupon = computed(() => ({ const coupon = computed(() => ({
@ -84,6 +88,15 @@ const onChange = (index: number) => {
chosenCoupon.value = index; chosenCoupon.value = index;
}; };
const onChangeArray = (chosenCoupon: []) => {
chosenCouponArray.value = chosenCoupon;
};
const onSubmit = () => {
showListArray.value = false;
chosenCouponArrayResult.value = unref(chosenCouponArray);
};
const onExchange = () => { const onExchange = () => {
showToast(t('exchange')); showToast(t('exchange'));
exchangedCoupons.value.push({ exchangedCoupons.value.push({
@ -115,4 +128,38 @@ const onExchange = () => {
/> />
</van-popup> </van-popup>
</demo-block> </demo-block>
<demo-block :title="t('checkboxUsage')">
<van-coupon-cell
:coupons="coupons"
:chosen-coupon="chosenCouponArrayResult"
@click="showListArray = true"
/>
<van-popup
v-model:show="showListArray"
round
position="bottom"
style="height: 90%; padding-top: 4px"
>
<van-coupon-list
:coupons="coupons"
:chosen-coupon="chosenCouponArray"
:disabled-coupons="disabledCoupons"
:show-close-button="false"
@change="onChangeArray"
@exchange="onExchange"
>
<template #list-button>
<van-button
round
block
type="primary"
text="确定"
style="height: 40px"
@click="onSubmit"
/>
</template>
</van-coupon-list>
</van-popup>
</demo-block>
</template> </template>

View File

@ -1,6 +1,36 @@
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
exports[`should render demo and match snapshot 1`] = ` exports[`should render demo and match snapshot 1`] = `
<!--[-->
<div>
<!--[-->
<div
class="van-cell van-cell--clickable van-coupon-cell"
role="button"
tabindex="0"
>
<div
class="van-cell__title"
style
>
<span>
Coupon
</span>
</div>
<div class="van-cell__value van-coupon-cell__value">
<span>
You have 2 coupons
</span>
</div>
<i
class="van-badge__wrapper van-icon van-icon-arrow van-cell__right-icon"
style
>
<!--[-->
</i>
</div>
<!--[-->
</div>
<div> <div>
<!--[--> <!--[-->
<div <div

View File

@ -37,4 +37,40 @@ exports[`should render demo and match snapshot 1`] = `
> >
</transition-stub> </transition-stub>
</div> </div>
<div>
<div
class="van-cell van-cell--clickable van-coupon-cell"
role="button"
tabindex="0"
>
<div class="van-cell__title">
<span>
Coupon
</span>
</div>
<div class="van-cell__value van-coupon-cell__value">
<span>
You have 2 coupons
</span>
</div>
<i class="van-badge__wrapper van-icon van-icon-arrow van-cell__right-icon">
</i>
</div>
<transition-stub
name="van-fade"
appear="true"
persisted="false"
css="true"
role="button"
tabindex="0"
>
</transition-stub>
<transition-stub
name="van-popup-slide-bottom"
appear="false"
persisted="false"
css="true"
>
</transition-stub>
</div>
`; `;