欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

swift4.0實現(xiàn)視頻播放、屏幕旋轉(zhuǎn)、倍速播放、手勢調(diào)節(jié)及鎖屏面板等功能實例

 更新時間:2018年01月28日 12:00:11   作者:ControlM  
這篇文章主要給大家介紹了關(guān)于swift4.0實現(xiàn)視頻播放、屏幕旋轉(zhuǎn)、倍速播放、手勢調(diào)節(jié)及鎖屏面板等功能的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。

前言

學(xué)習(xí)swift有段時間了,原來寫過一個基于 swift 3.0 的視頻播放,后來有同學(xué)聯(lián)系我說,在音頻鎖屏的情況下,無法用控制面板拖動進(jìn)度條調(diào)節(jié)播放進(jìn)度,所以又將原來的代碼拿過來重新整理了下也順便更新到了4.0版本。在把原來的代碼拿來的時候發(fā)現(xiàn)原來有好多地方都是錯誤的,原來在 OC 項目里面已經(jīng)寫過一遍關(guān)于視頻播放的東西所以就按照原來的邏輯寫了 swift 版本,其實里面很多代碼我也是通過查找資料和看文檔拼湊出來的,對于 swift 的語句也是一知半解,希望各位看官多多包涵。

先來看一下實現(xiàn)的效果,一圖勝千言(第一張是 iOS 10系統(tǒng),第二張是 iOS 11系統(tǒng))。


 

demo下載地址  (本地下載)

工程介紹

簡單說一下工程結(jié)構(gòu),所有關(guān)于布局都是在Player文件夾下的MPlayerViewModel文件中,考慮到耦合度的原因,所以將視頻播放的所有 UI 布局全部抽離出來,在播放器 view 里將會頻繁看到一個叫viewModel的對象,它既 UI 布局也是布局控件的所有者。視頻播放的布局是基于SnapKit三方庫來布局了,因為在OC里用慣了Masonry所以工程里依然沿用這個庫。主要代碼是放到MPlayerView這個文件中的,其中還有一個由 OC 寫的DeviceTool文件主要用來做頁面強制旋轉(zhuǎn)用的,強制旋轉(zhuǎn)這一部分我現(xiàn)在還沒有更好的解決辦法只能橋接 OC 里的方法。

初始化播放器方法

視頻播放界面我用的是一個單例實現(xiàn)的,剛開始不是用單例實現(xiàn),但是為了把代碼拆出來放到各自的功能區(qū)所以用單例實現(xiàn)是最好的方法。由于swift放棄了OC里的dispatch_once實現(xiàn)單例方法,swift3.0以后的單例寫法:

