首页 > 其他分享 >如何通过显示动画实现书籍翻页动效(OpenHarmony)

如何通过显示动画实现书籍翻页动效(OpenHarmony)

时间:2023-05-18 22:04:54浏览次数:64  
标签:OpenHarmony rotate 翻页 State number 动效 num 组件 position

如何通过显示动画实现书籍翻页动效

场景介绍

翻页动效是应用开发中常见的动效场景,常见的如书籍翻页、日历翻页等。本文就为大家举例讲解如何通过ArkUI提供的显示动画接口animateTo实现书籍翻页的效果。

效果呈现

本例最终实现效果如下:

翻页动效示例图.gif

环境要求

  • IDE:DevEco Studio 3.1 Beta1
  • SDK:Ohos_sdk_public 3.2.11.9 (API Version 9 Release)

实现思路

如图,分上下两层、左右两侧建立4个文本组件(下文用A、B、C、D代称),左右两侧分别代表打开书籍的左右两面,上下两层堆叠放置。 当B沿旋转轴旋转180度覆盖在A上时,就体现为翻页效果。一个翻页动作的完成包括以下几步:

  1. B沿旋转轴旋转180度。
  2. B旋转时,D会在右侧显示出来,作为书籍的下一页,此时D承载的内容要变为下一页的内容。
  3. B旋转到左侧后,A承载的内容变为B的内容。
  4. 由于A和B互为镜像,所以A显示为B的内容后,需要以A的中间为轴旋转180度。
  5. B重新旋转到右边,其承载的内容变为下一页的内容。

说明:C用来占位,不需要做动作。 连续重复上述动作即可实现连续翻页动效。

翻页动效.png

开发步骤

  1. 创建文本组件。

    动效中用到了4个文本组件,因此可以先定义一个文本组件,然后对其进行重复调用。同时为文本组件添加rotate属性,用来控制组件的旋转。 由于各组件旋转的角度和旋转中心不同,需要父组件在调用时传入对应的参数,所以需要为对应变量添加@Prop装饰器,用来控制变量传递。具体代码如下:

    @Component
    struct BookCard{
      // 为变量添加@Prop装饰器,用于接收父组件的动态传参
      @Prop num:number
      @Prop y_position:string
      @Prop x_position:string
      @Prop rotate_angle:number
      build(){
        Text(`${this.num}`)
          .fontWeight(FontWeight.Bold)
          .backgroundColor('#18183C')
          .fontColor('white')
          .fontSize(80)
          .width('25%')
          .height('30%')
          .fontFamily('Monospace')
          .textAlign(TextAlign.Center)
          .borderRadius(20)
          // 使用rotate属性控制旋转
          .rotate({
            x: 0,
            y: 1,
            z: 0,
            angle: this.rotate_angle,
            centerY: this.y_position,
            centerX: this.x_position
          })
      }
    }
    
  2. 创建父组件框架。

    由于文本组件分为上下两层,所以在父组件中采用Stack组件进行层叠布局。同时使用Divider组件作为书籍两个页面间的分隔线。具体代码如下:

    @Entry
    @Component
    struct BookAnimation {
    
      build(){
        Stack(){
          Row(){
            // 组件C
            BookCard()
            // 组件D
            BookCard()
          }
          Row(){
            // 组件A
            BookCard()
            // 组件B
            BookCard()
          }
          // 添加两个页面间的分隔线
          Divider()
          .strokeWidth(5)
          .color('white')
          .height('26%')
          .vertical(true)
        }
        .width('100%')
        .height('100%')
        .backgroundColor('#A4AE77')
      }
    }
    
  3. 添加翻页动效。

    最后通过以下几点来为静态的组件添加动效:

    • 根据实现思路章节的分析,在父组件中定义对应的变量,并在调用子组件时分别传入子组件。
    • 自定义book_animate函数,在其中使用animateTo方法添加动画效果,同时控制动画的时长,以及动画过程中各元素状态的改变。
    • aboutToAppear方法中,使用setInterval方法重复调用book_animate函数,以实现连续翻页动效。 具体代码如下:
    @Entry
    @Component
    struct BookAnimation {
      // 父组件变量设置,注意使用@State做状态管理
      @State rotate_angle1:number = 0
      @State rotate_angle2:number = 0
      @State rotate_angle3:number = 0
      @State num_before: number = 0;
      @State num: number = 1;
      @State num_next: number = 0;
      @State y_center1:string = '50%'
      @State x_center1:string = '50%'
      @State y_center2:string = '0%'
      @State x_center2:string = '0%'
    
      // 在UI显示前,传入各项变量的具体值
      aboutToAppear() {
        // 通过setInterval函数每秒调用一次动画效果,实现连续翻页
        setInterval(() => {
          this.book_animate()
        }, 1000)//函数调用周期要大于每次动画持续的时长
      }
    
      private book_animate(){
        // 通过animateTo方法为组件添加动效,动效时长要小于setInterval函数调用周期
        animateTo({ duration:700,onFinish:()=>{
          // 动画结束时,A显示的数字跟B显示的数字相等
          this.num_before = this.num
          // 动画结束时,A以中心线为轴旋转180度
          this.rotate_angle3 = 180
          // 动画结束时,B返回至初始状态
          this.rotate_angle1 = 0
          // 动画结束时,B显示的数字加1
          this.num = (this.num + 1) % 10
        }
        },()=>{
          // 动画开始,B的旋转角度变为180度
          this.rotate_angle1 = 180
          // 动画开始,D的数字加1
          this.num_next = this.num+1
        })
      }
      build() {
        Stack(){
          Row(){
            // C组件的引用配置
            BookCard({num:0,rotate_angle:this.rotate_angle2,
            y_position:this.y_center2,x_position:this.x_center2})
            // D组件的引用配置
            BookCard({num:this.num_next,rotate_angle:this.rotate_angle2,
            y_position:this.y_center2,x_position:this.x_center2})
          }
          Row(){
            // A组件的引用配置
            BookCard({num:this.num_before,rotate_angle:this.rotate_angle3,
            y_position:this.y_center1,x_position:this.x_center1})
            // B组件的引用配置
            BookCard({num:this.num,rotate_angle:this.rotate_angle1,
            y_position:this.y_center2,x_position:this.x_center2})
          }
          Divider().strokeWidth(5).color('white').height('26%').vertical(true)
        }.width('100%').height('50%').backgroundColor('#A4AE77')
      }
    }
    

    通过以上步骤就可以实现翻页动效了。

