21、鸿蒙/布局/创建轮播 (Swiper)

作者: 圆梦人生 | 来源:发表于2024-07-17 09:09 被阅读0次

    概述

    Swiper组件提供滑动轮播显示的能力。Swiper本身是一个容器组件,当设置了多个子组件后,可以对这些子组件进行轮播显示。通常,在一些应用首页显示推荐的内容时,需要用到轮播显示的能力。

    针对复杂页面场景,可以使用 Swiper 组件的预加载机制,利用主线程的空闲时间来提前构建和布局绘制组件,优化滑动体验。

    布局与约束

    Swiper作为一个容器组件,如果设置了自身尺寸属性,则在轮播显示过程中均以该尺寸生效。如果自身尺寸属性未被设置,则分两种情况:如果设置了prevMargin或者nextMargin属性,则Swiper自身尺寸会跟随其父组件;如果未设置prevMargin或者nextMargin属性,则会自动根据子组件的大小设置自身的尺寸。

    循环播放

    通过loop属性控制是否循环播放,该属性默认值为true。

    当loop为true时,在显示第一页或最后一页时,可以继续往前切换到前一页或者往后切换到后一页。如果loop为false,则在第一页或最后一页时,无法继续向前或者向后切换页面。

    • loop为true
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(true)
    
    • loop为false
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(false)
    

    自动轮播

    Swiper通过设置autoPlay属性,控制是否自动轮播子组件。该属性默认值为false。

    autoPlay为true时,会自动切换播放子组件,子组件与子组件之间的播放间隔通过interval属性设置。interval属性默认值为3000,单位毫秒。

    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(true).autoPlay(true).interval(1000)
    

    导航点样式

    Swiper提供了默认的导航点样式和导航点箭头样式,导航点默认显示在Swiper下方居中位置,开发者也可以通过indicator属性自定义导航点的位置和样式,导航点箭头默认不显示。

    通过indicator属性,开发者可以设置导航点相对于Swiper组件上下左右四个方位的位置,同时也可以设置每个导航点的尺寸、颜色、蒙层和被选中导航点的颜色。

    • 自定义导航点样式
      导航点直径设为30vp,左边距为0,导航点颜色设为红色。
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(true).autoPlay(true).interval(1000).indicator(
            Indicator.dot()
              .left(0)
              .itemWidth(15)
              .itemHeight(15)
              .selectedItemWidth(30)
              .selectedItemHeight(15)
              .color(Color.Red)
              .selectedColor(Color.Blue)
    )
    

    Swiper通过设置displayArrow属性,可以控制导航点箭头的大小、位置、颜色,底板的大小及颜色,以及鼠标悬停时是否显示箭头。

    • 箭头使用默认样式
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(true)
          .autoPlay(true).interval(1000)
          .displayArrow(true, false)
    
    • 自定义箭头样式
      箭头显示在组件两侧,大小为18vp,导航点箭头颜色设为蓝色。
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).loop(true)
          .autoPlay(true).interval(1000)
          .displayArrow({
            showBackground: true,
            isSidebarMiddle: true,
            backgroundSize: 24,
            backgroundColor: Color.White,
            arrowSize: 18,
            arrowColor: Color.Blue
          }, false)
    

    页面切换方式

    Swiper支持手指滑动、点击导航点和通过控制器三种方式切换页面,以下示例展示通过控制器切换页面的方法。

    private swiperController: SwiperController = new SwiperController();
    //
    Swiper(this.swiperController){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
          }.height(100).loop(true)
          .autoPlay(true).interval(3000)
          .displayArrow({
            showBackground: true,
            isSidebarMiddle: true,
            backgroundSize: 24,
            backgroundColor: Color.White,
            arrowSize: 18,
            arrowColor: Color.Blue
          }, false)
    
          Row({ space: 12 }) {
            Button('showPrevious')
              .onClick(() => {
                this.swiperController.showPrevious(); // 通过controller切换到前一页
              })
            Button('showNext')
              .onClick(() => {
                this.swiperController.showNext(); // 通过controller切换到后一页
              })
    
          }.margin(5)
    }.width('100%')
        .margin({ top: 5 })
    

    轮播方向

    Swiper支持水平和垂直方向上进行轮播,主要通过vertical属性控制。

    当vertical为true时,表示在垂直方向上进行轮播;为false时,表示在水平方向上进行轮播。vertical默认值为false。

    • 设置水平方向上轮播。
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100)
          .indicator(true).vertical(false)
    
    • 设置垂直方向轮播。
    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100)
          .indicator(true).vertical(true)
    

    每页显示多个子页面

    Swiper支持在一个页面内同时显示多个子组件,通过displayCount属性设置。

    Swiper(){
            Text('1')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('2')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('3')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    
            Text('4')
              .width('100%').height('100%').backgroundColor(Color.Gray).textAlign(TextAlign.Center)
    }.height(100).displayCount(2)
    

    自定义切换动画

    Swiper支持通过customContentTransition设置自定义切换动画,可以在回调中对视窗内所有页面逐帧设置透明度、缩放比例、位移、渲染层级等属性实现自定义切换动画。

    private DISPLAY_COUNT: number = 2
      private MIN_SCALE: number = 0.75
    
      @State backgroundColors: Color[] = [Color.Green, Color.Blue, Color.Yellow, Color.Pink, Color.Gray, Color.Orange]
      @State opacityList: number[] = []
      @State scaleList: number[] = []
      @State translateList: number[] = []
      @State zIndexList: number[] = []
    
      aboutToAppear(): void {
        for (let i = 0; i < this.backgroundColors.length; i++) {
          this.opacityList.push(1.0)
          this.scaleList.push(1.0)
          this.translateList.push(0.0)
          this.zIndexList.push(0)
        }
      }
    // ...
    Swiper() {
            ForEach(this.backgroundColors, (backgroundColor: Color, index: number) => {
              Text(index.toString()).width('100%').height('100%').fontSize(50).textAlign(TextAlign.Center)
                .backgroundColor(backgroundColor)
                .opacity(this.opacityList[index])
                .scale({ x: this.scaleList[index], y: this.scaleList[index] })
                .translate({ x: this.translateList[index] })
                .zIndex(this.zIndexList[index])
            })
          }
          .height(300)
          .indicator(false)
          .displayCount(this.DISPLAY_COUNT, true)
          .customContentTransition({
            timeout: 1000,
            transition: (proxy: SwiperContentTransitionProxy) => {
              if (proxy.position <= proxy.index % this.DISPLAY_COUNT || proxy.position >= this.DISPLAY_COUNT + proxy.index % this.DISPLAY_COUNT) {
                // 同组页面完全滑出视窗外时,重置属性值
                this.opacityList[proxy.index] = 1.0
                this.scaleList[proxy.index] = 1.0
                this.translateList[proxy.index] = 0.0
                this.zIndexList[proxy.index] = 0
              } else {
                // 同组页面未滑出视窗外时,对同组中左右两个页面,逐帧根据position修改属性值
                if (proxy.index % this.DISPLAY_COUNT === 0) {
                  this.opacityList[proxy.index] = 1 - proxy.position / this.DISPLAY_COUNT
                  this.scaleList[proxy.index] = this.MIN_SCALE + (1 - this.MIN_SCALE) * (1 - proxy.position / this.DISPLAY_COUNT)
                  this.translateList[proxy.index] = - proxy.position * proxy.mainAxisLength + (1 - this.scaleList[proxy.index]) * proxy.mainAxisLength / 2.0
                } else {
                  this.opacityList[proxy.index] = 1 - (proxy.position - 1) / this.DISPLAY_COUNT
                  this.scaleList[proxy.index] = this.MIN_SCALE + (1 - this.MIN_SCALE) * (1 - (proxy.position - 1) / this.DISPLAY_COUNT)
                  this.translateList[proxy.index] = - (proxy.position - 1) * proxy.mainAxisLength - (1 - this.scaleList[proxy.index]) * proxy.mainAxisLength / 2.0
                }
                this.zIndexList[proxy.index] = -1
              }
            }
          })
    }.width('100%')
    

    相关文章

      网友评论

        本文标题:21、鸿蒙/布局/创建轮播 (Swiper)

        本文链接:https://www.haomeiwen.com/subject/igpzcjtx.html