mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
feat(tab): add before-change event support (#5139)
* feat(tab): add before-change event support * docs(tab): delete unnecessary content
This commit is contained in:
parent
e7426dba36
commit
76522a173f
@ -185,6 +185,49 @@ Page({
|
||||
</van-popup>
|
||||
```
|
||||
|
||||
### 异步切换
|
||||
|
||||
通过 `before-change` 事件可以在切换标签前执行特定的逻辑,实现切换前校验、异步切换的目的
|
||||
|
||||
```html
|
||||
<van-tabs active="{{ active }}" use-before-change="{{ true }}" bind:change="onChange" bind:before-change="onBeforeChange" >
|
||||
<van-tab title="标签 1">内容 1</van-tab>
|
||||
<van-tab title="标签 2">内容 2</van-tab>
|
||||
<van-tab title="标签 3">内容 3</van-tab>
|
||||
<van-tab title="标签 4">内容 4</van-tab>
|
||||
</van-tabs>
|
||||
```
|
||||
|
||||
```js
|
||||
Page({
|
||||
data: {
|
||||
active: 1,
|
||||
},
|
||||
|
||||
onChange(event) {
|
||||
wx.showToast({
|
||||
title: `切换到标签 ${event.detail.name}`,
|
||||
icon: 'none',
|
||||
});
|
||||
},
|
||||
onBeforeChange(event) {
|
||||
const { callback, title } = event.detail;
|
||||
|
||||
wx.showModal({
|
||||
title: '异步切换',
|
||||
content: `确定要切换至 ${title} tab吗?`,
|
||||
success: (res) => {
|
||||
if (res.confirm) {
|
||||
callback(true)
|
||||
} else if (res.cancel) {
|
||||
callback(false)
|
||||
}
|
||||
},
|
||||
})
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
## API
|
||||
|
||||
### Tabs Props
|
||||
@ -208,6 +251,7 @@ Page({
|
||||
| title-active-color | 标题选中态颜色 | _string_ | - |
|
||||
| title-inactive-color | 标题默认态颜色 | _string_ | - |
|
||||
| z-index | z-index 层级 | _number_ | `1` |
|
||||
| use-before-change `v1.10.10` | 是否开启切换前校验 | _boolean_ | `false` |
|
||||
|
||||
### Tab Props
|
||||
|
||||
@ -238,18 +282,19 @@ Page({
|
||||
| 事件名 | 说明 | 参数 |
|
||||
| --- | --- | --- |
|
||||
| bind:click | 点击标签时触发 | name:标签标识符,title:标题 |
|
||||
| bind:before-change `v1.10.10` | tab 切换前会触发,在回调函数中返回 `false` 可终止 tab 切换,绑定事件的同时需要将`use-before-change`属性设置为`true` | `event.detail.name`: 当前切换的 tab 标识符, `event.detail.title`: 当前切换的 tab 标题, `event.detail.index`: 当前切换的 tab 下标,`event.detail.callback`: 回调函数,调用`callback(false)`终止 tab 切换 |
|
||||
| bind:change | 当前激活的标签改变时触发 | name:标签标识符,title:标题 |
|
||||
| bind:disabled | 点击被禁用的标签时触发 | name:标签标识符,title:标题 |
|
||||
| bind:scroll | 滚动时触发 | { scrollTop: 距离顶部位置, isFixed: 是否吸顶 } |
|
||||
|
||||
### 外部样式类
|
||||
|
||||
| 类名 | 说明 |
|
||||
| ---------------- | ---------------- |
|
||||
| custom-class | 根节点样式类 |
|
||||
| nav-class | 标签栏样式类 |
|
||||
| tab-class | 标签样式类 |
|
||||
| tab-active-class | 标签激活态样式类 |
|
||||
| 类名 | 说明 |
|
||||
| ---------------- | ------------------ |
|
||||
| custom-class | 根节点样式类 |
|
||||
| nav-class | 标签栏样式类 |
|
||||
| tab-class | 标签样式类 |
|
||||
| tab-active-class | 标签激活态样式类 |
|
||||
| wrap-class | 标签栏根节点样式类 |
|
||||
|
||||
### 方法
|
||||
|
@ -41,5 +41,20 @@ VantComponent({
|
||||
icon: 'none',
|
||||
});
|
||||
},
|
||||
onBeforeChange(event) {
|
||||
const { callback, title } = event.detail;
|
||||
|
||||
wx.showModal({
|
||||
title: '异步切换',
|
||||
content: `确定要切换至 ${title} tab吗?`,
|
||||
success: (res) => {
|
||||
if (res.confirm) {
|
||||
callback(true);
|
||||
} else if (res.cancel) {
|
||||
callback(false);
|
||||
}
|
||||
},
|
||||
});
|
||||
},
|
||||
},
|
||||
});
|
||||
|
@ -147,3 +147,18 @@
|
||||
</van-tab>
|
||||
</van-tabs>
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="异步切换">
|
||||
<van-tabs active="{{ 1 }}" swipeable use-before-change bind:change="onChange" bind:before-change="onBeforeChange" >
|
||||
<van-tab
|
||||
wx:for="{{ tabs4 }}"
|
||||
wx:key="index"
|
||||
title="{{ '标签 ' + item }}"
|
||||
>
|
||||
<view class="content">
|
||||
{{ '内容' + item }}
|
||||
</view>
|
||||
</van-tab>
|
||||
</van-tabs>
|
||||
</demo-block>
|
||||
|
||||
|
@ -1548,5 +1548,165 @@ exports[`should render demo and match snapshot 1`] = `
|
||||
</van-tabs>
|
||||
</wx-view>
|
||||
</demo-block>
|
||||
<demo-block>
|
||||
<wx-view
|
||||
class="custom-class demo-block van-clearfix "
|
||||
>
|
||||
<wx-view
|
||||
class="demo-block__title"
|
||||
>
|
||||
异步切换
|
||||
</wx-view>
|
||||
<van-tabs
|
||||
bind:before-change="onBeforeChange"
|
||||
bind:change="onChange"
|
||||
>
|
||||
<wx-view
|
||||
class="custom-class van-tabs van-tabs--line"
|
||||
>
|
||||
<van-sticky
|
||||
bind:scroll="onTouchScroll"
|
||||
>
|
||||
<wx-view
|
||||
class="custom-class van-sticky"
|
||||
style="z-index:1"
|
||||
>
|
||||
<wx-view
|
||||
class="van-sticky-wrap"
|
||||
style="z-index:1"
|
||||
>
|
||||
<wx-view
|
||||
class="van-tabs__wrap wrap-class"
|
||||
>
|
||||
<wx-scroll-view
|
||||
class="van-tabs__scroll van-tabs__scroll--line"
|
||||
scrollLeft="{{0}}"
|
||||
scrollWithAnimation="{{false}}"
|
||||
scrollX="{{false}}"
|
||||
style=""
|
||||
>
|
||||
<wx-view
|
||||
class="van-tabs__nav van-tabs__nav--line nav-class"
|
||||
style=""
|
||||
>
|
||||
<wx-view
|
||||
class="van-tabs__line"
|
||||
style="width:40px;opacity:0;transform:translateX(0px);-webkit-transform:translateX(0px)"
|
||||
/>
|
||||
<wx-view
|
||||
class="tab-class van-ellipsis van-tab"
|
||||
data-index="{{0}}"
|
||||
style=""
|
||||
bind:tap="onTap"
|
||||
>
|
||||
<wx-view
|
||||
class="van-ellipsis"
|
||||
style=""
|
||||
>
|
||||
|
||||
标签 1
|
||||
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
<wx-view
|
||||
class="tab-class tab-active-class van-ellipsis van-tab van-tab--active"
|
||||
data-index="{{1}}"
|
||||
style=""
|
||||
bind:tap="onTap"
|
||||
>
|
||||
<wx-view
|
||||
class="van-ellipsis"
|
||||
style=""
|
||||
>
|
||||
|
||||
标签 2
|
||||
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
<wx-view
|
||||
class="tab-class van-ellipsis van-tab"
|
||||
data-index="{{2}}"
|
||||
style=""
|
||||
bind:tap="onTap"
|
||||
>
|
||||
<wx-view
|
||||
class="van-ellipsis"
|
||||
style=""
|
||||
>
|
||||
|
||||
标签 3
|
||||
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
<wx-view
|
||||
class="tab-class van-ellipsis van-tab"
|
||||
data-index="{{3}}"
|
||||
style=""
|
||||
bind:tap="onTap"
|
||||
>
|
||||
<wx-view
|
||||
class="van-ellipsis"
|
||||
style=""
|
||||
>
|
||||
|
||||
标签 4
|
||||
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</wx-scroll-view>
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</van-sticky>
|
||||
<wx-view
|
||||
class="van-tabs__content"
|
||||
bind:touchcancel="onTouchEnd"
|
||||
bind:touchend="onTouchEnd"
|
||||
bind:touchmove="onTouchMove"
|
||||
bind:touchstart="onTouchStart"
|
||||
>
|
||||
<wx-view
|
||||
class="van-tabs__track van-tabs__track"
|
||||
style=""
|
||||
>
|
||||
<van-tab>
|
||||
<wx-view
|
||||
class="custom-class van-tab__pane van-tab__pane--inactive"
|
||||
style="display: none;"
|
||||
/>
|
||||
</van-tab>
|
||||
<van-tab>
|
||||
<wx-view
|
||||
class="custom-class van-tab__pane van-tab__pane--active"
|
||||
style=""
|
||||
>
|
||||
<wx-view
|
||||
class="content"
|
||||
>
|
||||
|
||||
内容2
|
||||
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</van-tab>
|
||||
<van-tab>
|
||||
<wx-view
|
||||
class="custom-class van-tab__pane van-tab__pane--inactive"
|
||||
style="display: none;"
|
||||
/>
|
||||
</van-tab>
|
||||
<van-tab>
|
||||
<wx-view
|
||||
class="custom-class van-tab__pane van-tab__pane--inactive"
|
||||
style="display: none;"
|
||||
/>
|
||||
</van-tab>
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</wx-view>
|
||||
</van-tabs>
|
||||
</wx-view>
|
||||
</demo-block>
|
||||
</main>
|
||||
`;
|
||||
|
@ -93,6 +93,10 @@ VantComponent({
|
||||
type: Boolean,
|
||||
value: true,
|
||||
},
|
||||
useBeforeChange: {
|
||||
type: Boolean,
|
||||
value: false,
|
||||
},
|
||||
},
|
||||
|
||||
data: {
|
||||
@ -134,17 +138,13 @@ VantComponent({
|
||||
trigger(eventName: string, child?: TrivialInstance) {
|
||||
const { currentIndex } = this.data;
|
||||
|
||||
const currentChild = child || this.children[currentIndex];
|
||||
const data = this.getChildData(currentIndex, child);
|
||||
|
||||
if (!isDef(currentChild)) {
|
||||
if (!isDef(data)) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.$emit(eventName, {
|
||||
index: currentChild.index,
|
||||
name: currentChild.getComputedName(),
|
||||
title: currentChild.data.title,
|
||||
});
|
||||
this.$emit(eventName, data);
|
||||
},
|
||||
|
||||
onTap(event: WechatMiniprogram.TouchEvent) {
|
||||
@ -153,12 +153,15 @@ VantComponent({
|
||||
|
||||
if (child.data.disabled) {
|
||||
this.trigger('disabled', child);
|
||||
} else {
|
||||
return;
|
||||
}
|
||||
|
||||
this.onBeforeChange(index).then(() => {
|
||||
this.setCurrentIndex(index);
|
||||
nextTick(() => {
|
||||
this.trigger('click');
|
||||
});
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
// correct the index of active tab
|
||||
@ -313,7 +316,7 @@ VantComponent({
|
||||
if (direction === 'horizontal' && offsetX >= minSwipeDistance) {
|
||||
const index = this.getAvaiableTab(deltaX);
|
||||
if (index !== -1) {
|
||||
this.setCurrentIndex(index);
|
||||
this.onBeforeChange(index).then(() => this.setCurrentIndex(index));
|
||||
}
|
||||
}
|
||||
|
||||
@ -343,5 +346,32 @@ VantComponent({
|
||||
|
||||
return -1;
|
||||
},
|
||||
onBeforeChange(index: number): Promise<void> {
|
||||
const { useBeforeChange } = this.data;
|
||||
|
||||
if (!useBeforeChange) {
|
||||
return Promise.resolve();
|
||||
}
|
||||
|
||||
return new Promise((resolve, reject) => {
|
||||
this.$emit('before-change', {
|
||||
...this.getChildData(index),
|
||||
callback: (status) => (status ? resolve() : reject()),
|
||||
});
|
||||
});
|
||||
},
|
||||
getChildData(index: number, child?: TrivialInstance) {
|
||||
const currentChild = child || this.children[index];
|
||||
|
||||
if (!isDef(currentChild)) {
|
||||
return;
|
||||
}
|
||||
|
||||
return {
|
||||
index: currentChild.index,
|
||||
name: currentChild.getComputedName(),
|
||||
title: currentChild.data.title,
|
||||
};
|
||||
},
|
||||
},
|
||||
});
|
||||
|
Loading…
x
Reference in New Issue
Block a user