mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
[new feature] add Tabbar component (#435)
This commit is contained in:
parent
afedb9ec2e
commit
efdfd6691f
@ -33,6 +33,7 @@ const MAP = {
|
||||
steps: 'steps-201808092138.png',
|
||||
switch: 'switch-201808092138.png',
|
||||
tag: 'tag-201808092138.png',
|
||||
tabbar: 'tabbar-201808160922.png',
|
||||
'tree-select': 'tree-select-201808092138.png'
|
||||
};
|
||||
|
||||
|
@ -20,6 +20,7 @@
|
||||
"pages/switch/index",
|
||||
"pages/search/index",
|
||||
"pages/tag/index",
|
||||
"pages/tabbar/index",
|
||||
"pages/tree-select/index"
|
||||
],
|
||||
"window": {
|
||||
|
@ -49,6 +49,10 @@ export default [
|
||||
{
|
||||
path: '/tag',
|
||||
title: 'Tag 标记'
|
||||
},
|
||||
{
|
||||
path: '/tabbar',
|
||||
title: 'Tabbar 标签栏'
|
||||
}
|
||||
]
|
||||
},
|
||||
|
18
example/pages/tabbar/index.js
Normal file
18
example/pages/tabbar/index.js
Normal file
@ -0,0 +1,18 @@
|
||||
import Page from '../../common/page';
|
||||
|
||||
Page({
|
||||
data: {
|
||||
active: 0,
|
||||
active2: 0,
|
||||
icon: {
|
||||
normal:
|
||||
'https://img.yzcdn.cn/public_files/2017/10/13/c547715be149dd3faa817e4a948b40c4.png',
|
||||
active:
|
||||
'https://img.yzcdn.cn/public_files/2017/10/13/793c77793db8641c4c325b7f25bf130d.png'
|
||||
}
|
||||
},
|
||||
|
||||
onChange(event) {
|
||||
console.log(event.detail);
|
||||
}
|
||||
});
|
8
example/pages/tabbar/index.json
Normal file
8
example/pages/tabbar/index.json
Normal file
@ -0,0 +1,8 @@
|
||||
{
|
||||
"navigationBarTitleText": "Tabbar 标签页",
|
||||
"usingComponents": {
|
||||
"demo-block": "../../components/demo-block/index",
|
||||
"van-tabbar": "../../dist/tabbar/index",
|
||||
"van-tabbar-item": "../../dist/tabbar-item/index"
|
||||
}
|
||||
}
|
20
example/pages/tabbar/index.wxml
Normal file
20
example/pages/tabbar/index.wxml
Normal file
@ -0,0 +1,20 @@
|
||||
<demo-block title="基础用法">
|
||||
<van-tabbar active="{{ active }}" custom-class="tabbar" bind:change="onChange">
|
||||
<van-tabbar-item icon="shop">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="chat" dot>标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="records" info="5">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="gold-coin" info="20">标签</van-tabbar-item>
|
||||
</van-tabbar>
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="自定义图标">
|
||||
<van-tabbar active="{{ active2 }}" custom-class="tabbar" bind:change="onChange">
|
||||
<van-tabbar-item>
|
||||
<span>自定义</span>
|
||||
<image slot="icon" src="{{ icon.normal }}" class="icon" mode="aspectFit" />
|
||||
<image slot="icon-active" src="{{ icon.active }}" mode="aspectFit" />
|
||||
</van-tabbar-item>
|
||||
<van-tabbar-item icon="chat">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="records">标签</van-tabbar-item>
|
||||
</van-tabbar>
|
||||
</demo-block>
|
3
example/pages/tabbar/index.wxss
Normal file
3
example/pages/tabbar/index.wxss
Normal file
@ -0,0 +1,3 @@
|
||||
.tabbar {
|
||||
position: relative !important;
|
||||
}
|
36
packages/tabbar-item/index.js
Normal file
36
packages/tabbar-item/index.js
Normal file
@ -0,0 +1,36 @@
|
||||
const TABBAR_PATH = '../tabbar/index';
|
||||
|
||||
Component({
|
||||
name: 'tabbar-item',
|
||||
|
||||
properties: {
|
||||
info: null,
|
||||
icon: String,
|
||||
dot: Boolean
|
||||
},
|
||||
|
||||
options: {
|
||||
multipleSlots: true
|
||||
},
|
||||
|
||||
relations: {
|
||||
[TABBAR_PATH]: {
|
||||
type: 'ancestor'
|
||||
}
|
||||
},
|
||||
|
||||
data: {
|
||||
active: false,
|
||||
count: 0
|
||||
},
|
||||
|
||||
methods: {
|
||||
onClick() {
|
||||
const parent = this.getRelationNodes(TABBAR_PATH)[0];
|
||||
if (parent) {
|
||||
parent.onChange(this);
|
||||
}
|
||||
this.triggerEvent('click');
|
||||
}
|
||||
}
|
||||
});
|
6
packages/tabbar-item/index.json
Normal file
6
packages/tabbar-item/index.json
Normal file
@ -0,0 +1,6 @@
|
||||
{
|
||||
"component": true,
|
||||
"usingComponents": {
|
||||
"van-icon": "../icon/index"
|
||||
}
|
||||
}
|
45
packages/tabbar-item/index.pcss
Normal file
45
packages/tabbar-item/index.pcss
Normal file
@ -0,0 +1,45 @@
|
||||
@import '../common/style/var.pcss';
|
||||
|
||||
.van-tabbar-item {
|
||||
float: left;
|
||||
color: #666;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
line-height: 1;
|
||||
font-size: 12px;
|
||||
align-items: center;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
|
||||
&__icon {
|
||||
font-size: 18px;
|
||||
margin-bottom: 5px;
|
||||
position: relative;
|
||||
|
||||
.van-icon {
|
||||
display: block;
|
||||
}
|
||||
|
||||
&--dot {
|
||||
&::after {
|
||||
top: 0;
|
||||
right: -8px;
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
content: ' ';
|
||||
position: absolute;
|
||||
border-radius: 100%;
|
||||
background-color: $red;
|
||||
}
|
||||
}
|
||||
|
||||
image {
|
||||
width: 50px;
|
||||
height: 18px;
|
||||
}
|
||||
}
|
||||
|
||||
&--active {
|
||||
color: $blue;
|
||||
}
|
||||
}
|
18
packages/tabbar-item/index.wxml
Normal file
18
packages/tabbar-item/index.wxml
Normal file
@ -0,0 +1,18 @@
|
||||
<view
|
||||
class="van-tabbar-item {{ active ? 'van-tabbar-item--active' : '' }}"
|
||||
style="{{ count ? 'width: ' + 100 / count + '%' : '' }}"
|
||||
bind:tap="onClick"
|
||||
>
|
||||
<view class="van-tabbar-item__icon {{ dot ? 'van-tabbar-item__icon--dot' : '' }}">
|
||||
<block wx:if="{{ active }}">
|
||||
<slot name="icon-active" />
|
||||
</block>
|
||||
<block wx:else>
|
||||
<slot name="icon" />
|
||||
</block>
|
||||
<van-icon wx:if="{{ icon }}" name="{{ icon }}" info="{{ info }}" />
|
||||
</view>
|
||||
<view class="van-tabbar-item__text">
|
||||
<slot />
|
||||
</view>
|
||||
</view>
|
97
packages/tabbar/README.md
Normal file
97
packages/tabbar/README.md
Normal file
@ -0,0 +1,97 @@
|
||||
## Tabbar 标签栏
|
||||
|
||||
### 使用指南
|
||||
在 index.json 中引入组件
|
||||
```json
|
||||
"usingComponents": {
|
||||
"van-tabbar": "/packages/tabbar/index",
|
||||
"van-tabbar-item": "/packages/tabbar-item/index"
|
||||
}
|
||||
```
|
||||
|
||||
### 代码演示
|
||||
|
||||
#### 基础用法
|
||||
|
||||
|
||||
```html
|
||||
<van-tabbar active="{{ active }}" bind:change="onChange">
|
||||
<van-tabbar-item icon="shop">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="chat" dot>标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="records" info="5">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="gold-coin" info="20">标签</van-tabbar-item>
|
||||
</van-tabbar>
|
||||
```
|
||||
|
||||
```javascript
|
||||
Page({
|
||||
data: {
|
||||
active: 0
|
||||
},
|
||||
// event.detail 的值为当前选中项的索引
|
||||
onChange(event) {
|
||||
console.log(event.detail);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
|
||||
#### 自定义图标
|
||||
|
||||
可以通过 slot 自定义图标,其中 icon slot 代表未选中状态下的图标,icon-active slot 代表选中状态下的图标
|
||||
|
||||
```html
|
||||
<van-tabbar active="{{ active }}" bind:change="onChange">
|
||||
<van-tabbar-item>
|
||||
<span>自定义</span>
|
||||
<image slot="icon" src="{{ icon.normal }}" class="icon" mode="aspectFit" />
|
||||
<image slot="icon-active" src="{{ icon.active }}" mode="aspectFit" />
|
||||
</van-tabbar-item>
|
||||
<van-tabbar-item icon="chat">标签</van-tabbar-item>
|
||||
<van-tabbar-item icon="records">标签</van-tabbar-item>
|
||||
</van-tabbar>
|
||||
```
|
||||
|
||||
```javascript
|
||||
Page({
|
||||
data() {
|
||||
active: 0,
|
||||
icon: {
|
||||
normal: '//img.yzcdn.cn/icon-normal.png',
|
||||
active: '//img.yzcdn.cn/icon-active.png'
|
||||
}
|
||||
},
|
||||
onChange(event) {
|
||||
console.log(event.detail);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
### Tabbar API
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 |
|
||||
|-----------|-----------|-----------|-------------|
|
||||
| active | 当前选中标签的索引 | `Number` | - |
|
||||
| fixed | 是否固定在底部 | `Boolean` | `true` |
|
||||
| z-index | 元素 z-index | `Number` | `1` |
|
||||
|
||||
### Tabbar Event
|
||||
|
||||
| 事件名 | 说明 | 参数 |
|
||||
|-----------|-----------|-----------|
|
||||
| bind:change | 切换标签时触发 | event.detail: 当前选中标签的索引 |
|
||||
|
||||
### TabbarItem API
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 |
|
||||
|-----------|-----------|-----------|-----------|
|
||||
| icon | 图标名称 (可选值见 Icon 组件) | `String` | - |
|
||||
| dot | 是否显示小红点 | `Boolean` | - |
|
||||
| info | 图标右上角提示信息 | `String | Number` | - |
|
||||
|
||||
### TabbarItem Slot
|
||||
|
||||
| 名称 | 说明 |
|
||||
|-----------|-----------|
|
||||
| icon | 未选中时的图标 |
|
||||
| icon-active | 选中时的图标 |
|
68
packages/tabbar/index.js
Normal file
68
packages/tabbar/index.js
Normal file
@ -0,0 +1,68 @@
|
||||
const ITEM_PATH = '../tabbar-item/index';
|
||||
|
||||
Component({
|
||||
externalClasses: ['custom-class'],
|
||||
|
||||
properties: {
|
||||
active: {
|
||||
type: Number,
|
||||
observer(active) {
|
||||
this.setData({ currentActive: active });
|
||||
this.setActiveItem();
|
||||
}
|
||||
},
|
||||
fixed: {
|
||||
type: Boolean,
|
||||
value: true
|
||||
},
|
||||
zIndex: {
|
||||
type: Number,
|
||||
value: 1
|
||||
}
|
||||
},
|
||||
|
||||
data: {
|
||||
items: [],
|
||||
currentActive: -1
|
||||
},
|
||||
|
||||
attached() {
|
||||
this.setData({ currentActive: this.data.active });
|
||||
},
|
||||
|
||||
relations: {
|
||||
[ITEM_PATH]: {
|
||||
type: 'descendant',
|
||||
|
||||
linked(target) {
|
||||
this.data.items.push(target);
|
||||
this.setActiveItem();
|
||||
},
|
||||
|
||||
unlinked(target) {
|
||||
this.data.items = this.data.items.filter(item => item !== target);
|
||||
this.setActiveItem();
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
methods: {
|
||||
setActiveItem() {
|
||||
this.data.items.forEach((item, index) => {
|
||||
item.setData({
|
||||
active: index === this.data.currentActive,
|
||||
count: this.data.items.length
|
||||
});
|
||||
});
|
||||
},
|
||||
|
||||
onChange(child) {
|
||||
const active = this.data.items.indexOf(child);
|
||||
if (active !== this.data.currentActive && active !== -1) {
|
||||
this.triggerEvent('change', active);
|
||||
this.setData({ currentActive: active });
|
||||
this.setActiveItem();
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
3
packages/tabbar/index.json
Normal file
3
packages/tabbar/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"component": true
|
||||
}
|
13
packages/tabbar/index.pcss
Normal file
13
packages/tabbar/index.pcss
Normal file
@ -0,0 +1,13 @@
|
||||
@import '../common/style/var.pcss';
|
||||
|
||||
.van-tabbar {
|
||||
width: 100%;
|
||||
height: 50px;
|
||||
background-color: #fff;
|
||||
|
||||
&--fixed {
|
||||
left: 0;
|
||||
bottom: 0;
|
||||
position: fixed;
|
||||
}
|
||||
}
|
6
packages/tabbar/index.wxml
Normal file
6
packages/tabbar/index.wxml
Normal file
@ -0,0 +1,6 @@
|
||||
<view
|
||||
class="custom-class van-tabbar van-hairline--top-bottom {{ fixed ? 'van-tabbar--fixed' : '' }}"
|
||||
style="{{ zIndex ? 'style: ' + this.zIndex : '' }}"
|
||||
>
|
||||
<slot />
|
||||
</view>
|
Loading…
x
Reference in New Issue
Block a user