[new feature] add Tabbar component (#435)

This commit is contained in:
neverland 2018-08-16 21:29:11 +08:00 committed by GitHub
parent afedb9ec2e
commit efdfd6691f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 347 additions and 0 deletions

View File

@ -33,6 +33,7 @@ const MAP = {
steps: 'steps-201808092138.png', steps: 'steps-201808092138.png',
switch: 'switch-201808092138.png', switch: 'switch-201808092138.png',
tag: 'tag-201808092138.png', tag: 'tag-201808092138.png',
tabbar: 'tabbar-201808160922.png',
'tree-select': 'tree-select-201808092138.png' 'tree-select': 'tree-select-201808092138.png'
}; };

View File

@ -20,6 +20,7 @@
"pages/switch/index", "pages/switch/index",
"pages/search/index", "pages/search/index",
"pages/tag/index", "pages/tag/index",
"pages/tabbar/index",
"pages/tree-select/index" "pages/tree-select/index"
], ],
"window": { "window": {

View File

@ -49,6 +49,10 @@ export default [
{ {
path: '/tag', path: '/tag',
title: 'Tag 标记' title: 'Tag 标记'
},
{
path: '/tabbar',
title: 'Tabbar 标签栏'
} }
] ]
}, },

View 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);
}
});

View 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"
}
}

View 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>

View File

@ -0,0 +1,3 @@
.tabbar {
position: relative !important;
}

View 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');
}
}
});

View File

@ -0,0 +1,6 @@
{
"component": true,
"usingComponents": {
"van-icon": "../icon/index"
}
}

View 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;
}
}

View 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
View 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
View 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();
}
}
}
});

View File

@ -0,0 +1,3 @@
{
"component": true
}

View 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;
}
}

View 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>