webview 和 React Native 中吸顶效果实现

473次阅读  |  发布于2年以前

一前言

在跨端开发中,离不开一些吸顶的交互场景,可以参考淘宝或是京东类电商 app 中一些 tab ,在整个容器滑动的过程中,吸顶效果非常的连贯和丝滑的,当然这些 tab 可能是用 native 开发的,但是跨端应用也能实现很不错的吸顶效果,那么今天我们就来研究一下跨端开发是如何实现吸顶的。

希望通过这篇文章,你将学习到:

创作不易,希望屏幕前的你能给笔者赏个赞,以此鼓励我继续创作前端硬文。

二 webview 吸顶实现方式

在移动端开发中,webview 已经成为很重要的一部分,比如 app 中内嵌的 web 页面,或者小程序的视图载体,本质上都是 webview。

基于 webview 的混合开发模式非常受到欢迎,回到今天的主题上来,在 webview 中如何实现吸顶效果呢?

2.1 position :sticky

webview 本质上就是 web 应用,所以我们可以使用 css 属性来做很多交互效果。如果说到吸顶效果,这里首先想到的就是 position:sticky 粘性属性。

position:sticky 是一个新的css3属性,它的表现类似于 position:relative 和 position:fixed 的交集。

通过上面可以得出,如果实现吸顶效果,设置一个 css 属性就能实现。

WechatIMG2225.jpeg

如上图所示,图中 head 部分是需要吸顶的内容,那么把 head 加上 position:sticky 就可以了。

sticky 的缺点:

当然 sticky 也有一些缺点:

2.2 scrollview

webview 还有一种实现吸顶的方法,就是通过 scrollview ,scrollview 是什么?scrollview 是一个滚动的容器组件,web 中并没有现成的 scrollview 组件,常见的 scrollview 组件主要存在小程序或者一些跨段解决方案中,比如 Taro 中的 Scrollview,这些组件并非是原生组件,都是在各个平台底层基于原生的 DOM 元素和 EventListener 封装的。

以微信小程序为例子,看一下 scroll-view 如何实现吸顶,这种方式主要是依靠计算的方式,来确定什么时候元素应该吸顶了。因为 scroll-view 上有回调函数 bindscroll ,可以实时的得到滚动的距离,使用滚动距离,可以推导出吸顶临界点,比如:

WechatIMG2243.jpeg

通过上面可以推导出 offsetTop === scrollTop 此时就是 current 吸顶的临界点。当然在不同场景下,这个临界点可以会有区别,但大体思路是不变的。

但是目前可能存在一些问题,就是如果我们继续通过 position:absolute 来触发吸顶的话,还会有 2.1 面临的问题——在 scroll-view 中使用了定位产生抖动,那么应该如何处理呢?

笔者在这里推荐大家一种方法就是,用两个吸顶模块,来模拟吸顶效果的实现:

WechatIMG2257.jpeg

用代码简单描述一下过程:

wxml中:

<current1 wx-if="{{ show }}" >
<scroll-view bindscroll="{{ handleScroll }}"  >
   <view class="hold" wx-if="{{ show }}" />
   <current2 wx-if="{{ !show }}" >
</scroll-view>

js 中:

/* 处理滚动事件 */
handleScroll(event){
    const { scrollTop } = event.detail
    const { offsetTop,show }= this.data
    const isCeiling = scrollTop === offsetTop
    if(isCeiling !== show ){
        /* 当吸顶状态发生变化时 */
        this.setData({
           show:isCeiling
        })
    }
}

这种方式实现吸顶也有一些缺点,就是当快速滑动的时候,比如小程序,因为触发吸顶调用 setData ,setData 底层会调用于 native 通信的方法,这样视图上的更新会滞后,直观上的感受就是置顶效果滞后。

三 React Native 中的吸顶方式

React Native 是跨端开发的一个解决方案,不同于 webview,webview 的渲染还是走 web 那一套,而 RN 这个一点就不同于 webview,采用了 Native 方式来渲染,所以就渲染性能上要优于 webview。

RN 中有很多中实现吸顶的方式,ScrollView ,FlatList ,和 SectionList 都能实现吸顶效果,

3.1 ScrollView 和 FlatList

ScrollView 和 FlatList 一般用于列表组件,两者中有一个stickyHeaderIndices 可以轻松实现吸顶效果。


<ScrollView
   stickyHeaderIndices={[0]}//第一个子元素即头部组件,上滑时吸顶  
/>

但是笔者在工作中,用到吸顶的场景,并不是单单列表中的某一个元素,有可能是视图中某一个 section 模块的头部。

所以接下来重点介绍一个场景,就是通过 SectionList 来实现吸顶效果。

3.2 SectionList 介绍及如何实现吸顶效果

SectionList 是高性能的分组(section)列表组件,支持下面这些常用的功能:

SectionList 顾名思义,就是分 Section 模块的列表。SectionList 的吸顶效果也是得益于一个属性——stickySectionHeadersEnabled。

当 stickySectionHeadersEnabled 为 true 的时候,当下一个 section 把它的前一个 section 的可视区推离屏幕的时候,让这个 section 的 header 粘连在屏幕的顶端。这个属性在 iOS 上是默认可用的,因为这是 iOS 的平台规范。

WechatIMG2259.jpeg

如上我们期望 section2 的 current 模块吸顶,那么当 section1 元素离开可视区域的时候,section2 的 current 就会吸顶了。这样说,有的同学可能不明白,我们来看一下具体使用。

具体使用:

const defaultSections = [
    { 
        data:[ name:'section1' ], 
        key:'section1',
    },
    { 
        data:[ name:'section2' ], 
        key:'section2',
    },
]
function Index(){
    //....省去一些逻辑
    const renderContent = ({ item:{ name } }) => ( name === 'section1' ? <Section1Content /> : <Section2Content />);
    /* 当只有 section2 有头部并且会吸顶 */
    const renderHeader = ({ section:{ key } }) => (key === 'section2' && <Current />)
    return <SectionList  
        sections={defaultSections}         // section 的配置项
        renderSectionHeader={renderHeader} // 分 section 渲染头部
        renderItem={renderContent}         // 分 section 渲染主体内容
        stickySectionHeadersEnabled        // 设置吸顶状态为 true
    />
}

四 总结

本文介绍了跨端开发中,webview 和 React Native 实现吸顶的主流方式,希望能给做此类功能的同学提供一个解决思路。

参考资料

[1]https://juejin.cn/post/7112770927082864653: https://juejin.cn/post/7112770927082864653

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8