/// 創(chuàng)建播放器單例
static let shared = MPlayerView()
private override init(frame: CGRect) {
super.init(frame: frame)
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

在swift3.0之后重寫init方法必須實現(xiàn)required init方法,這么做也是為了安全,因為在OC里init方法并不能保證子類完成初始化,增加required“這是由初始化方法的完備性需求所決定的,以保證類型的安全。在創(chuàng)建視頻播放視圖有兩種創(chuàng)建方式:1.用單利創(chuàng)建。2.init 初始化 ,這兩種方法都可以達(dá)到視頻播放的效果。

1.單利初始化

self.playerView = MPlayerView.shared.initWithFrame(frame: self.view.frame, videoUrl: videoUrl, type: "VIDEO")

2.init 初始化

self.playerView = MPlayerView().initWithFrame(frame: CGRect.init(x: 0, y: 0, width: Screen_width, height: Screen_width * 9/16), videoUrl: videoUrl, type: "VIDEO")

手勢滑動及注意事項

由于swift里面有嚴(yán)格的類型檢查,就比如在做手勢滑動的時候,手勢剛開始滑動的時候肯定需要記錄一下當(dāng)前播放器的位置我在項目中是定義的sumTime屬性是一個CMTime類型,如果在OC里大可不必這樣,來看一下swift與OC代碼的區(qū)別

swift寫法

/// 給sumTime初值
let time = self.player?.currentTime()
self.sumTime = CMTimeMake((time?.value)!, (time?.timescale)!)

OC寫法

// 給sumTime初值
CMTime time = self.player.currentTime;
self.sumTime = time.value/time.timescale;

滑動的距離是一個Double類型,而self.sumTime是CMTime類型,倆者肯定不能想加算出結(jié)束滑動的距離,所以將double類型轉(zhuǎn)換成CMTime類型用以下方法:

CMTime.init(seconds: Double.init(value/200), preferredTimescale: CMTimeScale(NSEC_PER_SEC))

如果是OC的話直接括號強轉(zhuǎn)類型即可實現(xiàn)。

知道滑動的距離和記錄滑動前的距離倆者想加即是當(dāng)前位置,轉(zhuǎn)化成CMTime類型:

self.sumTime = CMTimeAdd(self.sumTime!, addend)

手勢是滑動了,但是進(jìn)度條也是要跟著一起滑動的,有人說我把進(jìn)度條刷新放到player的代理里面,手勢滑動完只需要把時間傳給播放器,播放器根據(jù)當(dāng)前時間和總時間去更新進(jìn)度條,這樣做也對,但是有一點就是,如果網(wǎng)速不好,手勢已經(jīng)滑動到5分鐘了,而進(jìn)度條還停留在1分鐘的地方,播放器緩存完畢了,進(jìn)度條會瞬間跳到5分鐘,從而造成卡頓的假象體驗也不是很好,所以解決這個方法是手勢滑動的時候也更新進(jìn)度條,但是手勢滑動的時候都是CMTime類型,怎么轉(zhuǎn)成Float類型,因為slider?.value是float類型??梢赃@樣:通過CMTimeGetSeconds方法得到一個Float64再通過Float.init方法得到一個float類型,看一下實現(xiàn):

let sliderTime = CMTimeGetSeconds(self.sumTime!)/CMTimeGetSeconds(totalMovieDuration)
self.slider?.value = Float.init(sliderTime)

想查看整個過程可以看播放器手勢添加與創(chuàng)建這一塊,我已經(jīng)用MARK:標(biāo)記起來了。

設(shè)置控制面板信息

在視頻播放過程中,對視頻的監(jiān)聽是必不可少的,監(jiān)聽播放器狀態(tài),播放器緩存...等,由于播放器比較簡單,功能較少,剛開始我只監(jiān)聽了status屬性,后來我加上來loadedTimeRanges緩存狀態(tài),緩存這部分的緩存進(jìn)度計算我已經(jīng)實現(xiàn)了,但是沒有用到只是簡單的打印了一下。

在對播放器status屬性監(jiān)聽中加入了控制面板信息,是由MPNowPlayingInfoCenter來實現(xiàn)的,通過改變nowPlayingInfo里面對應(yīng)的信息來更新面板信息,里面有好多屬性,比如MPMediaItemPropertyTitle設(shè)置音頻標(biāo)題,MPMediaItemPropertyArtist作者、MPNowPlayingInfoPropertyElapsedPlaybackTime當(dāng)前播放過的時間、MPMediaItemPropertyPlaybackDuration播放總時間等等。剛開始做的時候因為鎖屏要更新時間,而nowPlayingInfo又是一個字典類型的再加上需要更新界面布局的時間和進(jìn)度條,直接將播放器時間強制轉(zhuǎn)換成 string 類型,所以將這一部分放到了時間觀察里面,因為時間觀察會一直進(jìn)行所以鎖屏界面信息也會一直更新,這樣帶來一個問題就是鎖屏界面的圖片如果是網(wǎng)絡(luò)圖片,每1秒就要請求一下圖片而且要不斷的更新這樣帶來的結(jié)果可想而知。后來才知道,將MPNowPlayingInfoPropertyElapsedPlaybackTime屬性設(shè)置成self.player!.currentTime()播放器當(dāng)前時間就會自動更新控制面板信息,調(diào)用的地方也很關(guān)鍵,必須放在播放器已經(jīng)播放的監(jiān)聽里面。

配置遠(yuǎn)程控制顯示的信息

響應(yīng)遠(yuǎn)程控制是由MPRemoteCommandCenter來實現(xiàn)的,里面有很多屬性,比如:playCommand播放響應(yīng)事件、pauseCommand 暫停響應(yīng)事件、nextTrackCommand下一曲響應(yīng)事件、likeCommand喜歡按鈕,類似網(wǎng)易云音樂的那個鎖屏,如果設(shè)置了likeCommand則dislikeCommand是上一首響應(yīng)事件、previousTrackCommand上一首,外部拖動進(jìn)度條是changePlaybackPositionCommand,系統(tǒng)有一個專門的方法來出來遠(yuǎn)程拖動進(jìn)度條響應(yīng)事件:

open func addTarget(handler: @escaping (MPRemoteCommandEvent) -> MPRemoteCommandHandlerStatus) -> Any

大概控制面板能用到的這些信息差不多也就這么多,如果想了解更多的可以看一下文檔或者查閱資料。

屏幕旋轉(zhuǎn)問題

一個視頻播放實現(xiàn)起來并不困難,只要處理好player與platitem就行了。最難的就是,如果手機屏幕旋轉(zhuǎn),怎么能讓視頻跟著屏幕自適應(yīng)呢,我在工程里面通過UIDevice變化添加的是屏幕旋轉(zhuǎn)監(jiān)聽:

/**
* 監(jiān)聽設(shè)備旋轉(zhuǎn)通知
*/
private func listeningRotating() {
UIDevice.current.beginGeneratingDeviceOrientationNotifications()
NotificationCenter.default.addObserver(self, selector: #selector(onDeviceOrientationChange), name:NSNotification.Name.UIDeviceOrientationDidChange, object: nil)
}

如果用戶把屏幕旋轉(zhuǎn)關(guān)掉,就是控制中心那個開關(guān),用戶旋轉(zhuǎn)屏幕,怎么能讓畫面跟著跑呢,我百度的很多資料,試了也很多方法,但是都不理想,用的還是OC的代碼,因為swift里面移除了NSInvocation屬性,用的依然是OC的屏幕強制旋轉(zhuǎn),只能使用橋接文件:

//這個方法是在網(wǎng)上找的
+ (void)interfaceOrientation:(UIInterfaceOrientation)orientation{
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];
 }
}

