# Tab 标签页
### 介绍
选项卡组件,用于在不同的内容区域之间进行切换。
### 引入
通过以下方式来全局注册组件,更多注册方式请参考[组件注册](#/zh-CN/advanced-usage#zu-jian-zhu-ce)。
```js
import { createApp } from 'vue';
import { Tab, Tabs } from 'vant';
const app = createApp();
app.use(Tab);
app.use(Tabs);
```
## 代码演示
### 基础用法
通过 `v-model:active` 绑定当前激活标签对应的索引值,默认情况下启用第一个标签。
```html
内容 1
内容 2
内容 3
内容 4
```
```js
import { ref } from 'vue';
export default {
setup() {
const active = ref(0);
return { active };
},
};
```
### 通过名称匹配
在标签指定 `name` 属性的情况下,`v-model:active` 的值为当前标签的 `name`(此时无法通过索引值来匹配标签)。
```html
内容 1
内容 2
内容 3
```
```js
import { ref } from 'vue';
export default {
setup() {
const activeName = ref('b');
return { activeName };
},
};
```
### 标签栏滚动
标签数量超过 5 个时,标签栏可以在水平方向上滚动,切换时会自动将当前标签居中。
```html
内容 {{ index }}
```
### 禁用标签
设置 `disabled` 属性即可禁用标签。
```html
内容 1
内容 2
内容 3
```
### 样式风格
`Tab` 支持两种样式风格:`line` 和`card`,默认为 `line` 样式,可以通过 `type` 属性切换样式风格。
```html
内容 1
内容 2
内容 3
```
### 点击事件
点击标签页时,会触发 `click-tab` 事件。
```html
内容 1
内容 2
```
```js
import { ref } from 'vue';
import { showToast } from 'vant';
export default {
setup() {
const active = ref(0);
const onClickTab = ({ title }) => showToast(title);
return {
active,
onClickTab,
};
},
};
```
### 粘性布局
通过 `sticky` 属性可以开启粘性布局,粘性布局下,标签页滚动到顶部时会自动吸顶。
```html
内容 {{ index }}
```
> Tips: 如果页面顶部有其他内容,可以通过 offset-top 属性设置吸顶时与顶部的距离。
### 收缩布局
通过 `shrink` 属性可以开启收缩布局,开启后,所有的标签会向左侧收缩对齐。
```html
内容 {{ index }}
内容 {{ index }}
```
### 自定义标签
通过 `title` 插槽可以自定义标签内容。
```html
标签
内容 {{ index }}
```
### 切换动画
通过 `animated` 属性可以开启切换标签内容时的转场动画。
```html
内容 {{ index }}
```
### 滑动切换
通过 `swipeable` 属性可以开启滑动切换标签页。
```html
内容 {{ index }}
```
### 滚动导航
通过 `scrollspy` 属性可以开启滚动导航模式,该模式下,内容将会平铺展示。
```html
内容 {{ index }}
```
### 异步切换
通过 `before-change` 属性可以在切换标签前执行特定的逻辑。
```html
内容 {{ index }}
```
```js
import { ref } from 'vue';
export default {
setup() {
const active = ref(0);
const beforeChange = (index) => {
// 返回 false 表示阻止此次切换
if (index === 1) {
return false;
}
// 返回 Promise 来执行异步逻辑
return new Promise((resolve) => {
// 在 resolve 函数中返回 true 或 false
setTimeout(() => resolve(index !== 3), 1000);
});
};
return {
beforeChange,
};
},
};
```
> Tips: 通过手势滑动不会触发 before-change 属性。
### 隐藏标题栏
通过将 `showHeader` 属性设置为 `false`,可以不渲染 Tabs 的标题栏。在这种情况下,你可以通过一些自定义组件来控制 Tabs 的 `active` 属性。
```html
内容 {{ index }}
```
## API
### Tabs Props
| 参数 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- |
| v-model:active | 绑定当前选中标签的标识符 | _number \| string_ | `0` |
| type | 样式风格类型,可选值为 `card` | _string_ | `line` |
| color | 标签主题色 | _string_ | `#1989fa` |
| background | 标签栏背景色 | _string_ | `white` |
| duration | 动画时间,单位秒,设置为 0 可以禁用动画 | _number \| string_ | `0.3` |
| line-width | 底部条宽度,默认单位 `px` | _number \| string_ | `40px` |
| line-height | 底部条高度,默认单位 `px` | _number \| string_ | `3px` |
| animated | 是否开启切换标签内容时的转场动画(开启该属性后,内容区如果有粘性布局将会不达预期) | _boolean_ | `false` |
| border | 是否显示标签栏外边框,仅在 `type="line"` 时有效 | _boolean_ | `false` |
| ellipsis | 是否省略过长的标题文字(仅在 `shrink` 为 `false` 且 `tab` 数量小于等于 `swipe-threshold` 时生效) | _boolean_ | `true` |
| sticky | 是否使用粘性布局 | _boolean_ | `false` |
| shrink | 是否开启左侧收缩布局 | _boolean_ | `false` |
| swipeable | 是否开启手势左右滑动切换(开启该属性后,内容区如果有粘性布局将会不达预期) | _boolean_ | `false` |
| lazy-render | 是否开启延迟渲染(首次切换到标签时才触发内容渲染) | _boolean_ | `true` |
| scrollspy | 是否开启滚动导航 | _boolean_ | `false` |
| show-header `v4.7.3` | 是否显示标题栏 | _boolean_ | `true` |
| offset-top | 粘性布局下吸顶时与顶部的距离,支持 `px` `vw` `vh` `rem` 单位,默认 `px` | _number \| string_ | `0` |
| swipe-threshold | 滚动阈值,标签数量超过阈值且总宽度超过标签栏宽度时开始横向滚动(仅在 `shrink` 为 `false` 且 `ellipsis` 为 `true` 时生效) | _number \| string_ | `5` |
| title-active-color | 标题选中态颜色 | _string_ | - |
| title-inactive-color | 标题默认态颜色 | _string_ | - |
| before-change | 切换标签前的回调函数,返回 `false` 可阻止切换,支持返回 Promise | _(name: number \| string) => boolean \| Promise\_ | - |
### Tab Props
| 参数 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- |
| title | 标题 | _string_ | - |
| disabled | 是否禁用标签 | _boolean_ | `false` |
| dot | 是否在标题右上角显示小红点 | _boolean_ | `false` |
| badge | 图标右上角徽标的内容(`dot` 为 `fasle` 时生效) | _number \| string_ | - |
| name | 标签名称,作为匹配的标识符 | _number \| string_ | 标签的索引值 |
| url | 点击后跳转的链接地址 | _string_ | - |
| to | 点击后跳转的目标路由对象,等同于 Vue Router 的 [to 属性](https://router.vuejs.org/zh/api/interfaces/RouterLinkProps.html#Properties-to) | _string \| object_ | - |
| replace | 是否在跳转时替换当前页面历史 | _boolean_ | `false` |
| title-style | 自定义标题样式 | _string \| Array \| object_ | - |
| title-class | 自定义标题类名 | _string \| Array \| object_ | - |
| show-zero-badge | 当 badge 为数字 0 时,是否展示徽标 | _boolean_ | `true` |
### Tabs Events
| 事件名 | 说明 | 回调参数 |
| --- | --- | --- |
| click-tab | 点击标签时触发 | _{ name: string \| number, title: string, event: MouseEvent, disabled: boolean }_ |
| change | 当前激活的标签改变时触发 | _name: string \| number, title: string_ |
| rendered | 标签内容首次渲染时触发(仅在开启延迟渲染后触发) | _name: string \| number, title: string_ |
| scroll | 滚动时触发,仅在 sticky 模式下生效 | _{ scrollTop: number, isFixed: boolean }_ |
> 提示:click 和 disabled 事件已废弃,请使用 click-tab 事件代替。
### Tabs 方法
通过 ref 可以获取到 Tabs 实例并调用实例方法,详见[组件实例方法](#/zh-CN/advanced-usage#zu-jian-shi-li-fang-fa)。
| 方法名 | 说明 | 参数 | 返回值 |
| --- | --- | --- | --- |
| resize | 外层元素大小或组件显示状态变化时,可以调用此方法来触发重绘 | - | - |
| scrollTo | 滚动到指定的标签页,在滚动导航模式下可用 | _name: string \| number_ | - |
### 类型定义
组件导出以下类型定义:
```ts
import type { TabProps, TabsType, TabsProps, TabsInstance } from 'vant';
```
`TabsInstance` 是组件实例的类型,用法如下:
```ts
import { ref } from 'vue';
import type { TabsInstance } from 'vant';
const tabsRef = ref();
tabsRef.value?.scrollTo(0);
```
### Tabs Slots
| 名称 | 说明 |
| ---------- | -------------- |
| nav-left | 标签栏左侧内容 |
| nav-right | 标签栏右侧内容 |
| nav-bottom | 标签栏下方内容 |
### Tab Slots
| 名称 | 说明 |
| ------- | ---------- |
| default | 标签页内容 |
| title | 自定义标题 |
## 主题定制
### 样式变量
组件提供了下列 CSS 变量,可用于自定义样式,使用方法请参考 [ConfigProvider 组件](#/zh-CN/config-provider)。
| 名称 | 默认值 | 描述 |
| ----------------------------- | --------------------------- | ---- |
| --van-tab-text-color | _var(--van-gray-7)_ | - |
| --van-tab-active-text-color | _var(--van-text-color)_ | - |
| --van-tab-disabled-text-color | _var(--van-text-color-3)_ | - |
| --van-tab-font-size | _var(--van-font-size-md)_ | - |
| --van-tab-line-height | _var(--van-line-height-md)_ | - |
| --van-tabs-default-color | _var(--van-primary-color)_ | - |
| --van-tabs-line-height | _44px_ | - |
| --van-tabs-card-height | _30px_ | - |
| --van-tabs-nav-background | _var(--van-background-2)_ | - |
| --van-tabs-bottom-bar-width | _40px_ | - |
| --van-tabs-bottom-bar-height | _3px_ | - |
| --van-tabs-bottom-bar-color | _var(--van-primary-color)_ | - |
## 常见问题
### 组件从隐藏状态切换到显示状态时,底部条位置错误?
Tabs 组件在挂载时,会获取自身的宽度,并计算出底部条的位置。如果组件一开始处于隐藏状态,则获取到的宽度永远为 0,因此无法展示底部条位置。
#### 解决方法
方法一,如果是使用 `v-show` 来控制组件展示的,则替换为 `v-if` 即可解决此问题:
```html
```
方法二,调用组件的 resize 方法来主动触发重绘:
```html
```
```js
this.$refs.tabs.resize();
```
### Tabs 开启 swipeable 或 animated 属性后,内容区元素的 sticky 功能将不达预期
`Tabs` 开启 `swipeable` 或 `animated` 属性后,内容区将被带有 `transform` 属性的元素包裹,此时如果内容区的元素开启了 `sticky` 功能,那么该功能生效了,但显示位置将不达预期。
比如下面的代码:
```html
sticky button
```
这是因为 `transform` 元素内部的 `fixed` 定位会相对于该元素进行计算,而不是相对于整个文档,从而导致布局异常。
### 如何判断当前组件是否处于激活的 Tab 内?
可以在子组件中通过调用 `useTabStatus` 或 `useAllTabStatus` 来判断当前组件是否处于激活的 `Tab` 内部。
- `useTabStatus`:返回当前组件所在的 `Tab` 是否为激活状态,若组件不在 `Tab` 内部则返回 `null`。
- `useAllTabStatus`:在存在嵌套 `Tab` 的场景下,返回是否所有上层 `Tab` 为激活状态,若组件不在 `Tab` 内部则返回 `null`。
```js
const isActive = useTabStatus();
// 嵌套 Tab 场景
const isAllActive = useAllTabStatus();
```