完整代码

示例完整代码如下:

@Component
struct BookCard{
  @Prop num:number
  @Prop y_position:string
  @Prop x_position:string
  @Prop rotate_angle:number
  build(){
    Text(`${this.num}`)
      .fontWeight(FontWeight.Bold)
      .backgroundColor('#18183C')
      .fontColor('white')
      .fontSize(80)
      .width('25%')
      .height('30%')
      .fontFamily('Monospace')
      .textAlign(TextAlign.Center)
      .borderRadius(20)
      .rotate({
        x: 0,
        y: 1,
        z: 0,
        angle: this.rotate_angle,
        centerY: this.y_position,
        centerX: this.x_position
      })
  }
}


@Entry
@Component
struct BookAnimation {
  @State rotate_angle1:number = 0
  @State rotate_angle2:number = 0
  @State rotate_angle3:number = 0
  @State num_before: number = 0;
  @State num: number = 1;
  @State num_next: number = 0;
  @State y_center1:string = '50%'
  @State x_center1:string = '50%'
  @State y_center2:string = '0%'
  @State x_center2:string = '0%'


  aboutToAppear() {
    setInterval(() => {
      this.book_animate()
    }, 1000)
  }

  private book_animate(){
    animateTo({ duration:700,onFinish:()=>{
      this.num_before = this.num
      this.rotate_angle3 = 180
      this.rotate_angle1 = 0
      this.num = (this.num + 1) % 10
    }
    },()=>{
      this.rotate_angle1 = 180
      this.num_next = this.num+1
    })
  }


  build() {
    Stack(){
      Row(){
        BookCard({num:0,rotate_angle:this.rotate_angle2,y_position:this.y_center2,
        x_position:this.x_center2})
        BookCard({num:this.num_next,rotate_angle:this.rotate_angle2,y_position:this.y_center2,
        x_position:this.x_center2})
      }
      Row(){
        BookCard({num:this.num_before,rotate_angle:this.rotate_angle3,y_position:this.y_center1,
        x_position:this.x_center1})
        BookCard({num:this.num,rotate_angle:this.rotate_angle1,y_position:this.y_center2,
        x_position:this.x_center2})
      }
      Divider().strokeWidth(5).color('white').height('26%').vertical(true)
    }.width('100%').height('50%').backgroundColor('#A4AE77')
  }
}

