視頻播放器的封裝(支持橫豎屏切換单山、cell上播放)

用了幾天斷斷續(xù)續(xù)的空閑時間封裝了一個視頻播放器ZWPlayer,支持橫豎屏切換悠砚、cell上播放晓勇、滾動UITableView停止視頻、以及滾動UITableViewcell視頻播放器縮小到底部播放,具體請看如下效果圖宵蕉。這篇文章主要是講這個視頻播放器封裝的總體思路酝静。針對具體細節(jié)并沒有說太多,這里主要是針對幾個自己感覺值得注意的地方說明了下羡玛。具體實現(xiàn)請看Demo别智,Demo下載地址:https://github.com/ZhengYaWei1992/ZWPlayer

cell上播放動畫效果
全屏豎屏狀態(tài)
全屏橫屏狀態(tài)
非全屏狀態(tài)

在研究視頻播放相關(guān)的問題之前必須能弄懂一個很重要的問題,如何正確的控制橫豎屏切換問題稼稿。具體請參考我之前寫的文章:http://www.reibang.com/p/5be7eaf4a1a6薄榛。這里簡簡單說下,一般在項目中控制橫豎屏問題让歼,主要代碼一般會寫在UITabbarController中敞恋。而不是直接寫在UIViewController中,因為即使寫在UIViewController在push的情況下也不會起到什么作用谋右,最終的控制權(quán)是有UITabbarController控制的硬猫。當然這種情況只是針對push而言,如果僅僅是針對模態(tài)進入的控制器改执,情況又會完全相反啸蜜。看看tabBarController中是如何控制屏幕是否支持旋轉(zhuǎn)的辈挂。一般主要是重寫系統(tǒng)的下面另個方法衬横,來控制特定頁面是否支持旋轉(zhuǎn)。這里只是簡單說說终蒂,想了解更多蜂林,就參考上面的連接。

- (BOOL)shouldAutorotate{
    UINavigationController *nav = self.viewControllers[self.selectedIndex];
    if ([nav.topViewController isKindOfClass:[ViewController2 class]]) {
       return YES;
    }else if ([nav.topViewController isKindOfClass:[A_A_ViewController class]]){
        return YES;
    }
    return NO;
}
- (UIInterfaceOrientationMask)supportedInterfaceOrientations{
    UINavigationController *nav = self.selectedViewController;
    if ([nav.topViewController isKindOfClass:[ViewController2 class]]) {
        return UIInterfaceOrientationMaskAllButUpsideDown;
    }else if ([nav.topViewController isKindOfClass:[A_A_ViewController class]]){
         return UIInterfaceOrientationMaskAllButUpsideDown;
    }
    // 其他頁面
    return UIInterfaceOrientationMaskPortrait;
}

說下視頻播放器的層次結(jié)構(gòu)拇泣,這里我將控制層和播放曾完全給剝離開來編寫代碼噪叙,在視頻播放器的最上層是一個透明的UIView,這個UIView上放置了多個控件霉翔,諸如:返回按鈕构眯、暫停、加載提示早龟、時間label惫霸、UISlider、精度條以及全屏按鈕葱弟。而播放層主要用于播放視屏壹店,處理一些業(yè)務(wù)邏輯,諸如屏幕旋轉(zhuǎn)芝加、緩沖進度計算硅卢、視屏播放狀態(tài)監(jiān)控以及是否處于前臺射窒、是否退出應(yīng)用等。所以在實際風裝這個視頻播放器的過程中将塑,主要是創(chuàng)建了兩個類ZWPlayerControlView(控制層)和ZWPlayerView(處理播放相關(guān)業(yè)務(wù)邏輯)脉顿,以及一個ZWPlayer,這個類主要是對外提供接口点寥,以及一些常量的宏都放置在這個類里面艾疟。

控制層布局雖然簡單,但是有一點值得去說的敢辩”卫常可能會有人疑惑類似這種帶有加載精度的控制條是怎么弄得,可能有人會猜測UISlider難道還有一些其他自己不知道的屬性戚长。其實并不是這樣的盗冷,帶有緩沖的進度條主要是通過UISlider和UIProgerssView兩個控件相互結(jié)合的。其中UIProgressView主要負責顯示加載進度同廉,剩下的播放進度仪糖、寬進、后退都是由UISlider完成的迫肖。

