chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

電子發(fā)燒友App

硬聲App

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示
創(chuàng)作
電子發(fā)燒友網(wǎng)>電子資料下載>顯示及光電>自定義播放器XLVideoPlayer詳解

自定義播放器XLVideoPlayer詳解

2017-09-26 | rar | 0.3 MB | 次下載 | 1積分

資料介紹

  AVPlayer:可以理解為播放器對象,靈活性好,可以高度化的自定義UI,但它本身不能顯示視頻,顯示需要另一個類AVPlayerLayer來顯示,繼承于CALayer,下面是摘自官方的一段介紹:

  AVPlayer works equally well with local and remote media files.

  You can display the visual content of items played by an instance of AVPlayer in a CoreAnimation layer of class AVPlayerLayer.

  You can observe the status of a player using key-value observing.

  主要是說它支持本地/網(wǎng)絡(luò)媒體播放,需要CoreAnimation下的AVPlayerLayer來顯示視頻,我們可以通過KVO監(jiān)聽player的播放狀態(tài)。

  AVPlayerItem:存有相關(guān)媒體信息的類,一個視頻資源對應(yīng)一個AVPlayerItem對象,當(dāng)你需要循環(huán)播放多個視頻資源時也需創(chuàng)建多個AVPlayerItem對象。建議大家可以多看看官方的英文文檔解釋(題外話)。

  An AVPlayerItem represents the presentation state of an asset that’s played by an AVPlayer object, and lets you observe that state.

  AVAsset:主要用于獲取多媒體信息,可以理解為一個抽象類,不能直接使用,操作針對它的子類AVURLAsset,根據(jù)你視頻的url創(chuàng)建一個包含視頻媒體信息的AVURLAsset對象。

  CMTime:還會用到這個媒體時間相關(guān)的類,如有不明白可以看之前一個帖子的解釋。

  層級關(guān)系:

  基于以上幾個類就能實現(xiàn)視頻的基本功能了,例如暫停、播放,快進(jìn)、后退、顯示播放/緩沖進(jìn)度。然后UI層面,層級很簡單,XLVideoPlayer繼承于UIView,上面我們說到顯示視頻需要AVPlayerLayer,我們將AVPlayerLayer加到view的layer上。

  自定義播放器XLVideoPlayer詳解

  下面貼出主要的代碼,初始化AVPlayer對象

  - (AVPlayerLayer *)playerLayer {

  if (!_playerLayer) {

  _playerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];

  _playerLayer.backgroundColor = kPlayerBackgroundColor;

  _playerLayer.videoGravity = AVLayerVideoGravityResizeAspect;//視頻填充模式

  }

  return _playerLayer;

  }

  - (AVPlayer *)player{

  if (!_player) {

  AVPlayerItem *playerItem = [self getAVPlayItem];

  self.playerItem = playerItem;

  _player = [AVPlayer playerWithPlayerItem:playerItem];

 ?。踫elf addProgressObserver];

 ?。踫elf addObserverToPlayerItem:playerItem];

  }

  return _player;

  }

  //initialize AVPlayerItem

  - (AVPlayerItem *)getAVPlayItem{

  NSAssert(self.videoUrl != nil, @“必須先傳入視頻url?。?!”);

  if ([self.videoUrl rangeOfString:@“http”].location != NSNotFound) {

  AVPlayerItem *playerItem=[AVPlayerItem playerItemWithURL:[NSURL URLWithString:[self.videoUrl stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]]];

  return playerItem;

  }else{

  AVAsset *movieAsset = [[AVURLAsset alloc]initWithURL:[NSURL fileURLWithPath:self.videoUrl] options:nil];

  AVPlayerItem *playerItem = [AVPlayerItem playerItemWithAsset:movieAsset];

  return playerItem;

  }

  }

  同時我們注冊KVO,監(jiān)控視頻播放過程,這可以獲取視頻的播放進(jìn)度。AVPlayer有一個屬性currentItem是AVPlayerItem類型,表示當(dāng)前播放的視頻對象。

  #pragma mark - monitor video playing course

  -(void)addProgressObserver{

  //get current playerItem object

  AVPlayerItem *playerItem = self.player.currentItem;

  __weak typeof(self) weakSelf = self;

  //Set once per second

 ?。踫elf.player addPeriodicTimeObserverForInterval:CMTimeMake(1.0, 1.0) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {

  float current = CMTimeGetSeconds(time);

  float total = CMTimeGetSeconds([playerItem duration]);

  weakSelf.progressLabel.text = [weakSelf timeFormatted:current];

  if (current) {

  // NSLog(@“%f”, current / total);

  weakSelf.slider.value = current / total;

  if (weakSelf.slider.value == 1) { //complete block

  if (weakSelf.completedPlayingBlock) {

  weakSelf.completedPlayingBlock(weakSelf);

  }else { //finish and loop playback

  weakSelf.playOrPauseBtn.selected = NO;

 ?。踳eakSelf showOrHidenBar];

  CMTime currentCMTime = CMTimeMake(0, 1);

 ?。踳eakSelf.player seekToTime:currentCMTime completionHandler:^(BOOL finished) {

  weakSelf.slider.value = 0.0f;

  }];

  }

  }

  }

  }];

  }

  以及監(jiān)聽AVPlayerItem對象的status/loadedTimeRanges屬性變化,status對應(yīng)播放狀態(tài),loadedTimeRanges網(wǎng)絡(luò)緩沖狀態(tài),當(dāng)loadedTimeRanges的改變時,每緩沖一部分?jǐn)?shù)據(jù)就會更新此屬性,可以獲得本次緩沖加載的視頻范圍(包含起始時間、本次網(wǎng)絡(luò)加載時長)

  #pragma mark - PlayerItem (status,loadedTimeRanges)

  -(void)addObserverToPlayerItem:(AVPlayerItem *)playerItem{

  //監(jiān)控狀態(tài)屬性,注意AVPlayer也有一個status屬性,通過監(jiān)控它的status也可以獲得播放狀態(tài)

 ?。踦layerItem addObserver:self forKeyPath:@“status” options:NSKeyValueObservingOptionNew context:nil];

  //network loading progress

 ?。踦layerItem addObserver:self forKeyPath:@“l(fā)oadedTimeRanges” options:NSKeyValueObservingOptionNew context:nil];

  }

  在這獲取視頻的總時長,網(wǎng)絡(luò)的視頻緩沖進(jìn)度,做相應(yīng)的顯示。

  - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context{

  AVPlayerItem *playerItem = object;

  if ([keyPath isEqualToString:@“status”]) {

  AVPlayerStatus status = [[change objectForKey:@“new”] intValue];

  if(status == AVPlayerStatusReadyToPlay){

  self.totalDuration = CMTimeGetSeconds(playerItem.duration);

  self.totalDurationLabel.text = [self timeFormatted:self.totalDuration];

  }

  }else if([keyPath isEqualToString:@“l(fā)oadedTimeRanges”]){

  NSArray *array = playerItem.loadedTimeRanges;

  CMTimeRange timeRange = [array.firstObject CMTimeRangeValue];//本次緩沖時間范圍

  float startSeconds = CMTimeGetSeconds(timeRange.start);

  float durationSeconds = CMTimeGetSeconds(timeRange.duration);

  NSTimeInterval totalBuffer = startSeconds + durationSeconds;//緩沖總長度

  self.slider.middleValue = totalBuffer / CMTimeGetSeconds(playerItem.duration);

  // NSLog(@“totalBuffer:%.2f”,totalBuffer);

  //remove loading animation

  if (self.slider.middleValue 《= self.slider.value) {

  self.activityIndicatorView.center = self.center;

 ?。踫elf addSubview:self.activityIndicatorView];

  [self.activityIndicatorView startAnimating];

  }else {

 ?。踫elf.activityIndicatorView removeFromSuperview];

  }

  }

  }

  下面這部分是定位視頻的某個位置播放,也就是快進(jìn)后退。

  這里需要注意的是在用戶拖拽slider的過程中需要先暫停,否則手動改變進(jìn)度和播放的進(jìn)度會有沖突,用戶拖拽完畢再去播放視頻。

  - (void)finishChange {

  _inOperation = NO;

  [self hiden];

  CMTime currentCMTime = CMTimeMake(self.slider.value * self.totalDuration, 1);

  [self.player seekToTime:currentCMTime completionHandler:^(BOOL finished) {

 ?。踫elf.player play];

  self.playOrPauseBtn.selected = YES;

  }];

  }