因為做的是視頻播放,所以進(jìn)入后臺后視頻會暫停,這個屬于正常現(xiàn)象,如果在視頻模式下,進(jìn)入后臺利用控制面板是無法將視頻播放的,如果在音頻模式下,進(jìn)入后臺利用控制面板是可以讓視頻播放的。大概就介紹這么多,一言半句也說得不是很明白,如果還有不明白的知識點可以去demo中自己去查,我也是一個初學(xué)者里面很多東西都是查資料得來的并不能保證其內(nèi)容的正確性。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關(guān)文章

  • SwiftUI 登錄界面布局實現(xiàn)示例詳解

    SwiftUI 登錄界面布局實現(xiàn)示例詳解

    這篇文章主要為大家介紹了SwiftUI 登錄界面布局實現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Swift類型創(chuàng)建之自定義一個類型詳解

    Swift類型創(chuàng)建之自定義一個類型詳解

    這篇文章主要介紹了Swift類型創(chuàng)建之自定義一個類型詳解,本文講解了自定義原型、實現(xiàn)默認(rèn)值、支持基本布爾型初始化、支持Bool類型判斷、支持兼容各們各派的類型、完善OCBool的布爾基因體系等內(nèi)容,需要的朋友可以參考下
    2015-05-05
  • SwiftUI開發(fā)總結(jié)combine原理簡單示例詳解

    SwiftUI開發(fā)總結(jié)combine原理簡單示例詳解

    這篇文章主要為大家介紹了SwiftUI開發(fā)總結(jié)combine原理簡單示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-02-02
  • 如何利用SwiftUI實現(xiàn)可縮放的圖片預(yù)覽器

    如何利用SwiftUI實現(xiàn)可縮放的圖片預(yù)覽器

    這篇文章主要給大家介紹了關(guān)于如何利用SwiftUI實現(xiàn)可縮放圖片預(yù)覽器的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用SwiftUI具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2021-09-09
  • Swift中常量和變量的區(qū)別與聲明詳解

    Swift中常量和變量的區(qū)別與聲明詳解

    Swift語言同樣和Java和OC等語言一樣是同樣是需要聲明常量和變量的,下面就讓我們來學(xué)習(xí)一下Swift的常量和變量。這篇文章主要給大家介紹了關(guān)于Swift中常量和變量的區(qū)別與聲明的相關(guān)資料,需要的朋友可以參考下。
    2017-11-11
  • 深入理解swift變量和函數(shù)

    深入理解swift變量和函數(shù)

    Swift 函數(shù)用來完成特定任務(wù)的獨立的代碼塊。這篇文章主要介紹了swift變量和函數(shù)的相關(guān)資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-08-08
  • Swift中非可選的可選值類型處理方法詳解

    Swift中非可選的可選值類型處理方法詳解

    Optional是Objective-C沒有的數(shù)據(jù)類型,是蘋果引入到Swift語言中的全新類型,它的特點就和它的名字一樣:可以有值,也可以沒有值,當(dāng)它沒有值時,就是nil。下面這篇文章主要給大家介紹了關(guān)于Swift中非可選的可選值類型處理方法的相關(guān)資料,需要的朋友可以參考下。
    2017-11-11
  • Swift代碼實現(xiàn)冒泡排序算法的簡單實例

    Swift代碼實現(xiàn)冒泡排序算法的簡單實例

    冒牌排序可謂最基本的排序算法之一,穩(wěn)定而沒有優(yōu)化空間:D 下面就一起來看一下Swift代碼實現(xiàn)冒泡排序算法的簡單實例:
    2016-06-06
  • Swift學(xué)習(xí)筆記之邏輯分支與循環(huán)體

    Swift學(xué)習(xí)筆記之邏輯分支與循環(huán)體

    這篇文章主要給大家介紹了關(guān)于Swift學(xué)習(xí)筆記之邏輯分支與循環(huán)體的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-12-12
  • Swift讀取App的版本信息與PCH文件詳解

    Swift讀取App的版本信息與PCH文件詳解

    這篇文章主要介紹了Swift讀取App的版本信息與PCH文件的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家具有一定的參考價值,需要的朋友們下面來一起看看吧。
    2017-03-03

最新評論