看一下ZWPlayer中是如何監(jiān)聽屏幕旋轉(zhuǎn)的锅劝。這里監(jiān)聽屏幕旋轉(zhuǎn)的主要目的是為了更變?nèi)涟粹o的狀態(tài),至于控制層使用的是Masonry設(shè)置的約束咒程。

 [[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications];
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(onDeviceOrientationChange)
                                                 name:UIDeviceOrientationDidChangeNotification
                                               object:nil
     ];


- (void)onDeviceOrientationChange{
    if (_isCellVideo) {
        return;
    }
    UIDeviceOrientation orientation             = [UIDevice currentDevice].orientation;
    UIInterfaceOrientation interfaceOrientation = (UIInterfaceOrientation)orientation;
    switch (interfaceOrientation) {
        case UIInterfaceOrientationPortraitUpsideDown:{
            self.controlView.fullScreenBtn.selected = YES;
            self.isFullScreen = YES;
        }
            break;
        case UIInterfaceOrientationPortrait:{
            self.isFullScreen = !self.isFullScreen;
            self.controlView.fullScreenBtn.selected = NO;
        
            self.isFullScreen = NO;
            
        }
            break;
        case UIInterfaceOrientationLandscapeLeft:{
            self.controlView.fullScreenBtn.selected = YES;
     
            self.isFullScreen = YES;
        }
            break;
        case UIInterfaceOrientationLandscapeRight:{
            self.controlView.fullScreenBtn.selected = YES;
            self.isFullScreen = YES;
        }
            break;
            
        default:
            break;
    }
}

雖然上面的代碼片段可以在設(shè)備屏幕旋轉(zhuǎn)的時候鸠天,更新全屏按鈕的狀態(tài)讼育。但針對于設(shè)備并沒有旋轉(zhuǎn)帐姻,點擊了全屏按鈕在豎屏和橫屏之間切換,這又是如何做到的呢奶段?這里主要是強制更新設(shè)備方向饥瓷,具體請看如下代碼片段。下面的代碼片段針對ARC和非ARC兩種情況均提供了解決方案痹籍,但是有一點要注意: [[UIDevice currentDevice] setValue:[NSNumber numberWithInteger:UIInterfaceOrientationLandscapeRight] forKey:@"orientation"];這具代碼不能直接拿來使用呢铆,否則可能會被拒絕上架。

- (void)interfaceOrientation:(UIInterfaceOrientation)orientation{
    // arc下
    if ([[UIDevice currentDevice] respondsToSelector:@selector(setOrientation:)]) {
        SEL selector             = NSSelectorFromString(@"setOrientation:");
        NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[UIDevice instanceMethodSignatureForSelector:selector]];
        [invocation setSelector:selector];
        [invocation setTarget:[UIDevice currentDevice]];
        int val                  = orientation;
        // 從2開始是因為0 1 兩個參數(shù)已經(jīng)被selector和target占用
        [invocation setArgument:&val atIndex:2];
        [invocation invoke];
    }
    /*
     // 非arc下
     if ([[UIDevice currentDevice] respondsToSelector:@selector(setOrientation:)]) {
     [[UIDevice currentDevice] performSelector:@selector(setOrientation:)
     withObject:@(orientation)];
     }
     // 直接調(diào)用這個方法通不過apple上架審核
     [[UIDevice currentDevice] setValue:[NSNumber numberWithInteger:UIInterfaceOrientationLandscapeRight] forKey:@"orientation"];
     */
}

針對于常規(guī)的視屏播放就講這么多蹲缠,下面針對cell上播放視頻簡單的說一下棺克。cell上播放視屏,首先要考慮一點线定,ZWPlayer負責播放視頻的這個類娜谊,最好是創(chuàng)建一個單例對象。除了這一點斤讥,接下來著重要考慮的兩點是關(guān)于復(fù)用的問題以及如何監(jiān)控UITableView的滾啊東纱皆,判斷當前播放器是否依然在可視區(qū)域。

首先來看看如何在滾動UITablview的時候解決復(fù)用問題。如下代碼主要是寫在創(chuàng)建UITablViewCell的一個代理方法中派草。

