perf(tab): improve wxs style assemble & export resize methods (#3827)

fix #3676
This commit is contained in:
rex 2020-12-09 13:53:58 +08:00 committed by GitHub
parent 1104e07fd7
commit 0cd64e92e5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 170 additions and 146 deletions

View File

@ -1,4 +1,5 @@
<wxs src="../wxs/utils.wxs" module="utils" /> <wxs src="../wxs/utils.wxs" module="utils" />
<wxs src="./index.wxs" module="computed" />
<van-overlay <van-overlay
wx:if="{{ overlay }}" wx:if="{{ overlay }}"
@ -11,7 +12,7 @@
<view <view
wx:if="{{ inited }}" wx:if="{{ inited }}"
class="custom-class {{ classes }} {{ utils.bem('popup', [position, { round, safe: safeAreaInsetBottom, safeTop: safeAreaInsetTop }]) }}" class="custom-class {{ classes }} {{ utils.bem('popup', [position, { round, safe: safeAreaInsetBottom, safeTop: safeAreaInsetTop }]) }}"
style="z-index: {{ zIndex }}; -webkit-transition-duration:{{ currentDuration }}ms; transition-duration:{{ currentDuration }}ms; {{ display ? '' : 'display: none;' }};{{ customStyle }}" style="{{ computed.popupClass({ zIndex, currentDuration, display, customStyle }) }}"
bind:transitionend="onTransitionEnd" bind:transitionend="onTransitionEnd"
> >
<slot /> <slot />

18
packages/popup/index.wxs Normal file
View File

@ -0,0 +1,18 @@
/* eslint-disable */
var style = require('../wxs/style.wxs');
function popupClass(data) {
return style([
{
'z-index': data.zIndex,
'-webkit-transition-duration': data.currentDuration + 'ms',
'transition-duration': data.currentDuration + 'ms',
},
data.display ? null : 'display: none',
data.customStyle,
]);
}
module.exports = {
popupClass: popupClass,
};

View File

@ -1,37 +1,24 @@
/* eslint-disable */ /* eslint-disable */
var style = require('../wxs/style.wxs');
function wrapStyle(data) { function wrapStyle(data) {
var style = ''; return style({
transform: data.transform
if (data.transform) { ? 'translate3d(0, ' + data.transform + 'px, 0)'
style += 'transform: translate3d(0, ' + data.transform + 'px, 0);'; : '',
} top: data.fixed ? data.offsetTop + 'px' : '',
'z-index': data.zIndex,
if (data.fixed) { });
style += 'top: ' + data.offsetTop + 'px;';
}
if (data.zIndex) {
style += 'z-index: ' + data.zIndex + ';';
}
return style;
} }
function containerStyle(data) { function containerStyle(data) {
var style = ''; return style({
height: data.fixed ? data.height + 'px' : '',
if (data.fixed) { 'z-index': data.zIndex,
style += 'height: ' + data.height + 'px;'; });
}
if (data.zIndex) {
style += 'z-index: ' + data.zIndex + ';';
}
return style;
} }
module.exports = { module.exports = {
wrapStyle: wrapStyle, wrapStyle: wrapStyle,
containerStyle: containerStyle containerStyle: containerStyle,
}; };

View File

@ -181,38 +181,64 @@ Page({
</van-popup> </van-popup>
``` ```
## 常见问题
### 组件从隐藏状态切换到显示状态时,底部条位置错误?
Tabs 组件在挂载时,会获取自身的宽度,并计算出底部条的位置。如果组件一开始处于隐藏状态,则获取到的宽度永远为 0因此无法展示底部条位置。
#### 解决方法
方法一,使用 `v-if` 来控制组件展示,使组件重新初始化:
```html
<van-tabs v-if="show" />
```
方法二,调用组件的 resize 方法来主动触发重绘:
```html
<van-tabs id="tabs" />
```
```js
this.selectComponent('#tabs').resize();
```
## API ## API
### Tabs Props ### Tabs Props
| 参数 | 说明 | 类型 | 默认值 | 版本 | | 参数 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- | --- | | --- | --- | --- | --- |
| active | 当前选中标签的标识符 | _string \| number_ | `0` | - | | type | 样式风格,可选值为`card` | _string_ | `line` |
| color | 标签颜色 | _string_ | `#ee0a24` | - | | color | 标签主题色 | _string_ | `#ee0a24` |
| z-index | z-index 层级 | _number_ | `1` | - | | active | 当前选中标签的标识符 | _string \| number_ | `0` |
| type | 样式风格,可选值为`card` | _string_ | `line` | - | | duration | 动画时间,单位秒 | _number_ | `0.3` |
| border | 是否展示外边框,仅在 `line` 风格下生效 | _boolean_ | `false` | - | | line-width | 底部条宽度,默认单位`px` | _string \| number_ | `40px` |
| duration | 动画时间 (单位秒) | _number_ | `0.3` | - | | line-height | 底部条高度,默认单位`px` | _string \| number_ | `3px` |
| line-width | 底部条宽度 (px) | _string \| number_ | `40px` | - | | animated | 是否开启切换标签内容时的转场动画 | _boolean_ | `false` |
| line-height | 底部条高度 (px) | _string \| number_ | `3px` | - | | border | 是否展示外边框,仅在 `line` 风格下生效 | _boolean_ | `false` |
| swipe-threshold | 滚动阈值,设置标签数量超过多少个可滚动 | _number_ | `5` | - | | ellipsis | 是否省略过长的标题文字 | _boolean_ | `true` |
| animated | 是否使用动画切换 Tabs | _boolean_ | `false` | - | | sticky | 是否使用粘性定位布局 | _boolean_ | `false` |
| ellipsis | 是否省略过长的标题文字 | _boolean_ | `true` | - | | swipeable | 是否开启手势滑动切换 | _boolean_ | `false` |
| sticky | 是否使用粘性定位布局 | _boolean_ | `false` | - | | lazy-render | 是否开启标签页内容延迟渲染 | _boolean_ | `true` |
| swipeable | 是否开启手势滑动切换 | _boolean_ | `false` | - | | offset-top | 粘性定位布局下与顶部的最小距离,单位`px` | _number_ | - |
| lazy-render | 是否开启标签页内容延迟渲染 | _boolean_ | `true` | - | | swipe-threshold | 滚动阈值,标签数量超过阈值且总宽度超过标签栏宽度时开始横向滚动 | _number_ | `5` |
| offset-top | 粘性定位布局下与顶部的最小距离,单位`px` | _number_ | - | - | | title-active-color | 标题选中态颜色 | _string_ | - |
| title-inactive-color | 标题默认态颜色 | _string_ | - |
| z-index | z-index 层级 | _number_ | `1` |
### Tab Props ### Tab Props
| 参数 | 说明 | 类型 | 默认值 | 版本 | | 参数 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- | --- | | ----------- | -------------------------- | ------------------ | ------------ |
| name | 标签名称,作为匹配的标识符 | _string \| number_ | 标签的索引值 | - | | name | 标签名称,作为匹配的标识符 | _string \| number_ | 标签的索引值 |
| title | 标题 | _string_ | - | - | | title | 标题 | _string_ | - |
| disabled | 是否禁用标签 | _boolean_ | `false` | - | | disabled | 是否禁用标签 | _boolean_ | `false` |
| dot | 是否显示小红点 | _boolean_ | - | - | | dot | 是否显示小红点 | _boolean_ | - |
| info | 图标右上角提示信息 | _string \| number_ | - | - | | info | 图标右上角提示信息 | _string \| number_ | - |
| title-style | 自定义标题样式 | _string_ | - | - | | title-style | 自定义标题样式 | _string_ | - |
### Tabs Slot ### Tabs Slot
@ -244,3 +270,11 @@ Page({
| nav-class | 标签栏样式类 | | nav-class | 标签栏样式类 |
| tab-class | 标签样式类 | | tab-class | 标签样式类 |
| tab-active-class | 标签激活态样式类 | | tab-active-class | 标签激活态样式类 |
### 方法
通过 selectComponent 可以获取到 Tabs 实例并调用实例方法
| 方法名 | 参数 | 返回值 | 介绍 |
| --- | --- | --- | --- |
| resize | - | - | 外层元素大小或组件显示状态变化时,可以调用此方法来触发重绘 |

View File

@ -46,7 +46,7 @@ VantComponent({
lineWidth: { lineWidth: {
type: [String, Number], type: [String, Number],
value: 40, value: 40,
observer: 'setLine', observer: 'resize',
}, },
lineHeight: { lineHeight: {
type: [String, Number], type: [String, Number],
@ -110,7 +110,7 @@ VantComponent({
mounted() { mounted() {
wx.nextTick(() => { wx.nextTick(() => {
this.setLine(true); this.resize(true);
this.scrollIntoView(); this.scrollIntoView();
}); });
}, },
@ -201,7 +201,7 @@ VantComponent({
this.setData({ currentIndex }); this.setData({ currentIndex });
wx.nextTick(() => { wx.nextTick(() => {
this.setLine(); this.resize();
this.scrollIntoView(); this.scrollIntoView();
this.updateContainer(); this.updateContainer();
@ -220,7 +220,7 @@ VantComponent({
} }
}, },
setLine(skipTransition = false) { resize(skipTransition = false) {
if (this.data.type !== 'line') { if (this.data.type !== 'line') {
return; return;
} }

View File

@ -1,5 +1,5 @@
<wxs src="../wxs/utils.wxs" module="utils" /> <wxs src="../wxs/utils.wxs" module="utils" />
<wxs src="./index.wxs" module="getters" /> <wxs src="./index.wxs" module="computed" />
<view class="custom-class {{ utils.bem('tabs', [type]) }}"> <view class="custom-class {{ utils.bem('tabs', [type]) }}">
<van-sticky <van-sticky
@ -19,14 +19,14 @@
class="{{ utils.bem('tabs__scroll', [type]) }}" class="{{ utils.bem('tabs__scroll', [type]) }}"
style="{{ color ? 'border-color: ' + color : '' }}" style="{{ color ? 'border-color: ' + color : '' }}"
> >
<view class="{{ utils.bem('tabs__nav', [type, { complete: !ellipsis }]) }} nav-class" style="{{ getters.tabCardTypeBorderStyle(color, type) }}"> <view class="{{ utils.bem('tabs__nav', [type, { complete: !ellipsis }]) }} nav-class" style="{{ computed.navStyle(color, type) }}">
<view wx:if="{{ type === 'line' }}" class="van-tabs__line" style="{{ getters.lineStyle({ color, lineOffsetLeft, lineHeight, skipTransition, duration, lineWidth }) }}" /> <view wx:if="{{ type === 'line' }}" class="van-tabs__line" style="{{ computed.lineStyle({ color, lineOffsetLeft, lineHeight, skipTransition, duration, lineWidth }) }}" />
<view <view
wx:for="{{ tabs }}" wx:for="{{ tabs }}"
wx:key="index" wx:key="index"
data-index="{{ index }}" data-index="{{ index }}"
class="{{ getters.tabClass(index === currentIndex, ellipsis) }} {{ utils.bem('tab', { active: index === currentIndex, disabled: item.disabled, complete: !ellipsis }) }}" class="{{ computed.tabClass(index === currentIndex, ellipsis) }} {{ utils.bem('tab', { active: index === currentIndex, disabled: item.disabled, complete: !ellipsis }) }}"
style="{{ getters.tabStyle(index === currentIndex, ellipsis, color, type, item.disabled, titleActiveColor, titleInactiveColor, swipeThreshold, scrollable) }}" style="{{ computed.tabStyle({ active: index === currentIndex, ellipsis, color, type, disabled: item.disabled, titleActiveColor, titleInactiveColor, swipeThreshold, scrollable }) }}"
bind:tap="onTap" bind:tap="onTap"
> >
<view class="{{ ellipsis ? 'van-ellipsis' : '' }}" style="{{ item.titleStyle }}"> <view class="{{ ellipsis ? 'van-ellipsis' : '' }}" style="{{ item.titleStyle }}">
@ -55,7 +55,7 @@
> >
<view <view
class="{{ utils.bem('tabs__track', [{ animated }]) }} van-tabs__track" class="{{ utils.bem('tabs__track', [{ animated }]) }} van-tabs__track"
style="{{ getters.trackStyle({ duration, currentIndex, animated }) }}" style="{{ computed.trackStyle({ duration, currentIndex, animated }) }}"
> >
<slot /> <slot />
</view> </view>

View File

@ -1,5 +1,6 @@
/* eslint-disable */ /* eslint-disable */
var utils = require('../wxs/utils.wxs'); var utils = require('../wxs/utils.wxs');
var style = require('../wxs/style.wxs');
function tabClass(active, ellipsis) { function tabClass(active, ellipsis) {
var classes = ['tab-class']; var classes = ['tab-class'];
@ -15,51 +16,33 @@ function tabClass(active, ellipsis) {
return classes.join(' '); return classes.join(' ');
} }
function tabStyle( function tabStyle(data) {
active, var titleColor = data.active
ellipsis, ? data.titleActiveColor
color, : data.titleInactiveColor;
type,
disabled, var ellipsis = data.scrollable && data.ellipsis;
activeColor,
inactiveColor,
swipeThreshold,
scrollable
) {
var styles = [];
var isCard = type === 'card';
// card theme color // card theme color
if (color && isCard) { if (data.type === 'card') {
styles.push('border-color:' + color); return style({
'border-color': data.color,
if (!disabled) { 'background-color': !data.disabled && data.active ? data.color : null,
if (active) { color: titleColor || (!data.disabled && !data.active ? data.color : null),
styles.push('background-color:' + color); 'flex-basis': ellipsis ? 88 / data.swipeThreshold + '%' : null,
} else { });
styles.push('color:' + color);
}
}
} }
var titleColor = active ? activeColor : inactiveColor; return style({
if (titleColor) { color: titleColor,
styles.push('color:' + titleColor); 'flex-basis': ellipsis ? 88 / data.swipeThreshold + '%' : null,
} });
if (scrollable && ellipsis) {
styles.push('flex-basis:' + 88 / swipeThreshold + '%');
}
return styles.join(';');
} }
function tabCardTypeBorderStyle(color, type) { function navStyle(color, type) {
var isCard = type === 'card'; return style({
var styles = []; 'border-color': type === 'card' && color ? color : null,
if (isCard && color) { });
styles.push('border-color:' + color);
}
return styles.join(';');
} }
function trackStyle(data) { function trackStyle(data) {
@ -67,43 +50,27 @@ function trackStyle(data) {
return ''; return '';
} }
return [ return style({
['left', -100 * data.currentIndex + '%'], left: -100 * data.currentIndex + '%',
['-webkit-transition-duration', data.duration + 's'], 'transition-duration': data.duration + 's',
['transition-duration: ', data.duration + 's'], '-webkit-transition-duration': data.duration + 's',
] });
.map(function (item) {
return item.join(':');
})
.join(';');
} }
function lineStyle(data) { function lineStyle(data) {
var styles = [ return style({
['width', utils.addUnit(data.lineWidth)], width: utils.addUnit(data.lineWidth),
['transform', 'translateX(' + data.lineOffsetLeft + 'px)'], transform: 'translateX(' + data.lineOffsetLeft + 'px)',
['-webkit-transform', 'translateX(' + data.lineOffsetLeft + 'px)'], '-webkit-transform': 'translateX(' + data.lineOffsetLeft + 'px)',
]; 'background-color': data.color,
height: data.lineHeight !== -1 ? utils.addUnit(data.lineHeight) : null,
if (data.color) { 'border-radius':
styles.push(['background-color', data.color]); data.lineHeight !== -1 ? utils.addUnit(data.lineHeight) : null,
} 'transition-duration': !data.skipTransition ? data.duration + 's' : null,
'-webkit-transition-duration': !data.skipTransition
if (data.lineHeight !== -1) { ? data.duration + 's'
styles.push(['height', utils.addUnit(data.lineHeight)]); : null,
styles.push(['border-radius', utils.addUnit(data.lineHeight)]); });
}
if (!data.skipTransition) {
styles.push(['transition-duration', data.duration + 's']);
styles.push(['-webkit-transition-duration', data.duration + 's']);
}
return styles
.map(function (item) {
return item.join(':');
})
.join(';');
} }
module.exports = { module.exports = {
@ -111,5 +78,5 @@ module.exports = {
tabStyle: tabStyle, tabStyle: tabStyle,
trackStyle: trackStyle, trackStyle: trackStyle,
lineStyle: lineStyle, lineStyle: lineStyle,
tabCardTypeBorderStyle: tabCardTypeBorderStyle, navStyle: navStyle,
}; };

View File

@ -1,15 +1,32 @@
/* eslint-disable */ /* eslint-disable */
var object = require('./object.wxs'); var object = require('./object.wxs');
var array = require('./array.wxs');
function style(styles) { function style(styles) {
return object.keys(styles) if (array.isArray(styles)) {
.filter(function (key) { return styles
return styles[key] != null; .filter(function (item) {
}) return item != null;
.map(function (key) { })
return [key, [styles[key]]].join(':'); .map(function (item) {
}) return style(item);
.join(';'); })
.join(';');
}
if ('Object' === styles.constructor) {
return object
.keys(styles)
.filter(function (key) {
return styles[key] != null;
})
.map(function (key) {
return [key, [styles[key]]].join(':');
})
.join(';');
}
return styles;
} }
module.exports = style; module.exports = style;