我有一个将一个导出AVMutableComposition
到.mov
文件的应用程序,我希望用户使用进度条查看导出状态,就像发送短信或上传文件一样。
当我知道任务的持续时间(例如播放音频文件)时,我知道如何创建进度条,但由于没有设置导出的持续时间,我不确定如何继续。
我目前有一个活动指示器,但它不能提供最佳的用户体验。
有没有人有任何指示?
我有一个将一个导出AVMutableComposition
到.mov
文件的应用程序,我希望用户使用进度条查看导出状态,就像发送短信或上传文件一样。
当我知道任务的持续时间(例如播放音频文件)时,我知道如何创建进度条,但由于没有设置导出的持续时间,我不确定如何继续。
我目前有一个活动指示器,但它不能提供最佳的用户体验。
有没有人有任何指示?
不久前我想出了一个答案,所以我会发布它以防它可以帮助某人:
首先,在您调用的方法中,您AVAssetExportSession
必须设置一个计时器,以便在UIProgressView
您启动导出后更新您的:
//`AVAssetExportSession` code here
self.exportProgressBarTimer = [NSTimer scheduledTimerWithTimeInterval:.1 target:self selector:@selector(updateExportDisplay) userInfo:nil repeats:YES];
...
然后,您需要一种方法来更新您的显示,同时考虑到进度属性AVAssetExportSession
从 0 到 1:
- (void)updateExportDisplay {
self.exportProgressBar.progress = exportSession.progress;
if (self.exportProgressBar.progress > .99) {
[self.exportProgressBarTimer invalidate];
}
}
斯威夫特 3 示例
使用通知中心向侦听器发送进度更新
//`AVAssetExportSession` code above
var exportProgressBarTimer = Timer() // initialize timer
if #available(iOS 10.0, *) {
exportProgressBarTimer = Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true) { timer in
// Get Progress
let progress = Float((exportSession?.progress)!);
if (progress < 0.99) {
let dict:[String: Float] = ["progress": progress]
NotificationCenter.default.post(name: Notification.Name("ProgressBarPercentage"), object: nil, userInfo: dict)
}
}
}
// on exportSession completed
exportSession?.exportAsynchronously(completionHandler: {
exportProgressBarTimer.invalidate(); // remove/invalidate timer
if exportSession?.status == AVAssetExportSessionStatus.completed {
// [....Some Completion Code Here]
}
})
然后在您想使用的任何地方设置通知中心侦听器
NotificationCenter.default.addObserver(self, selector: #selector(self.statusUpdate(_:)), name: NSNotification.Name(rawValue: "ProgressBarPercentage"), object: nil)
我在 iOS 8.0 中遇到过同样的问题,我使用 dispatch quee 解决了它
- (void)convertVideoToLowQuailtyWithInputURL:(NSURL*)inputURL outputURL:(NSURL*)outputURL handler:(void (^)(AVAssetExportSession*))handler{
[[NSFileManager defaultManager] removeItemAtURL:outputURL error:nil];
AVURLAsset *asset = [AVURLAsset URLAssetWithURL:inputURL options:nil];
exportSession2 = [[AVAssetExportSession alloc] initWithAsset:asset presetName:AVAssetExportPresetLowQuality];
exportSession2.outputURL = outputURL;
exportSession2.outputFileType = AVFileTypeQuickTimeMovie;
[exportSession2 exportAsynchronouslyWithCompletionHandler:^(void)
{
handler(exportSession2);
}];
dispatch_async(dispatch_get_main_queue(), ^(void){
self.exportProgressBarTimer = [NSTimer scheduledTimerWithTimeInterval:.1 target:self selector:@selector(updateExportDisplay) userInfo:nil repeats:YES];
});
}
使用下面的代码行。
AVAssetExportSession *session = [AVAssetExportSession exportSessionWithAsset:composition presetName:AVAssetExportPresetMediumQuality];
self.exportSession = session;
// 出力先(テンポラリファイル)の設定。
NSString *filePath = NSTemporaryDirectory();
filePath = [filePath stringByAppendingPathComponent:@"out.mov"];
[[NSFileManager defaultManager] removeItemAtPath:filePath error:nil];
session.outputURL = [NSURL fileURLWithPath:filePath];
// 出力タイプの設定。
session.outputFileType = AVFileTypeQuickTimeMovie;
// 非同期エクスポートの開始。
[session exportAsynchronouslyWithCompletionHandler:^{
if (session.status == AVAssetExportSessionStatusCompleted) {
// フォトアルバムへの書き込み。
ALAssetsLibrary *library = [[ALAssetsLibrary alloc] init];
[library writeVideoAtPathToSavedPhotosAlbum:session.outputURL completionBlock:^(NSURL *assetURL, NSError *error){
if (error) {
self.resultLabel.text = [NSString stringWithFormat:@"アセット書き込み失敗\n%@", error];
} else {
self.resultLabel.text = [NSString stringWithFormat:@"完了\n%@", assetURL];
}
}];
[library autorelease];
} else if (session.status == AVAssetExportSessionStatusCancelled) {
self.resultLabel.text = @"エクスポート中断";
} else {
self.resultLabel.text = [NSString stringWithFormat:@"エクスポート失敗\n%@", session.error];
}
}];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{
while (session.status == AVAssetExportSessionStatusExporting) {
dispatch_sync(dispatch_get_main_queue(), ^{
self.progressView.progress = session.progress;
});
}
});