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