鸿蒙OS 短视频轮播结果及会话控制

种地  论坛元老 | 2025-2-14 14:57:55 | 来自手机 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 1048|帖子 1048|积分 3144

代码拜见
gitee: https://gitee.com/jimmy_enjoy/hamonyos_video_demo/tree/master
结果


     鸿蒙模拟器-视频轮播结果
  
数据定义

  1. export interface  VideoItem{
  2.   name: string
  3.   url?: string // 网址播放
  4.   video?: string // rawfile 播放
  5.   head: Resource
  6.   index: number
  7. }
复制代码
组件

包含Swiper的父组件

  1. Swiper(this.swiperController) {
  2.           ForEach(videoDataList, (item: VideoItem, index: number) => {
  3.             VideoPlayer({
  4.               curSource: item,
  5.               isPageShow: this.isPageShow,
  6.               curTitle: item.name,
  7.               curIndex: this.curVideoIndex,
  8.               index: index,
  9.               total: this.total,
  10.               onShowPrevious: () => this.showPrevious(index),
  11.               onShowNext: () => this.showNext(index)
  12.             })
  13.           })
  14.         }
  15.         .cachedCount(2)
  16.         // .disableSwipe(true)
  17.         .vertical(true)
  18.         .indicator(false)
  19.         .loop(false)
  20.         .curve(Curve.Ease)
  21.         .duration(300)
  22.         .onAnimationStart((index: number, targetIndex: number, extraInfo: SwiperAnimationEvent) => {
  23.           Logger.info(TAG,
  24.             `onAnimationStart index: ${index},curIndex: ${targetIndex},extraInfo: ${JSON.stringify(extraInfo)}.`);
  25.           this.curVideoIndex = targetIndex;
  26.           // this.curTitle = this.data.getData(this.curVideoIndex)?.name ?? ''
  27.         })
  28.         .width('100%')
  29.         .height('100%')
