鸿蒙开发:自定义切换动画实现Swiper层叠滑动效果

程序员一鸣
• 阅读 2

前言

本文基于Api13

Swiper组件默认情况下是无论是从左往右,还是从右往左,都是平移滑动,效果如下:

鸿蒙开发:自定义切换动画实现Swiper层叠滑动效果

以上的效果是最常见的,基本上都够满足大部分的场景需求,也是很多主流App首选效果,但是,仍然有一部分需求,会不按常理出牌,这不,让实现一个层叠滑动的效果就来了,所谓的层叠,就是,在滑动条目的时候,上一个不动,下一个覆盖到上一个的上面进行滑动,为了更好的便于大家理解,简单的做一个小动画,模拟演示一下。

正常的平移滑动效果,两个条目会一起进行平移。

鸿蒙开发:自定义切换动画实现Swiper层叠滑动效果

层叠效果滑动,可以看到,上一个条目是不会跟随移动的。

鸿蒙开发:自定义切换动画实现Swiper层叠滑动效果

那么如何实现上一个条目不随着手势进行移动呢?这里就需要用到自定义页面切换动画了,也就是在手势滑动的时候,保持当前条目不动,让下一个条目进行移动,还有一点需要注意的是,那就是需要设置组件的堆叠顺序,使其要滑动的组件层级高于前一个组件。

实现之前,我们先了解一下自定义Swiper页面切换动画customContentTransition。

customContentTransition

customContentTransition是Swiper组件中的一个属性,主要用于页面切换动画,接收一个SwiperContentAnimatedTransition参数;它的作用,主要在于,在页面随着手势进行切换动画的过程中,会对视窗内所有页面进行逐帧触发回调,在这个回调中,我们就可以针对条目进行设置透明度、缩放比例、位移等属性来自定义切换动画。

/**
    * Custom swiper content transition animation.
    *
    * @param { SwiperContentAnimatedTransition } transition - custom content transition animation.
    * @returns { SwiperAttribute } the attribute of the swiper.
    * @syscap SystemCapability.ArkUI.ArkUI.Full
    * @crossplatform
    * @atomicservice
    * @since 12
    */
   customContentTransition(transition: SwiperContentAnimatedTransition): SwiperAttribute;

有一点需要注意的是,customContentTransition是自定义切换动画,但不会取消原有的动画,也就是说,我们在自定义动画的时候,原有的动画也是同步的执行的,为了让原有的动画不执行,我们可以设置主轴方向上负的位移(translate属性)来抵消页面滑动,这也是我们这个需求的主要实现方式。

准备数据

通过前边,我们已经知道,要实现前言中的需求,我们需要做的有两个动作,动作一,动态修改子元素的位移,我们可以通过translate属性来控制,动作二,动态改变子元素的层级权重,通过属性zIndex来设置。

基本UI如下,当然了,这只是一个简单的Demo案例,实际开发中,属性以及数据设置,要根据实际情况而定。

Swiper() {
        ForEach(this.backgroundColors, (backgroundColor: Color, index: number) => {
          Text(index.toString())
            .width("100%")
            .height("100%")
            .backgroundColor(backgroundColor)
            .textAlign(TextAlign.Center)
            .translate({ x: this.translateList[index] })
            .zIndex(this.zIndexList[index])
        })
      }
      .height(200)

以上的案例,我简单定义了三个数组,一个用于数据加载的backgroundColors,一个用于位移的translateList,最后一个是用于设置层级权重的zIndexList。

实现自定义切换动画

实现这个层叠滑动,实现customContentTransition方法之前,有一点需要知道,那就是我们目前是往左滑动还是往右滑动,因为在最后一个条目节点需要做特殊的逻辑判断,往左还是往右,我们可以通过onGestureSwipe方法中偏移量进行判断。

extraInfo.currentOffset小于0,往左,否则往右。

.onGestureSwipe((_: number, extraInfo: SwiperAnimationEvent) => {
        let currentOffset = extraInfo.currentOffset
        this.isScrollLeft = currentOffset < 0
      })

在customContentTransition方法中,我们正常执行滑动抵消即可,首先是在同组页面往左滑或往右完全滑出视窗外时,我们需要重置属性值,这是非常重要的。

如何把当前的滑动进行抵消呢,很简单,当往左滑动时,使用页面在主轴方向上的长度减去移动比例乘以主轴长度即可,代码如下:viewIndex是当前的索引。

this.translateList[viewIndex] = proxy.mainAxisLength - proxy.position * proxy.mainAxisLength

往右滑动,直接负的移动比例乘以主轴长度,代码如下:

this.translateList[proxy.index] = -(proxy.position) * proxy.mainAxisLength

整个切换动画代码如下,在往左滑动的过程中,由于最后一个,需要动态切换为第一个索引,所以这里做了单独的逻辑处理。

