腾讯DeepOcean原创文章:dopro.io/egame-weex-…
随着电竞业务的不断发展,页面功能越来越多,交互逻辑更加复杂,类似无限滚动、上拉刷新、横竖切换滚动等形式在业务中已是标配,经过重重优化后在H5中的体验一直达不到理想状态,没错,种种卡,H5的性能太差! 是持续优化还是破而后立选择新的技术方向呢?我们选择了更有效的后者。
为什么选择weex
相对H5来说,weex带来的用户体验更好,它结合了H5和Native各自的优势,既能像H5一样快速迭代,又能和Native一样流畅。H5 | Weex | Native | |
---|---|---|---|
开发成本 | 低 | 中 | 高 |
维护更新 | 简单 | 简单 | 复杂 |
用户体验 | 差 | 优 | 优 |
发版审核 | 不需要 | 不需要 | 需要 |
跨平台性 | 优 | 优 | 差 |
H5与Weex的区别
- 项目结构
- 标签
- 引入sass
- sass变量
- 通用样式
- 布局
- 组件
- 动画
- UI性能
H5与Weex的区别
项目结构
由于weex和H5是两套不同的技术方案,代码组织方式、构建工具、和开发侧对接方式都会不同。下图是电竞重构稿H5与weex目录结构对比,之前H5开发是基于jinja模版,采用grunt构建,在release中生成相应的html文件,而weex则主要在src中开发组件,采用webpack编译,最终会在dist中生成相对应地web和weex版jsbundle文件,再由weex.html生成的二维码查看weex版页面效果。
此外weex下的src目录内容是与开发侧保持一致的,这样的好处在于开发人员只需要关注组件的结构变化,其它资源直接更新替换即可。
标签
weex只提供了17个组件,如div、text、image等,其中text和H5中p标签等同,文字只能放到text下,text中不能嵌套其他标签。引入sass
1、安装sass依赖的npm包:sass-loader、node-sass、sass-loader、stylus-loader2、在build文件夹下webpack.base.conf.js的rules里面添加配置
rules:{
{
test: /\.(scss|css)$/,
loader: 'vue-style-loader!css-loader!sass-loader!stylus-loader?indentedSyntax',
}
}er-width:$border.width;
}
sass变量
weex中的sass变量类似于对象的写法//variable.scss
$border={
color: #E9E9E9
width:2px
}
@import "./variable.scss";
.border{
border-color:$border.color;
border-width:$border.width;
}
weex ui开发踩坑
通用样式
1、border
weex不支持使用border创建三角形,web可以正常显示,而ios和android上显示的是矩形,建议使用图片代替web: ios、android:
2、transform
1、rotate角度尽量避免设置负数,某些部分安卓机型会不生效2、不支持transform:skew 对于这一类角标需要做倾斜处理可以采用 图片加 渐变代码处理
3、图片
1、weex提供了image组件,但只支持远程图片链接2、避免在image标签上使用v-for,否则会导致安卓上图片渲染异常(如slider中的图片)
4、透明度
以下是涉及到颜色的相关属性对透明度的支持度列表,注意:box-shadow (本身不支持android),background-image不支持IOS透明
属性 | IOS | Android | H5 |
---|---|---|---|
color | 支持 | 支持 | 支持 |
opacity | 支持 | 支持 | 支持 |
border-color | 支持 | 支持 | 支持 |
box-shadow | 支持 | 不支持 | 支持 |
background-color | 支持 | 支持 | 支持 |
background-image | 不支持 | 支持 | 支持 |
5、点击态
项目比较常见的点击态多半是透明度的变化,如按钮、列表、链接等,css的做法是添加伪类 (:active),weex中也同样支持,但是weex需要在原样式中添加 opacity:1,否则点击后回不到初始状态;此外,:active使用时,background-image在ios下会失效。<template>
<div class="ui-btn">
<text class="ui-btn-text">下载</text>
</div>
</template>
<style scoped>
.ui-btn{
opacity: 1; /*必须添加*/
}
.ui-btn:active{
opacity: .5;
}
</style>
6、文本截断
文本从限制1行到不限制可以用lines:0;<template>
<text class="info-text"
@click="textClick"
:style="textStyle">城市赛战报,《王者荣耀》城市赛郑州站欢乐落幕城市赛战报,《王者荣耀》城市赛郑州站欢乐落幕城市赛战报,《王者荣耀》城市赛郑州站欢乐落幕城市赛战报,《王者荣耀》城市赛郑州站欢乐落幕</text>
</template>
<style scoped>
.info-text{
lines:1;
text-overflow:ellipsis;
}
</style>
<script>
export default {
data(){
return {
textStyle:{}
};
},
methods:{
textClick(){
this.textStyle = {
lines:0
}
}
}
}
</script>
</style>
7、层级问题
例如有a、b、c、d 四层结构,其中a、b、c均为absolute定位,z-index由大到小,d为普通结构,我们知道在css中a层应该是处于最上方,d在最下方,那么在weex中表现如何呢?<div class="wrapper">
<div class="box a">a</div>
<div class="box b">b</div>
<div class="box c">c</div>
<div class="d">d</div>
</div>
可以看到web和ios、android的表现不一致,ios、android中是以代码中dom顺序来依次添加的,和z-index无关,后面加载的视图会覆盖前面的视图。
所以要保证web、ios、android三端表现一致,改变dom书写顺序即可。
<div class="d"></div>
<div class="box c"></div>
<div class="box b"></div>
<div class="box a"></div>
8、安卓下遮挡问题
安卓下容器如果设置了宽高,那么子元素不能超出容器范围[cc lang="html"]
[/cc]
web、ios android
建议:fixed定位不会受父容器影响,如果需要超出限制,子元素可以设置fixed
9、v-if问题
在做一些操作切换状态时(如按钮点击置灰),应尽量避免使用v-if,使用v-if会闪,且部分安卓机子会发生不可描述的事情(如部分三星机型会出现按钮文字居顶),可采用添加class的方式布局
1、单行文本与图片并排方案
目前项目中存在这样的情形,昵称与直播标签并排,昵称文字短时直播要跟随,昵称很长时要做溢出截断(超出时加...)这种布局方式在css中要做到很容易,而在weex中利用提供的flex布局确很难实现,最后的解决方案是通过js动态设置文字与标签父级的宽度,从而控制文字的溢出
<template>
<div class="wrapper" @appear="onappear">
<div class="info-container" ref='info-container'>
<div class="info" ref='info' :class="isFullText?['info-full']:[]">
<text class="nick-text">{{isFullText}}企鹅电竞企鹅电竞企鹅电竞企鹅电竞</text>
<img src="http://119.29.8.64/vipstyle/egame/app/weex/tab/ERICKCHEN-MC0/dist/static/img/live.b467410.png" class="live-image">
</div>
</div>
</div>
</template>
<style scoped lang="stylus">
.info-full{
width:300px;
}
</style>
<script>
const dom = weex.requireModule('dom')
export default {
data(){
return {
isFullText:false
};
},
methods:{
onappear(){
dom.getComponentRect(this.$refs['info'],option1=>{
dom.getComponentRect(this.$refs['info-container'],option2=>{
if(option1.size.width>=option2.size.width){
this.isFullText=true;
}
})
});
}
}
}
</script>
2、多行文本与图片并排方案
场景一:图片位于段落左侧
css的float可以做到图文混排,而weex只提供了flex布局,并且text组件之间也不能进行嵌套,无法做到这种图文混排效果,不过weex的text组件比较奇特,那就是text组件中的空格是照代码原样输出的,如
<text> 战国鬼才传,这个名字想必很多人听都没有听过吧,这个名字说实话真的不是很吸引人…</text>
所以解决的方案可以利用填充空格给图片预留位置,先计算一个空格的宽度,再计算这张图片所需要的空格数量,最后空格连接字符串输出。
ios、android效果如下(红色色块为图片区域):
结构
<template>
<div class="wrapper">
<scroller class="scroller">
<div @appear="handleAppear">
<text>空格宽度:{{spaceWidth}}-空格数量:{{spaceNum}}</text>
<text class="demo-text" ref="demo-text1"> test</text>
<text class="demo-text" ref="demo-text2">test</text>
</div>
<div class="rich">
<div class="rich-icon"></div>
<text class="rich-text" :style="textStyle">{{content}}</text>
</div>
</scroller>
</div>
</template>
<style scoped>
.demo-text{
position: absolute;
font-size: 32px;/*文字大小与需要加空格文字大小保持一致*/
opacity: 0;
}
.rich{
position: relative;
}
.rich-icon{
position: absolute;
left:0;
top:4px;
width: 120px;
height: 32px;
background-color: red;
}
.rich-text{
font-size: 32px;
}
</style>
<script>
const dom = weex.requireModule('dom');
export default {
data(){
return {
spaceWidth:0,//空格宽度
spaceNum:0,//所需空格数量
opacity:0,//初始透明度为0,避免文案抖动
content:'王者荣耀游戏中的钻石用来做什么最合算?王者荣耀钻石用来干什么最好?在王者荣耀中钻石并不是唯一通用的货币,在游戏中还有金币和点券,小编个人觉得钻石在游戏中并没有其他两种货币有优势。'
};
},
computed:{
textStyle(){
return {
opacity:this.opacity
}
}
},
methods:{
handleAppear(){
setTimeout(()=>{
this.setTextContent();
},30)
},
async setTextContent(){
const text1El = this.$refs['demo-text1'];
const text2El = this.$refs['demo-text2'];
let textSize1,textSize2;
await this.getSpaceSize(text1El,(data)=>{
textSize1 = data;
});
await this.getSpaceSize(text2El,(data)=>{
textSize2 = data;
});
this.spaceWidth=Math.abs(textSize1-textSize2)/10;
this.content=this.getSpaceNum();
this.opacity=1;
},
getSpaceSize(el,callback){
return new Promise(function (resolve) {
dom.getComponentRect(el, option => {
if(option.result){
resolve(callback(option.size.width));
}
});
})
},
getSpaceNum(){
this.spaceNum = Math.ceil(120 / this.spaceWidth);//120为红色区块宽度
return new Array(this.spaceNum).join(' ')+ this.content;
}
}
}
</script>
场景二:图片位于段落末尾
很遗憾,目前这种特殊文本以及图片置于段落末尾并没有找到相应的解决方案,只能依赖终端添加相应的富文本功能。
组件
1、命名
组件命名应避免使用JS关键字和保留字,以及weex提供的组件名称,如用loading作为组件名称,在ios与android中将呈现空白。错误示例:
<template>
<div class="wrapper">
<Loading></Loading>/*改用其它名称*/
</div>
</template>
<script>
import Loading from './demo'
export default {
components:{
Loading
}
}
</script>
2、自定义slider组件
weex本身提供了slider组件,但轮播图指示器(indicator)只能修改颜色与位置,大小却无法更改,所以需要自定义slider组件weex轮播图指示器效果:
电竞项目轮播图指示器效果
weex slider提供了change事件,可以获取到当前播放的序号,从而做到自定义轮播指示器。 但是当中遇到一个诡异问题:如果“div.indicator-item”的内容为空的话,H5中的指示器并不会随着图片切换而变化(样式不生效),div中需要添加内容才行
<div class="indicator-item" v-for="(item,index) in data" ...>
{{index}}/*添加内容,解决H5中class切换样式不生效问题*/
</div>
<template>
<div class="slider-container" :style="sliderStyle">
<slider class="slider" :interval="interval" @change="change" :auto-play="autoPlay">
<div class="slider-item" v-for="img in data" >
<image class="slider-image" :style="sliderStyle" resize="cover" :src="img.src"></image>
</div>
</slider>
<div class="slider-indicator">
<div class="indicator-item" v-for="(item,index) in data" :class="[current == index ? 'indicator-active' : '']">
{{index}}
</div>
</div>
</div>
</template>
动画
关键帧动画是很常见的一种动画,css3中可以利用@keyframes规则达到动画效果css3: [cc lang="css"]
[/cc] 效果如图:weex中提供了transition,可以传入相应的style,通过setInterval控制动画循环播放,但setInterval比较耗性能,建议终端对weex sdk进行改造,加入相应的循环播放功能
<template>
<div class="wrapper">
<div class="demo" ref="demo"></div>
</div>
</template>
<style scoped>
.demo{
width: 200px;
height: 200px;
background-color: gold;
}
</style>
<script>
import * as animation from './animation.js'
export default {
mounted() {
setTimeout(()=>{
setInterval(() => {
animation.run(this.$refs.demo);
}, 2100);
},200)
}
}
</script>
const animation = weex.requireModule('animation');
export function transition(el, opts,dd) {
let duration = dd || 400
if (!el) {
return Promise.resolve();
}
return new Promise(function (resolve) {
animation.transition(el, {
duration: duration,
timingFunction: 'linear',
delay: 0,
...opts
}, resolve);
})
}
export async function run(el, obj) {
await transition(el, {
styles: {
backgroundColor: 'red',
}
},0.0001)
await transition(el, {
styles: {
backgroundColor: 'purple',
}
},1000)
await transition(el, {
styles: {
backgroundColor: 'lime',
}
},1000)
}
UI性能
安卓下打开“调试GPU过度绘制”选项,打开之后选择“显示过度区域绘制”后,会发现手机界面基本被蓝色,淡绿,淡红,深红所填充,这几种颜色代表了不同程度的绘制情况,其中蓝色绘制最少,而深红色绘制最多,可能会造成页面卡顿,应避免出现大面积红色区域。优化建议:
1、尽量不要设置背景色
2、不要过度嵌套,结构尽量扁平化