源碼閱讀請參考讀懂「 唱吧KTVHTTPCache 」設(shè)計思想
KTVCache播放不了android發(fā)送的視頻
問題
- iOS端播放不了android發(fā)送的視頻
- iOS播放端播放器:KTVCache+ZFPlayer,邊下邊播。
思考
- 排查視頻播放格式:ZFPlayer使用的AVPlayer播放;切換到IJKPlayer也播放不了胳徽。
- 對比發(fā)現(xiàn):播放不了的使用的KTVCache就已經(jīng)有錯誤日志了勾怒。下面是跟到報錯的第一條信息:
2021-04-22 16:57:50.815304+0800 BLing[6639:2793971] KTVHCDownload : 0x2816428b0, Complete
Error : Error Domain=NSURLErrorDomain Code=-999 "已取消" UserInfo={NSErrorFailingURLStringKey=https://bling-file.percent.cn/chat/video/f861b2cd-81a1-464e-bb25-316b67079c60_original.mp4, NSErrorFailingURLKey=https://bling-file.percent.cn/chat/video/f861b2cd-81a1-464e-bb25-316b67079c60_original.mp4, _NSURLErrorRelatedURLSessionTaskErrorKey=(
"LocalDataTask <55516087-B8D5-4AA0-824E-76A43B483301>.<8>"
), _NSURLErrorFailingURLSessionTaskErrorKey=LocalDataTask <55516087-B8D5-4AA0-824E-76A43B483301>.<8>, NSLocalizedDescription=已取消}
- 上網(wǎng)查了下以為是HTTPS的問題,然后去信任了HTTPS的證書腋颠。結(jié)果還是報錯。
- 考慮視頻格式,可能后綴是mp4,其實是其他視頻格式,找android同事確認焰雕,反饋說是mp4。我這邊下載了vlc和quicktime看著也是mp4芳杏。被這段迷惑了很久矩屁。
-
最后把視頻拷貝到手機端的safari:
后綴是amr。
思考
- amr支持邊下邊播嗎
- android統(tǒng)一發(fā)送格式
- 我們這邊框架能否支持
解決方案
- KTVCache不僅支持視頻爵赵,也支持其他文件格式吝秕。不過默認配置只支持了常用的視頻格式。
- 在播放失敗的時候空幻,downloadSetUnacceptableContentTypeDisposer打印類型烁峭,contentType = multipart/form-data
[KTVHTTPCache downloadSetUnacceptableContentTypeDisposer:^BOOL(NSURL *URL, NSString *contentType) {
BLLogVerbose(@"Unsupport Content-Type Filter reviced URL : %@, %@", URL, contentType);
return NO;
}];
- 把contentType添加支持的下載類型。問題解決
NSMutableArray * acceptableContentTypes = [NSMutableArray arrayWithArray:[KTVHCDownload download].acceptableContentTypes];
[acceptableContentTypes addObject:@"multipart/form-data"];
[KTVHCDownload download].acceptableContentTypes = acceptableContentTypes;
- 只修改了配置秕铛,不侵入業(yè)務(wù)
使用KTVCache需要考慮哪些問題
- 怎么分塊的
- 視頻播放器關(guān)閉之后则剃,邊下邊播任務(wù)是否停止。再次進入斷點續(xù)傳邏輯是怎么樣的?
- 拖到進度條之后如捅,文件是怎么下載的棍现。
針對上面的問題,我是直接分析了下載文件镜遣。KTVCache源碼相對比較復(fù)雜己肮,在滿足需求的情況下士袄,后續(xù)慢慢閱讀。
ZFPlayer橫屏后屏幕無響應(yīng)
- 這個bug跟了許久谎僻,view里所有代碼都注釋掉娄柳,只剩視頻播放器,橫屏后依然無響應(yīng)艘绍。只能從源碼角度分析:
-
ZFPlayer源碼分析了一遍赤拒,里面代碼view層級梳理下:
- 對比與正常視頻旋轉(zhuǎn)后的的不同:
a. ZFLandscapeWindow繼承自window,ZFLandscapeViewController是window的rootViewController
b. window.hidden一下吸引了我,查找下設(shè)置hidden诱鞠。跟蹤到下面代碼: - 旋轉(zhuǎn)后:
//ZFOrientationObserver
- (void)_rotationToLandscapeOrientation:(UIInterfaceOrientation)orientation {
if (UIInterfaceOrientationIsLandscape(orientation)) {
UIWindow *keyWindow = UIApplication.sharedApplication.keyWindow;
if (keyWindow != self.window && self.previousKeyWindow != keyWindow) {
self.previousKeyWindow = UIApplication.sharedApplication.keyWindow;
}
if (!self.window.isKeyWindow) {
self.window.hidden = NO;
[self.window makeKeyAndVisible];
}
}
}
- 再旋轉(zhuǎn)到正常:
//ZFOrientationObserver
- (void)ls_didRotateFromOrientation:(UIInterfaceOrientation)orientation {
if (self.orientationDidChanged) self.orientationDidChanged(self, self.isFullScreen);
if (!self.isFullScreen) {
UIView *containerView = nil;
if (self.rotateType == ZFRotateTypeCell) {
containerView = [self.cell viewWithTag:self.playerViewTag];
} else {
containerView = self.containerView;
}
[containerView addSubview:self.view];
self.view.frame = containerView.bounds;
UIWindow *previousKeyWindow = self.previousKeyWindow ?: UIApplication.sharedApplication.windows.firstObject;
[previousKeyWindow makeKeyAndVisible];
self.previousKeyWindow = nil;
self.window.hidden = YES;
}
}
- 旋轉(zhuǎn)后記錄了previousKeyWindow與當前生成的window
- 旋轉(zhuǎn)到橫屏?xí)|發(fā)ZFLandscapeViewController中的viewWillTransitionToSize挎挖,而無響應(yīng)的按鈕則不會觸發(fā)。
- 對比后發(fā)現(xiàn):旋轉(zhuǎn)能響應(yīng)的進入了forceRotaion條件
- (BOOL)ls_shouldAutorotate {
if (self.fullScreenMode == ZFFullScreenModePortrait) {
return NO;
}
UIInterfaceOrientation currentOrientation = (UIInterfaceOrientation)[UIDevice currentDevice].orientation;
if (![self _isSupported:currentOrientation]) {
return NO;
}
if (self.forceRotaion) {
[self _rotationToLandscapeOrientation:currentOrientation];
return YES;
}
if (!self.activeDeviceObserver) {
return NO;
}
[self _rotationToLandscapeOrientation:currentOrientation];
return YES;
}
- fullScreen
- (void)setFullScreen:(BOOL)fullScreen {
_fullScreen = fullScreen;
[self.window.landscapeViewController setNeedsStatusBarAppearanceUpdate];
[UIViewController attemptRotationToDeviceOrientation];
}
- 最后發(fā)現(xiàn)必現(xiàn)在使用播放器的地方設(shè)置:
- (BOOL)shouldAutorotate {
return NO;
}
小結(jié)
橫屏之后會調(diào)用[UIViewController attemptRotationToDeviceOrientation],接著調(diào)用了viewWillTransitionToSize:方法航夺,但viewController需要配置shouldAutorotate蕉朵。