.customContentTransition({
        timeout: 0,
        transition: (proxy: SwiperContentTransitionProxy) => {
          if (proxy.position <= 0 || proxy.position >= 1) {
            if (this.isScrollLeft) {
              if (proxy.index != this.translateList.length - 1) {
                this.translateList[proxy.index+1] = 0.0
                this.zIndexList[proxy.index+1] = 0
              }
            } else {
              this.translateList[proxy.index] = 0.0
              this.zIndexList[proxy.index] = 0
            }
          } else {
            if (this.isScrollLeft) {
              let viewIndex = proxy.index - 1
              if (viewIndex == -1) {
                this.translateList[this.translateList.length-1] =
                  proxy.mainAxisLength - proxy.position * proxy.mainAxisLength
                this.translateList[0] = 0
                this.zIndexList[0] = 1

              } else {
                this.translateList[viewIndex] = proxy.mainAxisLength - proxy.position * proxy.mainAxisLength
                this.zIndexList[viewIndex] = -1
              }
            } else {
              this.translateList[proxy.index] = -(proxy.position) * proxy.mainAxisLength
              this.zIndexList[proxy.index] = -1
            }
          }
        }
      })

通过以上的代码,一个层叠滑动效果就实现了,我们一起来看下最终的效果:

鸿蒙开发:自定义切换动画实现Swiper层叠滑动效果

相关总结

customContentTransition不仅仅可以实现平移上的改变,很多的效果,我们都可以实现,比如放大缩小,旋转等等。

本文标签:HarmonyOS/ArkUI

点赞
收藏
评论区
推荐文章
凯特林 凯特林
4年前
JS - 用 for 循环实现常见的数组迭代方法
常见的数组迭代方法有很多种,比如some,filter,map等等,底层也都可以用for来实现,我们来康一康。somejsconstsome(arr,fn)for(leti0;i<arr.length;i)if(fn(arri,i,arr))re
App复杂动画实现——Rive保姆级教程
在App开发过程中,如果想实现动画效果,可以粗略分为两种方式。一种是直接用代码编写,像平移、旋转等简单的动画效果,都可以这么干,如果稍微复杂点,就会对开发工程师的数学功底、图形图像学功底有很高的要求。
梦
4年前
微信小程序轮播图
实现效果wxml代码<viewstyle"height:20rpx;"</view<viewclass"swiper"
Wesley13 Wesley13
3年前
H5实现APP和原生方式有多大差距,多少坑?JS才是王道!
纯H5的APP,虽然开发起来要比纯原生开发畅快的多,但最终效果和性能还是和原生比起来还是有很多问题,主要聚集在以下几个方面:1、动画动画有很多种,比如侧边栏菜单的滑入滑出、元素的响应动画、页面切换之间的过场等等,在H5之下的众多实现方法都没有办法达到纯原生的性能。一般有这几种不同的选择:css3动画,javascript动画,原生动画。cs
陈杨 陈杨
2天前
鸿蒙5开发宝藏案例分享---体验流畅的首页信息流
🌟鸿蒙新闻类首页开发全攻略:流畅动效懒加载实战Hey各位鸿蒙开发者!今天要分享一个超实用的鸿蒙新闻类首页开发方案,官方文档里藏着的宝藏案例被我挖出来了!从流畅的Tab切换动效到丝滑的懒加载列表,手把手带大家实现高性能新闻首页👇🚀整体场景效果用户操作
布局王 布局王
16小时前
Uniapp开发鸿蒙购物项目实战教程:实现首页轮播图
过去几天的文章中我们讲过了如何创建跨平台项目,如何进行基础的布局、如何实现自定义导航栏等等,通过这一系列的文章教程,我们最终要实现一个购物app,今天我们要做购物应用首页的轮播图部分。对于轮播图uniapp提供了相应的组件swiper,并且是支持鸿蒙应用的
程序员一鸣 程序员一鸣
16小时前
鸿蒙开发:loading动画的几种实现方式
基本上没什么难的,都是非常简单的动画实现,虽然是一个loading动画,但是也可以应用与其他需要动画的地方。
美凌格栋栋酱 美凌格栋栋酱
5个月前
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(
程序员一鸣 程序员一鸣
2星期前
鸿蒙开发:实现AI打字机效果
需要注意的是,内容一般都是以markdown的形式输出,也就是真实的数据中,内容都是有样式的,比如加粗,图片,表格等等,所以,不能以单一的Text组件进行展示,需要针对markdown文本适配。
程序员一鸣 程序员一鸣
2天前
鸿蒙开发:沉浸式效果实现
沉浸式效果实现后,一定要注意安全区域的内容避让,防止内容延伸后被导航条或者状态栏遮挡,具体是选择安全区域或者窗口管理方式,按照需求进行处理,如果仅仅是某个页面,直接安全区域即可。