下載該資料的人也在下載 下載該資料的人還在閱讀
更多 >

評論

查看更多

下載排行

本周

  1. 1常用電子元器件使用手冊
  2. 2.40 MB   |  4次下載  |  免費(fèi)
  3. 2SR520-SR5100肖特基二極管規(guī)格書
  4. 0.11 MB   |  3次下載  |  免費(fèi)
  5. 3FS5281E 5V同步升壓充電兩串鋰電池充電管理IC技術(shù)手冊
  6. 3.30 MB   |  2次下載  |  免費(fèi)
  7. 4FS8205 20V N 溝道增強(qiáng)型MOS場效應(yīng)管技術(shù)手冊
  8. 1.70 MB   |  2次下載  |  免費(fèi)
  9. 5FS2117 高效 DC-DC 同步整流升壓轉(zhuǎn)換器中文手冊
  10. 2.13 MB   |  1次下載  |  免費(fèi)
  11. 6labview入門手冊
  12. 1.19 MB   |  1次下載  |  5 積分
  13. 7FS6202內(nèi)置濾波電容低功耗線性穩(wěn)壓器LDO技術(shù)資料
  14. 1.72 MB   |  次下載  |  免費(fèi)
  15. 8FSB628升壓電流模式PWM轉(zhuǎn)換器英文手冊
  16. 1.87 MB   |  次下載  |  免費(fèi)

