文章目录
- 前言
- 完整代码结构预览
- 第一部分:数据接收与状态初始化
- 第二部分:阶梯式入场动画编排 (startEnterAnimation)
- 第三部分:动态 UI 渲染与样式绑定
- 第四部分:平滑退场动画 (startExitAnimation)
- ️ 第五部分:自定义信息行组件 (@Builder)
- 完整代码
- 总结与实战建议
前言
在上一期的实战中,我们掌握了全场景自适应的响应式栅格布局。今天,我们将迎来 ArkUI 动画系统的“高光时刻”——详情页交错入场与平滑退场动画。
在现代高端 App 的交互设计中,页面切换早已不再是生硬的“闪现”。通过精心编排的动画,让图片、标题、正文等元素按照特定的时间顺序依次浮现,能够极大地提升应用的精致感和沉浸感。
这个实战案例虽然代码精炼,但完美诠释了 ArkTS 动画系统的精髓,涵盖了以下核心知识点:
- 生命周期与路由传参:在
aboutToAppear中接收列表页传递的数据。 - 交错动画编排:利用
setTimeout配合animateTo,实现多元素的阶梯式入场。 - 状态驱动属性动画:通过绑定
scale(缩放)和opacity(透明度),实现丝滑的视觉过渡。 - 双向动画闭环:不仅实现了进入时的优雅展开,还完美复刻了退出时的平滑收缩。
下面,我们就对这段实现电影级质感的详情页代码进行一次深度解析。
完整代码结构预览
首先,让我们从整体上把握代码结构。它定义了一个Detail入口组件,核心是接收参数、编排入场动画、渲染详情页 UI 以及处理退出动画。
importrouterfrom'@ohos.router'interfaceCardData{...}@Entry@Componentstruct Detail{// 1. 数据与动画状态定义@StatecardData:CardData={...}@StateimageScale:number=0.8;@StateimageOpacity:number=0@StatetitleScale:number=0.8;@StatetitleOpacity:number=0@StatecontentOpacity:number=0// 2. 生命周期与动画逻辑aboutToAppear(){...}privatestartEnterAnimation():void{...}privatestartExitAnimation():void{...}// 3. 页面主体与自定义构建build(){...}@BuilderInfoRow(label:string,value:string){...}}第一部分:数据接收与状态初始化
详情页的动画效果完全由几个@State变量驱动。在页面加载之初,我们需要先接收列表页传来的数据。
@StatecardData:CardData={id:'',title:'',subtitle:'',color:'#667EEA',image:''}@StateimageScale:number=0.8;@StateimageOpacity:number=0@StatetitleScale:number=0.8;@StatetitleOpacity:number=0@StatecontentOpacity:number=0aboutToAppear(){constparams=router.getParams()asRecord<string,string>if(params.cardData){this.cardData=JSON.parse(params.cardData)setTimeout(()=>{this.startEnterAnimation()},50)}}@State动画变量:我们将图片、标题、正文的缩放和透明度分别定义为独立的状态变量。初始状态下,图片和标题缩小为0.8且完全透明(0),正文也处于透明状态。这为后续的“从无到有”动画做好了铺垫。aboutToAppear生命周期:这是 ArkUI 组件即将出现时触发的生命周期函数。我们在这里通过router.getParams()获取列表页传递过来的卡片数据(通过JSON.parse反序列化)。- 延迟触发动画:在获取数据后,我们使用了
setTimeout(..., 50)延迟 50 毫秒再执行startEnterAnimation()。这是一个非常实用的技巧,它能确保页面 UI 已经完成了初次渲染,再开始执行动画,避免动画在页面加载瞬间被“吞掉”。
第二部分:阶梯式入场动画编排 (startEnterAnimation)
这是整个详情页交互的灵魂。为了让用户的视觉焦点能够自然地从图片过渡到文字,我们采用了“阶梯式”的动画编排。
privatestartEnterAnimation():void{// 1. 图片率先浮现animateTo({duration:400,curve:Curve.Friction},()=>{this.imageScale=1;this.imageOpacity=1;})// 2. 100ms 后,标题开始浮现setTimeout(()=>{animateTo({duration:300,curve:Curve.Friction},()=>{this.titleScale=1;this.titleOpacity=1;})},100)// 3. 250ms 后,正文内容开始浮现setTimeout(()=>{animateTo({duration:300,curve:Curve.Friction},()=>{this.contentOpacity=1;})},250)}animateTo显式动画:这是 ArkTS 中实现属性动画的核心 API。我们将状态变量的变化包裹在它的回调中,系统会自动补间生成平滑的动画。Curve.Friction摩擦曲线:我们选用了摩擦曲线,这种曲线自带自然的物理减速效果,比线性的动画看起来更加高级和舒适。setTimeout制造时间差:- 图片作为视觉重心,最先在 400ms 内从 0.8 倍放大至 1 倍并显现。
- 标题延迟 100ms 启动,与图片形成微小的错落感。
- 正文内容延迟 250ms 启动,最后缓缓浮现。这种层层递进的效果,极大地缓解了用户等待页面加载的枯燥感。
第三部分:动态 UI 渲染与样式绑定
在build函数中,我们将动画状态变量精确地绑定到了各个 UI 组件的属性上。
// 图片区域Image(this.cardData.image).width('100%').height(400).objectFit(ImageFit.Cover).scale({x:this.imageScale,y:this.imageScale}).opacity(this.imageOpacity)// 标题区域Text(this.cardData.title).fontSize(40).fontWeight(FontWeight.Bold).fontColor('#FFFFFF').scale({x:this.titleScale,y:this.titleScale}).opacity(this.titleOpacity)// 正文与按钮区域Text(this.cardData.subtitle).opacity(0.9*this.contentOpacity)Button('立即体验').scale({x:this.contentOpacity>0?1:0.8,y:...}).opacity(this.contentOpacity)- 动态缩放与透明度:
Image和Text直接绑定了各自的scale和opacity状态。 - 整体淡入效果:副标题、信息行(InfoRow)和底部按钮共享
contentOpacity状态。 - 按钮的微动画:底部的“立即体验”按钮不仅跟随
contentOpacity改变透明度,还通过三元运算符this.contentOpacity > 0 ? 1 : 0.8绑定了缩放。这意味着在正文淡入之前,按钮会保持在 0.8 倍的隐藏状态,随正文一起平滑放大至 1 倍。
第四部分:平滑退场动画 (startExitAnimation)
一个优秀的交互体验不仅要有华丽的入场,还要有得体的退场。当用户点击左上角的返回按钮时,我们需要执行与入场相反的动画。
Button(){Text('←')...}.onClick(()=>{this.startExitAnimation()})privatestartExitAnimation():void{animateTo({duration:300,curve:Curve.Friction,onFinish:()=>{router.back()}// 动画结束后再执行路由返回},()=>{// 所有元素同时缩小并淡出this.imageScale=0.8;this.imageOpacity=0;this.titleScale=0.8;this.titleOpacity=0;this.contentOpacity=0;})}onFinish回调的妙用:在退出动画中,我们绝对不能在点击按钮时直接调用router.back(),否则页面会瞬间销毁,动画根本来不及播放。正确的做法是将router.back()放在animateTo的onFinish回调中,确保 300ms 的收缩淡出动画完整播放完毕后,再销毁当前页面。- 同步收缩:与入场时的“阶梯式”不同,退场时我们将所有元素的缩放和透明度同时还原到初始值,营造出一种“页面整体收缩回列表”的连贯视觉体验。
️ 第五部分:自定义信息行组件 (@Builder)
为了保持代码的整洁,详情页底部的元数据(日期、分类、阅读量)被封装成了一个独立的@Builder函数。
@BuilderInfoRow(label:string,value:string){Row(){Text(label).fontSize(16).fontColor('#FFFFFF').opacity(0.6)Blank()// 弹性空白,将左右文字推到两端Text(value).fontSize(16).fontColor('#FFFFFF').fontWeight(FontWeight.Medium)}.width('100%').padding({top:8,bottom:8}).borderWidth({bottom:1}).borderColor('rgba(255,255,255,0.1)')}Blank()组件:在Row布局中,Blank()会自动填充左右两个Text之间的剩余空间,轻松实现“左侧标签、右侧数值”的两端对齐效果。- 半透明边框:通过
rgba(255,255,255,0.1)设置极淡的白色底边框,在深色背景下增加了界面的精致层次感。
完整代码
importrouter from'@ohos.router'interfaceCardData{id:string title:string subtitle:string color:string image:string}@Entry@ComponentstructDetail{@StatecardData:CardData={id:'',title:'',subtitle:'',color:'#667EEA',image:''}@StateisLoaded:boolean=false@StateimageScale:number=0.8@StateimageOpacity:number=0@StatetitleScale:number=0.8@StatetitleOpacity:number=0@StatecontentOpacity:number=0aboutToAppear(){constparams=router.getParams()asRecord<string,string>if(params.cardData){this.cardData=JSON.parse(params.cardData)setTimeout(()=>{this.startEnterAnimation()},50)}}privatestartEnterAnimation():void{animateTo({duration:400,curve:Curve.Friction},()=>{this.imageScale=1this.imageOpacity=1})setTimeout(()=>{animateTo({duration:300,curve:Curve.Friction},()=>{this.titleScale=1this.titleOpacity=1})},100)setTimeout(()=>{animateTo({duration:300,curve:Curve.Friction},()=>{this.contentOpacity=1})},250)}build(){Column(){Stack({alignContent:Alignment.TopStart}){Column().width('100%').height('100%').backgroundColor(this.cardData.color)Button(){Text('←').fontSize(28).fontColor('#FFFFFF')}.type(ButtonType.Circle).width(50).height(50).backgroundColor('rgba(255,255,255,0.2)').margin({top:50,left:20}).onClick(()=>{this.startExitAnimation()})Column(){Image(this.cardData.image).width('100%').height(400).objectFit(ImageFit.Cover).scale({x:this.imageScale,y:this.imageScale}).opacity(this.imageOpacity)Column({space:16}){Text(this.cardData.title).fontSize(40).fontWeight(FontWeight.Bold).fontColor('#FFFFFF').scale({x:this.titleScale,y:this.titleScale}).opacity(this.titleOpacity)Text(this.cardData.subtitle).fontSize(20).fontColor('#FFFFFF').opacity(0.9*this.contentOpacity)Text('这是一篇关于'+this.cardData.title+'的详细内容。在这里,您可以深入了解更多精彩信息,探索未知的领域,发现更多有趣的故事和知识。').fontSize(18).fontColor('#FFFFFF').opacity(0.8*this.contentOpacity).textAlign(TextAlign.JUSTIFY).lineHeight(32)Column({space:12}){this.InfoRow('日期','2026年7月5日')this.InfoRow('分类','精选推荐')this.InfoRow('阅读','1.2万次')}.margin({top:20}).opacity(this.contentOpacity)Button(){Text('立即体验').fontSize(18).fontColor(this.cardData.color).fontWeight(FontWeight.Bold)}.width('100%').height(56).backgroundColor('#FFFFFF').borderRadius(28).margin({top:30}).scale({x:this.contentOpacity>0?1:0.8,y:this.contentOpacity>0?1:0.8}).opacity(this.contentOpacity)}.width('100%').padding(30)}}.width('100%').height('100%')}.width('100%').height('100%').backgroundColor(this.cardData.color)}privatestartExitAnimation():void{animateTo({duration:300,curve:Curve.Friction,onFinish:()=>{router.back()}},()=>{this.imageScale=0.8this.imageOpacity=0this.titleScale=0.8this.titleOpacity=0this.contentOpacity=0})}@BuilderInfoRow(label:string,value:string){Row(){Text(label).fontSize(16).fontColor('#FFFFFF').opacity(0.6)Blank()Text(value).fontSize(16).fontColor('#FFFFFF').fontWeight(FontWeight.Medium)}.width('100%').padding({top:8,bottom:8}).borderWidth({bottom:1}).borderColor('rgba(255,255,255,0.1)')}}总结与实战建议
通过这个详情页交错入场动画的实战,我们掌握了以下 ArkTS 高阶动画技能:
- 动画编排思维:学会了如何使用
setTimeout错开多个animateTo的执行时间,创造出富有节奏感的阶梯式动画。 - 生命周期结合动画:掌握了在
aboutToAppear中延迟触发动画的技巧,确保 UI 渲染与动画执行的完美同步。 - 路由与动画的协同:深刻理解了在页面退出时,必须将
router.back()放入onFinish回调,这是实现平滑转场的关键细节。 - 精细化状态绑定:能够根据业务需求,将不同的 UI 元素绑定到独立的或共享的动画状态变量上,实现复杂的组合动画效果。
希望这篇详细的代码解析能帮你彻底掌握鸿蒙 ArkTS 的动画编排技巧!如果你觉得有用,欢迎点赞、收藏,我们下期再见!