当前位置: 首页 > news >正文

鸿蒙实况窗 (Live View) 实战:仿 iOS“灵动岛”效果,实时显示外卖/打车进度

🌟 前言:为什么它是“用户体验”的神?

传统的通知是“一次性”的:“骑手已接单”-> 划掉 ->“骑手已送达”-> 划掉。
用户想看中间的进度,必须解锁手机 -> 打开 App -> 等待加载 -> 查看地图。

实况窗的逻辑是“伴随”:
任务开始,它出现;任务进行中,它实时更新(剩余 500米 -> 300米 -> 100米);任务结束,它自动消失。

它有两种核心形态:

  1. 胶囊态 (Capsule):位于状态栏左上角,显示极简信息(如:图标 + 时间)。
  2. 卡片态 (Card):在通知中心或锁屏界面,显示详细信息(如:地图、司机电话、车牌)。

🏗️ 一、 架构原理:从后台到前台

实况窗本质上是一种特殊的Form (卡片)。它的生命周期由后台服务驱动。

数据流转图 (Mermaid):

锁屏卡片状态栏胶囊实况窗管理器 (System)你的 App (后台 Service)锁屏卡片状态栏胶囊实况窗管理器 (System)你的 App (后台 Service)携带初始数据: {status: "接单", time: "5min"}{status: "赶往中", distance: "300m"}loop[任务进行中]1. 创建实况窗 (Create)渲染胶囊 UI渲染卡片 UI2. 更新数据 (Update)刷新刷新3. 结束任务 (End)移除移除

🛠️ 二、 核心实战:编写 ArkTS UI

实况窗的 UI 代码写在EntryFormAbility对应的 Widget 页面中。
关键在于:一套代码,适配两种形态。我们需要判断ohos.extra.param.key.form_dimension或者是自定义的布局类型。