本月

  1. 12025年AI 智能終端和SoC芯片解讀
  2. 15.88 MB  |  82次下載  |  免費(fèi)
  3. 248V電源磚模塊市場分析報告:市場洞察和元器件機(jī)遇
  4. 4.68 MB   |  22次下載  |  免費(fèi)
  5. 3適應(yīng)邊緣AI全新時代的GPU架構(gòu)
  6. 628.91 KB   |  9次下載  |  免費(fèi)
  7. 4電路、電流和電壓介紹
  8. 14.96 MB   |  6次下載  |  免費(fèi)
  9. 5S7-200 可編程序控制器系統(tǒng)手冊
  10. 12.81 MB   |  5次下載  |  免費(fèi)
  11. 6索尼333ESL電路圖資料
  12. 10.11 MB   |  5次下載  |  免費(fèi)
  13. 7ATX電源 FSP_ATX-300PAF電路圖資料
  14. 0.18 MB   |  4次下載  |  2 積分
  15. 8常用電子元器件使用手冊
  16. 2.40 MB   |  4次下載  |  免費(fèi)

總榜

  1. 1matlab軟件下載入口
  2. 未知  |  935134次下載  |  10 積分
  3. 2開源硬件-PMP21529.1-4 開關(guān)降壓/升壓雙向直流/直流轉(zhuǎn)換器 PCB layout 設(shè)計
  4. 1.48MB  |  420064次下載  |  10 積分
  5. 3Altium DXP2002下載入口
  6. 未知  |  233089次下載  |  10 積分
  7. 4電路仿真軟件multisim 10.0免費(fèi)下載
  8. 340992  |  191422次下載  |  10 積分
  9. 5十天學(xué)會AVR單片機(jī)與C語言視頻教程 下載
  10. 158M  |  183352次下載  |  10 積分
  11. 6labview8.5下載
  12. 未知  |  81600次下載  |  10 積分
  13. 7Keil工具M(jìn)DK-Arm免費(fèi)下載
  14. 0.02 MB  |  73818次下載  |  10 積分
  15. 8LabVIEW 8.6下載
  16. 未知  |  65991次下載  |  10 積分