MobileVLCKit是開源播放器VLC的iOS平臺框架,在Mac OS上也有對應的VLCKit,搞直播的同學應該不陌生,不過它其實還是一款強大的本地播放器,支持幾乎所有的主流媒體格式。最近在研究app如何瀏覽電腦上文件,然后直接做到播放視頻的功能。用過iOS上的VLC播放器的童鞋應該知道,他能做到掃描本地端口,然后通過輸入用戶名和密碼瀏覽電腦的文件,點擊視頻和音頻還能直接播放。
通過Google知道這里用到一個叫SMB的協議,不光是Mac OS上,Windows和Linux都支持這種協議。只要本地開啟SMB的文件共享服務,同一個局域網內的設備就能通過它訪問電腦上的文件了。
上上gayhub發現了一個SMB的iOS框架,叫TOSMBClient,它將一個C語言的框架封裝成了OC的框架。還支持CocoaPods,使用起來非常方便。而MobileVLCKit原生就支持SMB協議的在線播放。所以解決方案是通過TOSMBClient獲取文件列表,VLC播放,想法很美好,但是實際實現還是踩了不少坑。
先說說SMB的格式,長這樣:smb://{hostname}:{password}@{ip}/path
比如桌面上的一個mp4文件就應該長這樣:smb://xiaoming:123456@192.168.1.100/xiaoming/Desktop/233.mp4
hostname是域名,一般創建SMB共享協議的時候,就需要指定。password是密碼,ip是服務器的ip。
TOSMBClient提供了一個登錄的類叫TOSMBSession
,常用屬性是這幾個
//服務器域名
@property (nonatomic, copy) NSString *hostName;
//服務器ip
@property (nonatomic, copy) NSString *ipAddress;
//登錄的用戶名
@property (nonatomic, copy) NSString *userName;
//登錄密碼
@property (nonatomic, copy) NSString *password;
其中域名和ip可以都設置,也可以只設置其中一個,框架會自動查找。
然后通過TOSMBSession
提供的方法
- (void)requestContentsOfDirectoryAtFilePath:(NSString *)path
success:(void (^)(NSArray *files))successHandler
error:(void (^)(NSError *))errorHandler;
獲取文件列表,很簡單。返回的files是TOSMBSessionFile
類型。它包含基本的文件信息,比如路徑,名稱,大小。想法挺美好,有URL,直接給VLC播放不就行了,然后就碰到了第一個坑。
TOSMBSessionFile
提供的路徑只是smb格式的一部分,也就只有path部分,所以需要播放還得自己拼接成完整路徑。
拼接好了之后嘗試下播放個文件,沒帶中文的,成功了,高興之余本著嚴謹的態度試了下中文路徑,結果失敗了。NSURL初始化如果包含標準ASCALL以外的字符,會返回nil,這是第二個坑。
作為程序員,很自然會想到,URL如果帶中文,瀏覽器會自動做URL轉碼。所以嘗試下轉碼,發現還是播放失敗。這就讓我懷疑人生了,怎么肥事?而且連iOS上的VLC的app都有這個問題,這是第三個坑。
通過查找API我發現播放器除了通過NSURL初始化,還可以通過NSString初始化,我想,NSURL不讓包含中文,NSString總可以吧。結果還是播放失敗,神奇的是即使不包含中文,通過NSString初始化還是失敗,但是NSURL就可以,這是第四個坑。
既然一定要實現功能,那就得搞明白為什么,這時候開源的好處就體現出來了,我看了下MobileVLCKit的源碼,它的媒體類VLCMedia是這么寫的:
- (instancetype)initWithPath:(NSString *)aPath
{
return [self initWithURL:[NSURL fileURLWithPath:aPath isDirectory:NO]];
}
- (instancetype)initWithURL:(NSURL *)anURL
{
if (self = [super init]) {
const char *url;
VLCLibrary *library = [VLCLibrary sharedLibrary];
NSAssert(library.instance, @"no library instance when creating media");
if (([[anURL absoluteString] hasPrefix:@"sftp://"]) ||
([[anURL absoluteString] hasPrefix:@"smb://"])) {
url = [[[anURL absoluteString] stringByRemovingPercentEncoding] UTF8String];
} else {
url = [[anURL absoluteString] UTF8String];
}
p_md = libvlc_media_new_location(library.instance, url);
_metaDictionary = [[NSMutableDictionary alloc] initWithCapacity:3];
[self initInternalMediaDescriptor];
}
return self;
}
可以看到,initWithPath:
方法把字符串通過fileURLWithPath:isDirectory:
方法初始化成URL了,所以smb格式的字符串路徑通過這個方法初始化得到的路徑肯定是錯的,因為它不是標準的本地路徑,自然會出現上面神奇的情況。
然后看下initWithURL:
方法,if語句判斷如果包含smb前綴,則做URL解碼操作。說明我們的想法是正確的,確實應該對URL進行編碼。但是,經過測試編碼還是不行,這種情況就很費解了。到現在我還不知道什么原因,因為它自己的APP都有這個問題。不過之后偶然發現了解決方法,很簡單
**
把URL編碼兩次即可!!
把URL編碼兩次即可!!
把URL編碼兩次即可!!
**
重要的事情說三遍,編碼兩次之后框架會對URL解碼一次,所以得到的URL實際是編碼了一次的內容,這樣就能播放了,非常神奇,這是第五個坑。在這里分享一下給需要的童鞋。
注意事項
應該只對path
、hostname
、password
部分做URL兩次編碼,smb://
前綴不需要,否則播放器會無法識別。
2017.10.16日更新 如果發現拼接之后還是沒法播放,大部分去因為VLC的版本比較舊的關系,cocoapods里有最新的unstable版本,用這個,不過這個版本也是最不穩定的。
2018.7.8日更新 文章Demo