1. 定义数据结构
classRideInfo{status:string;// "已接单", "行程中"plate:string;// 车牌号 "苏A·88888"time:string;// "3 分钟"icon:Resource;}
2. 编写响应式布局 (LiveViewCard.ets)

鸿蒙提供了LiveView专用组件布局结构。

@Entry@Componentstruct LiveViewCard{// 接收外部更新的数据@LocalStorageProp('status')status:string='等待接单';@LocalStorageProp('time')time:string='--';// 自动判断当前是胶囊还是卡片@StorageProp('formDimension')formDimension:number=0;build(){// 根容器RelativeContainer(){// --- 形态 1: 胶囊态 (状态栏左上角) ---// 只有在胶囊模式下显示if(this.isCapsule()){Row(){Image($r('app.media.car_icon')).width(18).height(18).margin({right:4})Text(this.time).fontSize(14).fontColor(Color.White)}.backgroundColor('#007DFF').borderRadius(20).padding({left:8,right:8,top:2,bottom:2})}// --- 形态 2: 卡片态 (锁屏/通知中心) ---// 只有在卡片模式下显示if(this.isCard()){Column(){// 标题栏Row(){Text('正在前往').fontSize(16).fontWeight(FontWeight.Bold)Blank()Text(this.time).fontSize(20).fontColor('#007DFF')}.width('100%')// 详细信息Row(){Text(`车牌: 苏A·88888`).fontSize(14).fontColor('#666')Text(this.status).fontSize(14).fontColor('#333')}.width('100%').margin({top:10})// 进度条Progress({value:60,total:100,type:ProgressType.Linear}).color('#007DFF').width('100%').margin({top:15})}.padding(15).backgroundColor(Color.White).borderRadius(16)}}}isCapsule():boolean{// 这里的判断逻辑需参考官方文档的 dimension 枚举returnthis.formDimension===2;// 假设 2 代表胶囊}isCard():boolean{returnthis.formDimension===3;// 假设 3 代表卡片}}

📡 三、 逻辑驱动:后台更新数据

UI 写好了,怎么让它动起来?
我们需要在后台服务(ServiceExtensionAbility)或者主进程中调用notificationManagerformProvider

注意:实况窗通常归类为Notification (通知)的一种特殊类型 (LiveView)。

importnotificationManagerfrom'@ohos.notificationManager';// 1. 创建实况窗 (Publish)asyncfunctionstartLiveView(){letnotificationRequest={id:1001,content:{contentType:notificationManager.ContentType.NOTIFICATION_CONTENT_LIVE_VIEW,liveView:{status:notificationManager.LiveViewStatus.LIVE_VIEW_STATUS_UPDATE,// 这里关联你的 UI Ability 和初始数据extraInfo:{"status":"司机已出发","time":"5 分钟"}}},// 设为常驻,用户无法划掉,直到任务结束isOngoing:true,isUnremovable:true};awaitnotificationManager.publish(notificationRequest);}// 2. 实时更新 (Update)// 比如每隔 10秒 调用一次asyncfunctionupdateProgress(timeLeft:string){letrequest={id:1001,// ID 必须一致content:{contentType:notificationManager.ContentType.NOTIFICATION_CONTENT_LIVE_VIEW,liveView:{status:notificationManager.LiveViewStatus.LIVE_VIEW_STATUS_UPDATE,extraInfo:{"status":"即将到达","time":timeLeft}}}};awaitnotificationManager.publish(request);}// 3. 结束实况窗 (End)asyncfunctionendLiveView(){// 发送结束状态,系统会自动做收尾动画// 或者直接 cancelawaitnotificationManager.cancel(1001);}

🎨 四、 交互细节:灵动的核心

要让实况窗看起来“高级”,必须处理好Transitions (状态流转)

  1. 颜色语义
  • 蓝色:进行中(如:司机赶往中)。
  • 绿色:已完成/到达(如:司机已到达)。
  • 红色:异常/警告(如:订单超时)。
  1. 点击跳转
    实况窗被点击时,应该直接跳转到 App 的详情页(如地图页),而不是首页。
    notificationRequest中配置wantAgent实现跳转。
  2. 动画过渡
    鸿蒙系统自动处理了从“胶囊”展开为“卡片”的动画,但你需要确保两个布局的数据是对应的,这样视觉上才不会突兀。

🎯 总结

实况窗是鸿蒙Next版本中最具辨识度的特性之一。
它打破了 App 的边界,将服务延伸到了 System UI 中。

对于打车、外卖、航班、比赛比分这类“长时效、强关注”的业务场景,实况窗不是“锦上添花”,而是“必选项”

Next Step:
在模拟器中运行上面的代码,点击“开始接单”按钮。观察状态栏左上角是否出现了一个蓝色的小图标。尝试下拉通知栏,看它是否平滑地展开成了一张详细的卡片。

http://www.jsqmd.com/news/225465/

相关文章:

  • AI智能实体侦测服务颜色标注逻辑揭秘:三色高亮原理详解
  • 大模型学习宝典:从Transformer到高效微调的系统化知识手册
  • AI智能实体侦测服务国际化准备:中英文混合文本识别挑战
  • HY-MT1.5-1.8B嵌入式设备部署:Jetson平台适配实战记录
  • Hunyuan HY-MT1.5部署教程:3步完成GPU算力适配,支持33语种互译
  • HY-MT1.5如何支持方言翻译?上下文学习部署实战指南
  • CA6140溜板箱加工工艺及工装设计
  • 全网最全8个AI论文工具,专科生轻松搞定毕业论文!
  • AI智能实体侦测服务显存不足?CPU适配优化部署教程来解决
  • 大模型智能体vs工作流:彻底理解Agent的运行时机制与工作流的设计时逻辑
  • DeepSeek-NER vs RaNER实战对比:信息抽取速度与精度全面评测
  • 法律文书信息提取实战:AI智能实体侦测服务精准识别当事人信息
  • HY-MT1.5-1.8B模型剪枝实验:进一步压缩体积可行性分析
  • 元宵节公众号互动怎么玩?基于 SVG 的 8 种交互方案拆解
  • Hunyuan-HY-MT1.5实战案例:企业多语种客服系统搭建详细步骤
  • AI出海必备趋势分析:HY-MT1.5开源翻译模型多场景落地实战
  • 混元模型1.5技术解析:解释性翻译优化原理
  • 腾讯HY-MT1.5翻译模型:高可用架构设计方案
  • 全球大模型第一股智谱华章上市,GLM-4.7登顶双榜,中国AGI迎来资本时代!
  • Qwen3-VL电商实战:商品描述生成,ROI提升200%
  • 开源翻译模型新标杆:HY-MT1.5-7B混合语言优化部署指南
  • HY-MT1.5-1.8B量化部署:边缘计算场景最佳实践
  • 215挖掘机结构设计
  • HY-MT1.5性能测试:不同batch size效率对比
  • 从小白到大神:大模型热门岗位全面解析与系统学习方法_程序员如何转行大模型?五大热门岗位推荐
  • Hunyuan HY-MT1.5省钱部署:免费镜像+按需GPU计费方案
  • HY-MT1.5-1.8B车载系统集成:驾驶场景语音翻译部署案例
  • 收藏!2026大模型浪潮下,程序员的必争赛道与转型指南
  • Qwen3-VL在线体验指南:不用下载,浏览器直接玩
  • 腾讯HY-MT1.5与传统翻译软件对比分析