复制代码
SwiperItem 子组件

  1. Stack({alignContent: Alignment.BottomStart}){
  2.       Column(){
  3.         XComponent({
  4.           id: 'XComponent',
  5.           type: XComponentType.SURFACE,
  6.           controller: this.xComponentController
  7.         })
  8.           .onLoad(async () => {
  9.             // this.xComponentController.setXComponentSurfaceRect({
  10.             //   surfaceWidth: CommonConstants.SURFACE_WIDTH,
  11.             //   surfaceHeight: CommonConstants.SURFACE_HEIGHT
  12.             // })
  13.             this.surfaceID = this.xComponentController.getXComponentSurfaceId()
  14.             hilog.info(0x0000, TAG,
  15.               'surfaceID=' + this.surfaceID + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  16.             this.initAVPlayer()
  17.           })
  18.           .aspectRatio(CommonConstants.ASPECT)
  19.           .width('100%')
  20.           .height(240)
  21.       }
  22.       .justifyContent(FlexAlign.Center)
  23.       .width(CommonConstants.WIDTH_FULL_PERCENT)
  24.       .height(CommonConstants.HEIGHT_FULL_PERCENT)
  25.       .zIndex(CommonConstants.Z_INDEX_BASE)
  26.       // 播放控制,这里需要设置更高的Z-Index,防止事件被最上层的手势监听覆盖
  27.       Column(){
  28.         this.PlayControl()
  29.       }
  30.       .zIndex(CommonConstants.Z_INDEX_MAX)
  31.     }
  32.     .gesture(
  33.     // 监听滑动手势
  34.       PanGesture({direction: PanDirection.Horizontal})
  35.         .onActionStart((event: GestureEvent) => {
  36.           this.isSliderGesture = true
  37.           this.panStartX = event.offsetX
  38.           this.panStartTime = this.currentTime
  39.           this.sliderOnchange(this.panStartTime, SliderChangeMode.Begin)
  40.         })
  41.         .onActionUpdate((event: GestureEvent) => {
  42.           this.isSliderGesture = true
  43.           let panTime = this.panStartTime +
  44.             (this.panStartX + event.offsetX) / this.slideWidth * this.durationTime
  45.           this.panEndTime = Math.min(Math.max(0, panTime), this.durationTime)
  46.           this.sliderOnchange(this.panEndTime, SliderChangeMode.Moving)
  47.         })
  48.         .onActionEnd(() => {
  49.           this.sliderOnchange(this.panEndTime, SliderChangeMode.End)
  50.           this.isSliderGesture = false
  51.         })
  52.     )
  53.   @Builder
  54.   PlayControl(){
  55.     Column(){
  56.     // 滑动时显示,当前时长/总时长
  57.       Row(){
  58.         Text(this.currentStringTime)
  59.           .fontSize($r('app.float.font_size_14'))
  60.           .fontColor(Color.White)
  61.           .opacity($r('app.float.opacity_9'))
  62.           .margin({ left: CommonConstants.TEXT_MARGIN_LEFT })
  63.           .width(CommonConstants.TEXT_LEFT_WIDTH)
  64.           .textAlign(TextAlign.End)
  65.           .zIndex(CommonConstants.SLIDER_INDEX)
  66.         Divider()
  67.           .vertical(true)
  68.           .height($r('app.float.padding_14'))
  69.           .width(CommonConstants.DIVIDER_WIDTH)
  70.           .backgroundBlurStyle(BlurStyle.Regular, { colorMode: ThemeColorMode.LIGHT })
  71.           .color(Color.White)
  72.           .opacity($r('app.float.opacity_9'))
  73.           .margin({ left: $r('app.float.margin_small'), right: $r('app.float.margin_small') })
  74.           .rotate({
  75.             x: CommonConstants.DIVIDER_X,
  76.             y: CommonConstants.DIVIDER_Y,
  77.             z: CommonConstants.DIVIDER_Z,
  78.             centerX: CommonConstants.DIVIDER_CENTER_X,
  79.             centerY: CommonConstants.DIVIDER_CENTER_Y,
  80.             angle: CommonConstants.DIVIDER_ANGLE
  81.           })
  82.         Text(this.durationStringTime)
  83.           .fontSize($r('app.float.font_size_14'))
  84.           .fontColor(Color.White)
  85.           .margin({ right: CommonConstants.TEXT_MARGIN_LEFT })
  86.           .width(CommonConstants.TEXT_LEFT_WIDTH)
  87.           .textAlign(TextAlign.Start)
  88.           .opacity($r('app.float.opacity_4'))
  89.           .zIndex(CommonConstants.SLIDER_INDEX)
  90.       }
  91.       .margin({ bottom: $r('app.float.margin_small') })
  92.       .alignItems(VerticalAlign.Center)
  93.       .opacity(this.isTimeDisplay)
  94.       Column() {
  95.         Row() {
  96.           Image(this.isPlaying ? $r('sys.media.ohos_ic_public_pause') :
  97.           $r('app.media.ic_video_menu_play'))
  98.             .width($r('app.float.size_24'))
  99.             .height($r('app.float.size_24'))
  100.             .fillColor(Color.White)
  101.             .margin({ right: $r('app.float.margin_small') })
  102.             .onClick(() => {
  103.               this.iconOnclick();
  104.             })
  105.             .visibility(
  106.               // this.isFloatWindow || this.isFullLandscapeScreen || this.isFullScreen ||
  107.             this.isSliderDragging ? Visibility.None : Visibility.Visible)
  108.           Slider({
  109.             value: this.isSliderGesture ? this.panEndTime : this.currentTime,
  110.             step: CommonConstants.SLIDER_STEP,
  111.             min: CommonConstants.SLIDER_MIN,
  112.             max: this.durationTime,
  113.             style: this.sliderStyle
  114.           })
  115.             .id('video_slider')
  116.             .height(this.isSliderDragging ? $r('app.float.side_width') : $r('app.float.size_24'))
  117.             .trackColor($r('app.color.white_opacity_1_color'))
  118.             .showSteps(false)
  119.             .blockSize({ width: this.blockSize, height: this.blockSize })
  120.             .blockColor($r('sys.color.background_primary'))
  121.             .layoutWeight(1)
  122.             .trackThickness(this.trackThicknessSize)
  123.             .trackBorderRadius(CommonConstants.TRACK_BORDER_RADIUS)
  124.             .selectedBorderRadius(CommonConstants.TRACK_BORDER_RADIUS)
  125.             .zIndex(CommonConstants.SLIDER_INDEX)
  126.             .onAreaChange(() => {
  127.               let videoSlider: componentUtils.ComponentInfo = componentUtils.getRectangleById('video_slider');
  128.               this.slideWidth = px2vp(videoSlider.size.width);
  129.               // this.offsetY = px2vp(videoSlider.localOffset.y);
  130.               // this.beginX = px2vp(videoSlider.localOffset.x);
  131.             })
  132.             .onChange((value: number, mode: SliderChangeMode) => {
  133.               this.sliderOnchange(value, mode);
  134.             })
  135.         }
  136.         .width(CommonConstants.WIDTH_FULL_PERCENT)
  137.       }
  138.     }
  139.   }
复制代码
AVPlayer

创建AVPlayer

  1.   async initAVPlayer(){
  2.     hilog.info(0x0000, TAG, 'createAVPlayer begin')
  3.     media.createAVPlayer().then((video: media.AVPlayer) => {
  4.       if(video !== null){
  5.         Logger.info(TAG, 'initAVPlayer video')
  6.         this.avPlayer = video
  7.         this.avPlayerStart(this.curSource, this.avPlayer)
  8.         this.setAVPlayerCallback(this.avPlayer)
  9.       }
  10.     }).catch((error: BusinessError) => {
  11.       Logger.error(TAG, `AVPlayer catchCallback, error message:${error.message}`);
  12.     })
  13.   }
复制代码
注册AVPlayer回调函数

  1.   setAVPlayerCallback(avPlayer: media.AVPlayer){
  2.           // 更新视频当前时间状态
  3.     avPlayer.on('timeUpdate', (time: number) => {
  4.       if(time > this.currentTime * 1000){
  5.         animateTo({duration: 1000, curve: Curve.Linear}, () => {
  6.           this.currentTime = Math.floor(time / 1000)
  7.         })
  8.       } else {
  9.         this.currentTime = Math.floor(time / 1000)
  10.       }
  11.       this.currentStringTime = secondToTime(Math.floor(time / CommonConstants.SECOND_TO_MS))
  12.     })
  13.         // 错误监听
  14.     avPlayer.on('error', (err: BusinessError) => {
  15.       hilog.error(0x0000, TAG, `Invoke avPlayer failed, code is ${err.code}, message is ${err.message}` +
  16.         `----state:${avPlayer.state} this.curIndex:${this.curIndex} this.index: ${this.index}`);
  17.       avPlayer.reset();
  18.     })
  19.     this.setAVPlayerStateListen(avPlayer)
  20.   }
  21.   setAVPlayerStateListen(avPlayer: media.AVPlayer){
  22.           // 监听状态变化
  23.     avPlayer.on('stateChange', async (state: string) => {
  24.       Logger.info(TAG, 'avplayer stateChange')
  25.       switch (state){
  26.         case 'idle': // call reset into idle state
  27.           hilog.info(0x0000, TAG, 'AVPlayer state idle called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`)
  28.           break
  29.         case 'initialized':
  30.           hilog.info(0x0000, TAG,
  31.             'AVPlayer state initialized called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  32.           avPlayer.surfaceId = this.surfaceID;
  33.           avPlayer.prepare();
  34.           break;
  35.         case 'prepared':
  36.           avPlayer.audioInterruptMode = audio.InterruptMode.INDEPENDENT_MODE;
  37.           hilog.info(0x0000, TAG,
  38.             'AVPlayer state prepared called.' +
  39.               ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  40.           this.flag = true;
  41.           avPlayer.loop = true;
  42.           this.duration = avPlayer.duration;
  43.           this.durationTime = Math.floor(this.duration / CommonConstants.SECOND_TO_MS);
  44.           this.durationStringTime = secondToTime(this.durationTime);
  45.           if (this.curIndex === this.index) {
  46.             this.playVideo()
  47.             // this.firstFlag = false;
  48.           }
  49.           break;
  50.         case 'completed':
  51.           hilog.info(0x0000, TAG,
  52.             'AVPlayer state completed called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  53.           this.isPlaying = false;
  54.           break;
  55.         case 'playing':
  56.           this.isPlaying = true;
  57.           hilog.info(0x0000, TAG,
  58.             'AVPlayer state playing called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}` +
  59.               ', source=' + this.curSource);
  60.           break;
  61.         case 'paused':
  62.           this.isPlaying = false
  63.           Logger.info(TAG,
  64.             'AVPlayer state paused called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  65.           break;
  66.         case 'stopped':
  67.           Logger.info(TAG,
  68.             'AVPlayer state stopped called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  69.           break;
  70.         case 'released':
  71.           Logger.info(TAG,
  72.             'AVPlayer state released called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  73.           break;
  74.         case 'error':
  75.           Logger.info(TAG, 'AVPlayer state error called' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  76.           // avPlayer.reset();
  77.           break;
  78.         default:
  79.           Logger.info(TAG,
  80.             'AVPlayer state unknown called.' + ` this.curIndex:${this.curIndex} this.index:${this.index}`);
  81.           break;
  82.       }
  83.     })
  84.   }
复制代码
AVSession会话控制



  • 需要申请长时使命
  • 在通知栏可以由系统视频控制器举行视频的控制,包括进度条滑动、快进、快退、暂停、下一首、上一首。
后台长时使命申请

需要在module.json中注册
  1.       {
  2.         "name": "ohos.permission.KEEP_BACKGROUND_RUNNING",
  3.         "reason": "$string:reason_background",
  4.         "usedScene": {
  5.           "abilities": [
  6.             "EntryAbility"
  7.           ],
  8.           "when": "always"
  9.         }
  10.       }
复制代码
  1. export class BackgroundTaskManager{
  2.   public static startContinuousTask(context?: common.UIAbilityContext): void{
  3.     if(!context) return
  4.     let wantAgentInfo: wantAgent.WantAgentInfo = {
  5.       wants: [
  6.         {
  7.           bundleName: context.abilityInfo.bundleName,
  8.           abilityName: context.abilityInfo.name
  9.         }
  10.       ],
  11.       actionType: wantAgent.OperationType.START_ABILITY,
  12.       requestCode: 0,
  13.       actionFlags: [wantAgent.WantAgentFlags.UPDATE_PRESENT_FLAG]
  14.     }
  15.     wantAgent.getWantAgent(wantAgentInfo).then((wantAgentObj) => {
  16.       try {
  17.       // 启动后台任务
  18.         backgroundTaskManager.startBackgroundRunning(context, backgroundTaskManager.BackgroundMode.AUDIO_PLAYBACK,
  19.           wantAgentObj).then(() => {
  20.           Logger.info(TAG, 'startBackgroundRunning succeeded');
  21.         }).catch((err: BusinessError) => {
  22.           Logger.error(TAG, `startBackgroundRunning failed Cause:  ${JSON.stringify(err)}`);
  23.         })
  24.       }catch (error){
  25.         Logger.error(TAG, `Operation startBackgroundRunning failed. code is ${error.code} message is ${error.message}`);
  26.       }
  27.     })
  28.   }
  29.   public static stopContinuousTask(context?: common.UIAbilityContext){
  30.     try {
  31.       if (!context) {
  32.         return
  33.       }
  34.       backgroundTaskManager.stopBackgroundRunning(context).then(() => {
  35.         Logger.info(TAG, 'stopBackgroundRunning succeeded');
  36.       }).catch((err: BusinessError) => {
  37.         Logger.error(TAG, `stopBackgroundRunning failed Cause:  ${JSON.stringify(err)}`);
  38.       });
  39.     } catch (error) {
  40.       Logger.error(TAG, `stopBackgroundRunning failed. code is ${error.code} message is ${error.message}`);
  41.     }
  42.   }
  43. }
复制代码
AVSession

AVSession 在整个应用中,全局共享实例。
创建

  1. export class AvSessionController{
  2.   private static instance: AvSessionController | null
  3.   private context: common.UIAbilityContext | undefined = undefined;
  4.   private avSession: avSession.AVSession | undefined = undefined;
  5.   private avSessionMetadata: avSession.AVMetadata | undefined = undefined;
  6.   private constructor() {
  7.     this.initAvSession()
  8.   }
  9.   public getAvSession() {
  10.     return this.avSession;
  11.   }
  12.   public getAvSessionMetadata() {
  13.     return this.avSessionMetadata;
  14.   }
  15.   public static getInstance(): AvSessionController{
  16.     if(!AvSessionController.instance){
  17.       AvSessionController.instance = new AvSessionController()
  18.     }
  19.     return AvSessionController.instance
  20.   }
  21. // 初始化
  22.   public initAvSession(){
  23.     this.context = AppStorage.get('context')
  24.     if(!this.context){
  25.       Logger.info(TAG, `session create failed : conext is undefined`)
  26.       return
  27.     }
  28.     avSession.createAVSession(this.context, "SHORT_AUDIO_SESSION", 'video').then(async (avSession) => {
  29.       this.avSession = avSession
  30.       Logger.info(TAG, `session create successed : sessionId : ${this.avSession.sessionId}`);
  31.       // 启动后台长任务
  32.       BackgroundTaskManager.startContinuousTask(this.context);
  33.       // 设置点击后启动的activity
  34.       this.setLaunchAbility();
  35.       this.avSession.activate();
  36.     })
  37.   }
  38.   
  39.   private setLaunchAbility(){
  40.     if(!this.context) return
  41.     const wantAgentInfo: wantAgent.WantAgentInfo = {
  42.       wants: [
  43.         {
  44.           bundleName: this.context.abilityInfo.bundleName,
  45.           abilityName: this.context.abilityInfo.name
  46.         }
  47.       ],
  48.       actionType: wantAgent.OperationType.START_ABILITIES,
  49.       requestCode: 0,
  50.       actionFlags: [wantAgent.WantAgentFlags.UPDATE_PRESENT_FLAG]
  51.     }
  52.     wantAgent.getWantAgent(wantAgentInfo).then((agent) => {
  53.       if(this.avSession) this.avSession.setLaunchAbility(agent)
  54.     })
  55.   }
复制代码
在AVPlayer中结合AVSession

初始化

  1.   async initAVPlayer(){
  2.     hilog.info(0x0000, TAG, 'createAVPlayer begin')
  3.     media.createAVPlayer().then((video: media.AVPlayer) => {
  4.       if(video !== null){
  5.         // ...
  6.         if(this.curIndex === this.curSource.index){
  7.                 // 设置媒体显示信息
  8.           this.avSessionController.setAVMetadata(this.curSource, 0)
  9.         }
  10.         this.setAvSessionListener()
  11.       }
  12.     }).catch((error: BusinessError) => {
  13.       Logger.error(TAG, `AVPlayer catchCallback, error message:${error.message}`);
  14.     })
  15.   }
  16.         // 设置会话控制的监听,进行相应媒体播放器调整
  17.   public async setAvSessionListener(){
  18.     if(!this.avSessionController) return
  19.     this.avSessionController.getAvSession()?.on('play', () => this.sessionPlayCallback())
  20.     this.avSessionController.getAvSession()?.on('pause', () => this.sessionPauseCallback())
  21.     this.avSessionController.getAvSession()?.on('stop', () => this.sessionStopCallback())
  22.     this.avSessionController.getAvSession()?.on('fastForward',
  23.       (time?: number) => this.sessionFastForwardCallback(time))
  24.     this.avSessionController.getAvSession()?.on('rewind', (time?: number) =>                 this.sessionRewindCallback(time))
  25.     this.avSessionController.getAvSession()?.on('seek', (seekTime: number) => this.sessionSeekCallback(seekTime))
  26.     if(this.index < this.total - 1){
  27.       this.avSessionController.getAvSession()?.on('playNext', () => this.sessionPlayNextCallback())
  28.     }else {
  29.       this.avSessionController.getAvSession()?.off('playNext')
  30.     }
  31.     if(this.index > 0){
  32.       this.avSessionController.getAvSession()?.on('playPrevious', () => this.sessionPlayPreviousCallback())
  33.     }else {
  34.       this.avSessionController.getAvSession()?.off('playPrevious')
  35.     }
  36.   }
复制代码
上一首和下一首实现

通过父容器传递的showNext 和 showPrevious,具体为通过SwiperController实现切换
  1.   showNext(index: number): void{
  2.     if(this.swiperController && index < this.total - 1){
  3.         this.swiperController.changeIndex(index + 1, true)
  4.         getContext(this).eventHub.emit('videoShow', index + 1)
  5.     }
  6.   }
  7.   showPrevious(index: number): void{
  8.     if(this.swiperController && index > 0){
  9.       this.swiperController.changeIndex(index - 1, true)
  10.       getContext(this).eventHub.emit('videoShow', index + 1)
  11.     }
  12.   }
复制代码
重要公共方法实现

  1. // 播放视频
  2.   playVideo(){
  3.     if(this.avPlayer && this.curIndex === this.index){
  4.       if(this.avPlayer.state != AVPlayerState.PREPARED && this.avPlayer.state != AVPlayerState.PAUSED &&
  5.       this.avPlayer.state != AVPlayerState.COMPLETED){
  6.         return
  7.       }
  8.       // 重置会话监听状态
  9.       this.setAvSessionListener()
  10.       this.avSessionController.setAVMetadata(this.curSource, this.duration)
  11.       this.updateIsPlay(true)
  12.       this.avPlayer.play((err: BusinessError) => {
  13.         if(err){
  14.           this.updateIsPlay(false)
  15.           Logger.error(TAG, `playVideo failed, code is ${err.code}, message is ${err.message}`);
  16.         } else {
  17.           Logger.info(TAG, `playVideo success , this.curIndex:${this.curIndex}`);
  18.         }
  19.       })
  20.     }
  21.   }
  22. // 更新播放状态
  23.   updateIsPlay(isPlay: boolean){
  24.     if(this.curIndex !== this.curSource.index) return
  25.     this.isPlaying = isPlay
  26.     // 更新会话控制中,媒体播放状态。
  27.     this.avSessionController.setAvSessionPlayState({
  28.       state: isPlay ? avSession.PlaybackState.PLAYBACK_STATE_PLAY : avSession.PlaybackState.PLAYBACK_STATE_PAUSE,
  29.       position: {
  30.         elapsedTime: this.currentTime * 1000, // 已播放时长
  31.         updateTime: new Date().getTime() // 更新时间
  32.       },
  33.       duration: this.duration
  34.     })
  35.   }
复制代码



  • 完备代码
  • 可以用LazyForEach结合Swiper的cachecount实现懒加载与缓存。
  • 这里没有实现屏幕旋转的环境。
  • 更完备的示例可以参考 鸿蒙官方示例

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

种地

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表