标签:OpenHarmony,rotate,翻页,State,number,动效,num,组件,position
From: https://blog.51cto.com/u_14452884/6307206

相关文章

  • OpenHarmony Docker移植实践
      Docker简介从操作系统诞生之日起,虚拟化技术就不断的演进与发展,结合目前云原生的发展态势,容器无疑是其中的重要一环。Docker是一个开源的软件项目,可以在Linux操作系统上提供一层额外的抽象,让用户程序部署在一个相对隔离的运行环境,并提供自动管理机制。需要额外指出的是......
  • 如何实现逐帧动画(OpenHarmony)
    如何实现逐帧动画场景说明逐帧动画是常见的一种动画呈现形式,本例就为大家介绍如何通过OpenHarmony实现逐帧动画。效果呈现本例最终效果如下:点击“run”按钮,火柴人开始走动。点击“stop”按钮,火柴人停止走动。运行环境IDE:DevEcoStudio3.1Beta1SDK:Ohos_sdk_public3.2.11.9(APIV......
  • 聚能量赢未来,OpenHarmony开发者大会开发工具分论坛圆满落幕
    4月19日,以“开源正当时,共赢新未来”为主题的开放原子开源基金会OpenHarmony开发者大会2023(以下简称“大会”)在北京举行,“开发工具分论坛”于当天下午召开。在本次论坛上,各位演讲嘉宾重点分享了OpenAtomOpenHarmony(以下简称“OpenHarmony”)3.2Release版本的开发工具、开发方法......
  • 具有前后按钮切换+头图切换联动通用接口(应付不同的联动需要)的图片滑动效果
    -------scroll_tween.js----------CHRD.scroll.tween=function(){varTween={Quart:{easeOut:function(t,b,c,d){return-c*((t=t/d-1)*t*t*t-1)+b;}},Back:{easeOut......
  • OpenHarmony 资源调度之智能感知调度源码分析
    作者:张守忠1WPA_supplicant简介WPA是WiFiProtectedAccess的缩写,中文含义为“WiFi网络安全存取”。WPA是一种基于标准的可互操作的WLAN安全性增强解决方案,可大大增强现有以及未来无线局域网络的数据保护和访问控制水平。wpa_supplicant是开源项目源码,支持Linux,Windows以及很多......
  • 让任意view具有滑动效果的SlideUp
    基本的类,只有一个:importandroid.animation.Animator;importandroid.animation.ValueAnimator;importandroid.view.MotionEvent;importandroid.view.View;importandroid.view.ViewTreeObserver;importandroid.view.animation.DecelerateInterpolator;publicclassSl......
  • 记录-因为写不出拖拽移动效果,我恶补了一下Dom中的各种距离
    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助背景最近在项目中要实现一个拖拽头像的移动效果,一直对JSDom拖拽这一块不太熟悉,甚至在网上找一个示例,都看得云里雾里的,发现遇到最大的拦路虎就是JSDom各种各样的距离,让人头晕眼花,看到一个距离属性,大脑中的印象极......
  • 新翻页
    <divid="pagination"><%--显示翻页链接--%><%--&les=<%=les%>&pu=<%=pu%>&stu=<%=stu%>--%><ahref="?page=<%=Math.max(1,currentPage-1)%>&les=<%=les%>&pu=&......
  • 开启新时代,承接新使命,开放原子开源大赛OpenHarmony创新赛正式启航!
     开放原子开源大赛OpenHarmony创新赛,正式启动啦!“OpenHarmony创新赛”是开放原子全球开源大赛下开设的创新赛道,面向企业、个人、高校师生等广大开发者,聚焦OpenHarmony应用开发技术,打造竞技交流赛事,旨在提高开发者的动手实践能力以及运用新技术的创新创造能力。通过本次大赛......
  • OpenHarmony的线程间通信EventHandler
    一、初识EventHandler​ 在OpenHarmony的开发过程中,如果遇到处理下载、运算等较为耗时的操作时,会阻塞当前线程,但是实际操作中又不希望当前线程受到阻塞。比如:我们的app在界面上有一个下载文件的处理按钮,如果在按钮按下时,直接处理下载任务时,当前的界面就会阻塞,不允许操作界面上的任......