if (indexPath.row== _currentIndexPath.row) {
                [cell.playBtn.superview sendSubviewToBack:cell.playBtn];
            }else{
                [cell.playBtn.superview bringSubviewToFront:cell.playBtn];
            }
            NSArray *indexpaths = [tableView indexPathsForVisibleRows];
            if (![indexpaths containsObject:_currentIndexPath]) {//復(fù)用
                if ([[UIApplication sharedApplication].keyWindow.subviews containsObject:_playerView]) {
                    _playerView.hidden = NO;
                }else{
                    _playerView.hidden = YES;
                }
            }else{
                if ([cell.picView.subviews containsObject:_playerView]) {
                    [cell.picView addSubview:_playerView];
                    [_playerView play];
                    _playerView.hidden = NO;
                }
            }

再來看看搀缠,如何判斷當前播放視屏的cell是否已經(jīng)滾出可視區(qū)范圍之內(nèi)。

//tableViewCell離開界面近迁,視頻消失
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{
    if (scrollView == self.tableView) {
        if(_playerView == nil){
            return;
        }
        if (_playerView.superview) {//播放的cell在當前屏幕可視區(qū)
            CGRect rectInTableView = [self.tableView rectForRowAtIndexPath:_currentIndexPath];
            CGRect rectInSuperview = [self.tableView convertRect:rectInTableView toView:[self.tableView superview]];
            
            if (rectInSuperview.origin.y<-self.currentCell.picView.frame.size.height||rectInSuperview.origin.y>[UIScreen mainScreen].bounds.size.height-64-49) {//往上拖動
                if ([[UIApplication sharedApplication].keyWindow.subviews containsObject:_playerView]&&_isBottomVideo) {
                    _isBottomVideo = YES;
                }else{
                    //播放視頻的cell不在當前可視區(qū)范圍內(nèi),放到window艺普,在底部顯示
                    [self toBottomVideo];
                }
            }else{
                if ([self.currentCell.picView.subviews containsObject:_playerView]) {
                    
                }else{
                    //播放視頻的cell在當前可視區(qū)范圍內(nèi),回到原來的cell上繼續(xù)播放
                    [self toCell];
                }
            }
        }
    }
    
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末钳踊,一起剝皮案震驚了整個濱河市衷敌,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌拓瞪,老刑警劉巖缴罗,帶你破解...
    沈念sama閱讀 211,042評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異祭埂,居然都是意外死亡面氓,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評論 2 384
  • 文/潘曉璐 我一進店門蛆橡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來舌界,“玉大人,你說我怎么就攤上這事泰演∩氚瑁” “怎么了?”我有些...
    開封第一講書人閱讀 156,674評論 0 345
  • 文/不壞的土叔 我叫張陵睦焕,是天一觀的道長藐握。 經(jīng)常有香客問我,道長垃喊,這世上最難降的妖魔是什么猾普? 我笑而不...
    開封第一講書人閱讀 56,340評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮本谜,結(jié)果婚禮上初家,老公的妹妹穿的比我還像新娘。我一直安慰自己乌助,他們只是感情好溜在,可當我...
    茶點故事閱讀 65,404評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著他托,像睡著了一般掖肋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上上祈,一...
    開封第一講書人閱讀 49,749評論 1 289
  • 那天培遵,我揣著相機與錄音浙芙,去河邊找鬼。 笑死籽腕,一個胖子當著我的面吹牛嗡呼,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播皇耗,決...
    沈念sama閱讀 38,902評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼南窗,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了郎楼?” 一聲冷哼從身側(cè)響起万伤,我...
    開封第一講書人閱讀 37,662評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎呜袁,沒想到半個月后敌买,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,110評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡阶界,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年虹钮,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片膘融。...
    茶點故事閱讀 38,577評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡芙粱,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出氧映,到底是詐尸還是另有隱情春畔,我是刑警寧澤,帶...
    沈念sama閱讀 34,258評論 4 328
  • 正文 年R本政府宣布岛都,位于F島的核電站律姨,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏疗绣。R本人自食惡果不足惜线召,卻給世界環(huán)境...
    茶點故事閱讀 39,848評論 3 312
  • 文/蒙蒙 一铺韧、第九天 我趴在偏房一處隱蔽的房頂上張望多矮。 院中可真熱鬧,春花似錦哈打、人聲如沸塔逃。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽湾盗。三九已至,卻和暖如春立轧,著一層夾襖步出監(jiān)牢的瞬間格粪,已是汗流浹背躏吊。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留帐萎,地道東北人比伏。 一個月前我還...
    沈念sama閱讀 46,271評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像疆导,于是被迫代替她去往敵國和親赁项。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,452評論 2 348

推薦閱讀更多精彩內(nèi)容