mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-05 19:41:45 +08:00
feat(IndexBar): add IndexBar component (#2303)
This commit is contained in:
parent
53eb32ee9b
commit
c8d27f5ca7
@ -45,7 +45,8 @@
|
|||||||
"pages/overlay/index",
|
"pages/overlay/index",
|
||||||
"pages/circle/index",
|
"pages/circle/index",
|
||||||
"pages/grid/index",
|
"pages/grid/index",
|
||||||
"pages/dropdown-menu/index"
|
"pages/dropdown-menu/index",
|
||||||
|
"pages/index-bar/index"
|
||||||
],
|
],
|
||||||
"window": {
|
"window": {
|
||||||
"navigationBarBackgroundColor": "#f8f8f8",
|
"navigationBarBackgroundColor": "#f8f8f8",
|
||||||
@ -110,6 +111,8 @@
|
|||||||
"van-picker": "./dist/picker/index",
|
"van-picker": "./dist/picker/index",
|
||||||
"van-overlay": "./dist/overlay/index",
|
"van-overlay": "./dist/overlay/index",
|
||||||
"van-circle": "./dist/circle/index",
|
"van-circle": "./dist/circle/index",
|
||||||
|
"van-index-bar": "./dist/index-bar/index",
|
||||||
|
"van-index-anchor": "./dist/index-anchor/index",
|
||||||
"van-grid": "./dist/grid/index",
|
"van-grid": "./dist/grid/index",
|
||||||
"van-grid-item": "./dist/grid-item/index",
|
"van-grid-item": "./dist/grid-item/index",
|
||||||
"van-dropdown-menu": "./dist/dropdown-menu/index",
|
"van-dropdown-menu": "./dist/dropdown-menu/index",
|
||||||
|
@ -183,6 +183,10 @@ export default [
|
|||||||
path: '/grid',
|
path: '/grid',
|
||||||
title: 'Grid 宫格'
|
title: 'Grid 宫格'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: '/index-bar',
|
||||||
|
title: 'IndexBar 索引栏'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/sidebar',
|
path: '/sidebar',
|
||||||
title: 'Sidebar 侧边导航'
|
title: 'Sidebar 侧边导航'
|
||||||
|
28
example/pages/index-bar/index.js
Normal file
28
example/pages/index-bar/index.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import Page from '../../common/page';
|
||||||
|
|
||||||
|
const indexList = [];
|
||||||
|
const charCodeOfA = 'A'.charCodeAt(0);
|
||||||
|
for (let i = 0; i < 26; i++) {
|
||||||
|
indexList.push(String.fromCharCode(charCodeOfA + i));
|
||||||
|
}
|
||||||
|
|
||||||
|
Page({
|
||||||
|
data: {
|
||||||
|
activeTab: 0,
|
||||||
|
indexList,
|
||||||
|
customIndexList: [1, 2, 3, 4, 5, 6, 8, 9, 10],
|
||||||
|
scrollTop: 0,
|
||||||
|
},
|
||||||
|
|
||||||
|
onChange(event) {
|
||||||
|
this.setData({
|
||||||
|
activeTab: event.detail.name
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
onPageScroll(event) {
|
||||||
|
this.setData({
|
||||||
|
scrollTop: event.scrollTop
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
3
example/pages/index-bar/index.json
Normal file
3
example/pages/index-bar/index.json
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"navigationBarTitleText": "IndexBar 索引栏"
|
||||||
|
}
|
45
example/pages/index-bar/index.wxml
Normal file
45
example/pages/index-bar/index.wxml
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<van-tabs
|
||||||
|
active="{{ activeTab }}"
|
||||||
|
bind:change="onChange"
|
||||||
|
>
|
||||||
|
<van-tab title="基础用法">
|
||||||
|
<van-index-bar
|
||||||
|
wx:if="{{ activeTab === 0 }}"
|
||||||
|
scroll-top="{{ scrollTop }}"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
wx:for="{{ indexList }}"
|
||||||
|
wx:for-item="item"
|
||||||
|
wx:key="item"
|
||||||
|
>
|
||||||
|
<van-index-anchor index="{{ item }}" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
</view>
|
||||||
|
</van-index-bar>
|
||||||
|
</van-tab>
|
||||||
|
|
||||||
|
<van-tab title="自定义索引列表">
|
||||||
|
<van-index-bar
|
||||||
|
wx:if="{{ activeTab === 1 }}"
|
||||||
|
index-list="{{ customIndexList }}"
|
||||||
|
scroll-top="{{ scrollTop }}"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
wx:for="{{ customIndexList }}"
|
||||||
|
wx:key="index"
|
||||||
|
>
|
||||||
|
<van-index-anchor
|
||||||
|
use-slot
|
||||||
|
index="{{ item }}"
|
||||||
|
>
|
||||||
|
<text>标题{{ item }}</text>
|
||||||
|
</van-index-anchor>
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
</view>
|
||||||
|
</van-index-bar>
|
||||||
|
</van-tab>
|
||||||
|
</van-tabs>
|
0
example/pages/index-bar/index.wxss
Normal file
0
example/pages/index-bar/index.wxss
Normal file
@ -540,6 +540,20 @@
|
|||||||
@dropdown-menu-title-line-height: 18px;
|
@dropdown-menu-title-line-height: 18px;
|
||||||
@dropdown-menu-option-active-color: @blue;
|
@dropdown-menu-option-active-color: @blue;
|
||||||
|
|
||||||
|
// IndexAnchor
|
||||||
|
@index-anchor-padding: 0 @padding-md;
|
||||||
|
@index-anchor-text-color: @text-color;
|
||||||
|
@index-anchor-font-weight: 500;
|
||||||
|
@index-anchor-font-size: @font-size-md;
|
||||||
|
@index-anchor-line-height: 32px;
|
||||||
|
@index-anchor-background-color: transparent;
|
||||||
|
@index-anchor-active-background-color: @white;
|
||||||
|
@index-anchor-active-text-color: @green;
|
||||||
|
|
||||||
|
// IndexBar
|
||||||
|
@index-bar-index-font-size: @font-size-xs;
|
||||||
|
@index-bar-index-line-height: 14px;
|
||||||
|
|
||||||
// skeleton
|
// skeleton
|
||||||
@skeleton-padding: 0 @padding-md;
|
@skeleton-padding: 0 @padding-md;
|
||||||
@skeleton-row-height: 16px;
|
@skeleton-row-height: 16px;
|
||||||
|
3
packages/index-anchor/index.json
Normal file
3
packages/index-anchor/index.json
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"component": true
|
||||||
|
}
|
18
packages/index-anchor/index.less
Normal file
18
packages/index-anchor/index.less
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
@import '../common/style/var.less';
|
||||||
|
@import '../common/style/theme.less';
|
||||||
|
|
||||||
|
.van-index-anchor {
|
||||||
|
.theme(padding, '@index-anchor-padding');
|
||||||
|
.theme(color, '@index-anchor-text-color');
|
||||||
|
.theme(font-weight, '@index-anchor-font-weight');
|
||||||
|
.theme(font-size, '@index-anchor-font-size');
|
||||||
|
.theme(line-height, '@index-anchor-line-height');
|
||||||
|
.theme(background-color, '@index-anchor-background-color');
|
||||||
|
|
||||||
|
&--active {
|
||||||
|
right: 0;
|
||||||
|
left: 0;
|
||||||
|
.theme(color, '@index-anchor-active-text-color');
|
||||||
|
.theme(background-color, '@index-anchor-active-background-color');
|
||||||
|
}
|
||||||
|
}
|
25
packages/index-anchor/index.ts
Normal file
25
packages/index-anchor/index.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { VantComponent } from '../common/component';
|
||||||
|
|
||||||
|
VantComponent({
|
||||||
|
relation: {
|
||||||
|
name: 'index-bar',
|
||||||
|
type: 'ancestor',
|
||||||
|
linked(target) {
|
||||||
|
this.parent = target;
|
||||||
|
},
|
||||||
|
unlinked() {
|
||||||
|
this.parent = null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
useSlot: Boolean,
|
||||||
|
index: null
|
||||||
|
},
|
||||||
|
|
||||||
|
data: {
|
||||||
|
active: false,
|
||||||
|
wrapperStyle: '',
|
||||||
|
anchorStyle: ''
|
||||||
|
}
|
||||||
|
});
|
14
packages/index-anchor/index.wxml
Normal file
14
packages/index-anchor/index.wxml
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
<view
|
||||||
|
class="van-index-anchor-wrapper"
|
||||||
|
style="{{ wrapperStyle }}"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
class="van-index-anchor {{ active ? 'van-index-anchor--active van-hairline--bottom' : '' }}"
|
||||||
|
style="{{ anchorStyle }}"
|
||||||
|
>
|
||||||
|
<slot wx:if="{{ useSlot }}"/>
|
||||||
|
<block wx:else>
|
||||||
|
<text>{{ index }}</text>
|
||||||
|
</block>
|
||||||
|
</view>
|
||||||
|
</view>
|
110
packages/index-bar/README.md
Normal file
110
packages/index-bar/README.md
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
# IndexBar 索引栏
|
||||||
|
|
||||||
|
### 引入
|
||||||
|
|
||||||
|
在`app.json`或`index.json`中引入组件,详细介绍见[快速上手](#/quickstart#yin-ru-zu-jian)
|
||||||
|
|
||||||
|
```json
|
||||||
|
"usingComponents": {
|
||||||
|
"van-index-bar": "path/to/vant-weapp/dist/van-index-bar/index",
|
||||||
|
"van-index-anchor": "path/to/vant-weapp/dist/van-index-anchor/index"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 代码演示
|
||||||
|
|
||||||
|
### 基础用法
|
||||||
|
|
||||||
|
点击索引栏时,会自动跳转到对应的`IndexAnchor`锚点位置
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-index-bar scroll-top="{{ scrollTop }}">
|
||||||
|
<van-index-anchor index="A" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
|
||||||
|
<van-index-anchor index="B" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
|
||||||
|
...
|
||||||
|
</van-index-bar>
|
||||||
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
Page({
|
||||||
|
onPageScroll(event) {
|
||||||
|
this.setData({
|
||||||
|
scrollTop: event.scrollTop
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### 自定义索引列表
|
||||||
|
|
||||||
|
可以通过`index-list`属性自定义展示的索引字符列表,
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-index-bar index-list="{{ indexList }}">
|
||||||
|
<van-index-anchor index="1">标题1</van-index-anchor>
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
|
||||||
|
<van-index-anchor index="2">标题2</van-index-anchor>
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
<van-cell title="文本" />
|
||||||
|
|
||||||
|
...
|
||||||
|
</van-index-bar>
|
||||||
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
Page({
|
||||||
|
data: {
|
||||||
|
indexList: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
|
||||||
|
},
|
||||||
|
|
||||||
|
onPageScroll(event) {
|
||||||
|
this.setData({
|
||||||
|
scrollTop: event.scrollTop
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### IndexBar Props
|
||||||
|
|
||||||
|
| 参数 | 说明 | 类型 | 默认值 | 版本 |
|
||||||
|
|------|------|------|------|------|
|
||||||
|
| scroll-top | 当前滚动高度(自定义组件内部感知不到页面滚动,所以依赖接入方传入)| *Number* | 0 | - |
|
||||||
|
| index-list | 索引字符列表 | *string[] \| number[]* | `A-Z` | - |
|
||||||
|
| z-index | z-index 层级 | *number* | `1` | - |
|
||||||
|
| sticky | 是否开启锚点自动吸顶 | *boolean* | `true` | - |
|
||||||
|
| sticky-offset-top | 锚点自动吸顶时与顶部的距离 | *number* | `0` | - |
|
||||||
|
| highlight-color | 索引字符高亮颜色 | *string* | `#07c160` | - |
|
||||||
|
|
||||||
|
### IndexAnchor Props
|
||||||
|
|
||||||
|
| 参数 | 说明 | 类型 | 默认值 | 版本 |
|
||||||
|
|------|------|------|------|------|
|
||||||
|
| use-slot | 是否使用自定义内容的插槽 | *boolean* | `false` | - |
|
||||||
|
| index | 索引字符 | *string \| number* | - | - |
|
||||||
|
|
||||||
|
### IndexBar Events
|
||||||
|
|
||||||
|
| 事件名 | 说明 | 回调参数 |
|
||||||
|
|------|------|------|
|
||||||
|
| select | 选中字符时触发 | index: 索引字符 |
|
||||||
|
|
||||||
|
### IndexAnchor Slots
|
||||||
|
|
||||||
|
| 名称 | 说明 |
|
||||||
|
|------|------|
|
||||||
|
| default | 锚点位置显示内容,默认为索引字符 |
|
3
packages/index-bar/index.json
Normal file
3
packages/index-bar/index.json
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"component": true
|
||||||
|
}
|
24
packages/index-bar/index.less
Normal file
24
packages/index-bar/index.less
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
@import '../common/style/var.less';
|
||||||
|
@import '../common/style/theme.less';
|
||||||
|
|
||||||
|
.van-index-bar {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
&__sidebar {
|
||||||
|
position: fixed;
|
||||||
|
top: 50%;
|
||||||
|
right: 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
text-align: center;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__index {
|
||||||
|
font-weight: 500;
|
||||||
|
.theme(padding, '0 @padding-base 0 @padding-md');
|
||||||
|
.theme(font-size, '@index-bar-index-font-size');
|
||||||
|
.theme(line-height, '@index-bar-index-line-height');
|
||||||
|
}
|
||||||
|
}
|
308
packages/index-bar/index.ts
Normal file
308
packages/index-bar/index.ts
Normal file
@ -0,0 +1,308 @@
|
|||||||
|
import { VantComponent } from '../common/component';
|
||||||
|
import { GREEN } from '../common/color';
|
||||||
|
|
||||||
|
const indexList = () => {
|
||||||
|
const indexList = [];
|
||||||
|
const charCodeOfA = 'A'.charCodeAt(0);
|
||||||
|
|
||||||
|
for (let i = 0; i < 26; i++) {
|
||||||
|
indexList.push(String.fromCharCode(charCodeOfA + i));
|
||||||
|
}
|
||||||
|
|
||||||
|
return indexList;
|
||||||
|
};
|
||||||
|
|
||||||
|
VantComponent({
|
||||||
|
relation: {
|
||||||
|
name: 'index-anchor',
|
||||||
|
type: 'descendant',
|
||||||
|
linked() {
|
||||||
|
this.updateData();
|
||||||
|
},
|
||||||
|
linkChanged() {
|
||||||
|
this.updateData();
|
||||||
|
},
|
||||||
|
unlinked() {
|
||||||
|
this.updateData();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
sticky: {
|
||||||
|
type: Boolean,
|
||||||
|
value: true
|
||||||
|
},
|
||||||
|
zIndex: {
|
||||||
|
type: Number,
|
||||||
|
value: 1
|
||||||
|
},
|
||||||
|
highlightColor: {
|
||||||
|
type: String,
|
||||||
|
value: GREEN
|
||||||
|
},
|
||||||
|
scrollTop: {
|
||||||
|
type: Number,
|
||||||
|
value: 0,
|
||||||
|
observer: 'onScroll'
|
||||||
|
},
|
||||||
|
stickyOffsetTop: {
|
||||||
|
type: Number,
|
||||||
|
value: 0
|
||||||
|
},
|
||||||
|
indexList: {
|
||||||
|
type: Array,
|
||||||
|
value: indexList()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data: {
|
||||||
|
activeAnchorIndex: null,
|
||||||
|
showSidebar: false
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
updateData() {
|
||||||
|
this.timer && clearTimeout(this.timer);
|
||||||
|
|
||||||
|
this.timer = setTimeout(() => {
|
||||||
|
this.children = this.getRelationNodes('../index-anchor/index');
|
||||||
|
|
||||||
|
this.setData({
|
||||||
|
showSidebar: !!this.children.length
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setRect().then(() => {
|
||||||
|
this.onScroll();
|
||||||
|
});
|
||||||
|
}, 0);
|
||||||
|
},
|
||||||
|
|
||||||
|
setRect() {
|
||||||
|
return Promise.all([
|
||||||
|
this.setAnchorsRect(),
|
||||||
|
this.setListRect(),
|
||||||
|
this.setSiderbarRect()
|
||||||
|
]);
|
||||||
|
},
|
||||||
|
|
||||||
|
setAnchorsRect() {
|
||||||
|
return Promise.all(
|
||||||
|
this.children.map(anchor => (
|
||||||
|
anchor.getRect('.van-index-anchor-wrapper').then(
|
||||||
|
(rect: WechatMiniprogram.BoundingClientRectCallbackResult) => {
|
||||||
|
Object.assign(anchor, {
|
||||||
|
height: rect.height,
|
||||||
|
top: rect.top + this.data.scrollTop
|
||||||
|
});
|
||||||
|
}
|
||||||
|
)
|
||||||
|
))
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
setListRect() {
|
||||||
|
return this.getRect('.van-index-bar').then(
|
||||||
|
(rect: WechatMiniprogram.BoundingClientRectCallbackResult) => {
|
||||||
|
Object.assign(this, {
|
||||||
|
height: rect.height,
|
||||||
|
top: rect.top + this.data.scrollTop
|
||||||
|
});
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
setSiderbarRect() {
|
||||||
|
return this.getRect('.van-index-bar__sidebar').then(res => {
|
||||||
|
this.sidebar = {
|
||||||
|
height: res.height,
|
||||||
|
top: res.top
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
setDiffData({ target, data }) {
|
||||||
|
const diffData = {};
|
||||||
|
|
||||||
|
Object.keys(data).forEach(key => {
|
||||||
|
if (target.data[key] !== data[key]) {
|
||||||
|
diffData[key] = data[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (Object.keys(diffData).length) {
|
||||||
|
target.setData(diffData);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getAnchorRect(anchor) {
|
||||||
|
return anchor.getRect('.van-index-anchor-wrapper').then(
|
||||||
|
(rect: WechatMiniprogram.BoundingClientRectCallbackResult) => (
|
||||||
|
{
|
||||||
|
height: rect.height,
|
||||||
|
top: rect.top
|
||||||
|
}
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
getActiveAnchorIndex() {
|
||||||
|
const { children } = this;
|
||||||
|
const {
|
||||||
|
sticky,
|
||||||
|
scrollTop,
|
||||||
|
stickyOffsetTop
|
||||||
|
} = this.data;
|
||||||
|
|
||||||
|
for (let i = this.children.length - 1; i >= 0; i--) {
|
||||||
|
const preAnchorHeight = i > 0 ? children[i - 1].height : 0;
|
||||||
|
const reachTop = sticky ? preAnchorHeight + stickyOffsetTop : 0;
|
||||||
|
|
||||||
|
if (reachTop + scrollTop >= children[i].top) {
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
},
|
||||||
|
|
||||||
|
onScroll() {
|
||||||
|
const {
|
||||||
|
children = []
|
||||||
|
} = this;
|
||||||
|
|
||||||
|
if (!children.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const {
|
||||||
|
sticky,
|
||||||
|
stickyOffsetTop,
|
||||||
|
zIndex,
|
||||||
|
highlightColor,
|
||||||
|
scrollTop
|
||||||
|
} = this.data;
|
||||||
|
|
||||||
|
const active = this.getActiveAnchorIndex();
|
||||||
|
|
||||||
|
this.setDiffData({
|
||||||
|
target: this,
|
||||||
|
data: {
|
||||||
|
activeAnchorIndex: active
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (sticky) {
|
||||||
|
let isActiveAnchorSticky = false;
|
||||||
|
|
||||||
|
if (active !== -1) {
|
||||||
|
isActiveAnchorSticky = children[active].top <= stickyOffsetTop + scrollTop;
|
||||||
|
}
|
||||||
|
|
||||||
|
children.forEach((item, index) => {
|
||||||
|
if (index === active) {
|
||||||
|
let wrapperStyle = '';
|
||||||
|
let anchorStyle = `
|
||||||
|
color: ${highlightColor};
|
||||||
|
`;
|
||||||
|
|
||||||
|
if (isActiveAnchorSticky) {
|
||||||
|
wrapperStyle = `
|
||||||
|
height: ${children[index].height}px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
anchorStyle = `
|
||||||
|
position: fixed;
|
||||||
|
top: ${stickyOffsetTop}px;
|
||||||
|
z-index: ${zIndex};
|
||||||
|
color: ${highlightColor};
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setDiffData({
|
||||||
|
target: item,
|
||||||
|
data: {
|
||||||
|
active: true,
|
||||||
|
anchorStyle,
|
||||||
|
wrapperStyle
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (index === active - 1) {
|
||||||
|
const currentAnchor = children[index];
|
||||||
|
|
||||||
|
const currentOffsetTop = currentAnchor.top;
|
||||||
|
const targetOffsetTop = index === children.length - 1
|
||||||
|
? this.top
|
||||||
|
: children[index + 1].top;
|
||||||
|
|
||||||
|
const parentOffsetHeight = targetOffsetTop - currentOffsetTop;
|
||||||
|
const translateY = parentOffsetHeight - currentAnchor.height;
|
||||||
|
|
||||||
|
const anchorStyle = `
|
||||||
|
position: relative;
|
||||||
|
transform: translate3d(0, ${translateY}px, 0);
|
||||||
|
z-index: ${zIndex};
|
||||||
|
color: ${highlightColor};
|
||||||
|
`;
|
||||||
|
|
||||||
|
this.setDiffData({
|
||||||
|
target: item,
|
||||||
|
data: {
|
||||||
|
active: true,
|
||||||
|
anchorStyle
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.setDiffData({
|
||||||
|
target: item,
|
||||||
|
data: {
|
||||||
|
active: false,
|
||||||
|
anchorStyle: '',
|
||||||
|
wrapperStyle: '',
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
this.scrollToAnchor(event.target.dataset.index);
|
||||||
|
},
|
||||||
|
|
||||||
|
onTouchMove(event) {
|
||||||
|
const sidebarLength = this.children.length;
|
||||||
|
const touch = event.touches[0];
|
||||||
|
const itemHeight = this.sidebar.height / sidebarLength;
|
||||||
|
let index = Math.floor((touch.clientY - this.sidebar.top) / itemHeight);
|
||||||
|
|
||||||
|
if (index < 0) {
|
||||||
|
index = 0;
|
||||||
|
} else if (index > sidebarLength - 1) {
|
||||||
|
index = sidebarLength - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.scrollToAnchor(index);
|
||||||
|
},
|
||||||
|
|
||||||
|
onTouchStop() {
|
||||||
|
this.scrollToAnchorIndex = null;
|
||||||
|
},
|
||||||
|
|
||||||
|
scrollToAnchor(index) {
|
||||||
|
if (typeof index !== 'number' || this.scrollToAnchorIndex === index) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.scrollToAnchorIndex = index;
|
||||||
|
|
||||||
|
const anchor = this.children.filter(item => item.data.index === this.data.indexList[index])[0];
|
||||||
|
|
||||||
|
this.$emit('select', anchor.data.index);
|
||||||
|
|
||||||
|
anchor && wx.pageScrollTo({
|
||||||
|
duration: 0,
|
||||||
|
scrollTop: anchor.top
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
22
packages/index-bar/index.wxml
Normal file
22
packages/index-bar/index.wxml
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<view class="van-index-bar">
|
||||||
|
<slot />
|
||||||
|
|
||||||
|
<view
|
||||||
|
wx:if="{{ showSidebar }}"
|
||||||
|
class="van-index-bar__sidebar"
|
||||||
|
catch:tap="onClick"
|
||||||
|
catch:touchmove="onTouchMove"
|
||||||
|
catch:touchend="onTouchStop"
|
||||||
|
catch:touchcancel="onTouchStop"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
wx:for="{{ indexList }}"
|
||||||
|
wx:key="index"
|
||||||
|
class="van-index-bar__index"
|
||||||
|
style="z-index: {{ zIndex + 1 }}; color: {{ activeAnchorIndex === index ? highlightColor : '' }}"
|
||||||
|
data-index="{{ index }}"
|
||||||
|
>
|
||||||
|
{{ item }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
Loading…
x
Reference in New Issue
Block a user