mirror of https://github.com/oxen-io/session-ios
Add audio attachment player.
* Fix two bugs around play/pause button appearance. * Fix bugs around stopping playback when leaving view/entering background. * Fix bugs around cleaning up playback state when leaving view/entering background. * Fix audio playback vs. hardware mute button. * Improve handling of invalid audio attachments. // FREEBIEpull/1/head
parent
61d72f8e95
commit
980d726a48
@ -0,0 +1,22 @@
|
|||||||
|
//
|
||||||
|
// Copyright (c) 2017 Open Whisper Systems. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
#import <AVFoundation/AVFoundation.h>
|
||||||
|
|
||||||
|
@class TSVideoAttachmentAdapter;
|
||||||
|
@class YapDatabaseConnection;
|
||||||
|
|
||||||
|
@interface OWSAudioAttachmentPlayer : NSObject <AVAudioPlayerDelegate>
|
||||||
|
|
||||||
|
@property (nonatomic, readonly) TSVideoAttachmentAdapter *mediaAdapter;
|
||||||
|
|
||||||
|
- (instancetype)initWithMediaAdapter:(TSVideoAttachmentAdapter *)mediaAdapter
|
||||||
|
databaseConnection:(YapDatabaseConnection *)databaseConnection;
|
||||||
|
|
||||||
|
- (void)play;
|
||||||
|
- (void)pause;
|
||||||
|
- (void)stop;
|
||||||
|
- (void)togglePlayState;
|
||||||
|
|
||||||
|
@end
|
@ -0,0 +1,173 @@
|
|||||||
|
//
|
||||||
|
// Copyright (c) 2017 Open Whisper Systems. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
#import "OWSAudioAttachmentPlayer.h"
|
||||||
|
#import "TSAttachment.h"
|
||||||
|
#import "TSAttachmentStream.h"
|
||||||
|
#import "TSVideoAttachmentAdapter.h"
|
||||||
|
#import "ViewControllerUtils.h"
|
||||||
|
#import <YapDatabase/YapDatabaseConnection.h>
|
||||||
|
|
||||||
|
@interface OWSAudioAttachmentPlayer ()
|
||||||
|
|
||||||
|
@property (nonatomic) TSAttachmentStream *attachmentStream;
|
||||||
|
|
||||||
|
@property (nonatomic, nullable) AVAudioPlayer *audioPlayer;
|
||||||
|
@property (nonatomic, nullable) NSTimer *audioPlayerPoller;
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
#pragma mark -
|
||||||
|
|
||||||
|
@implementation OWSAudioAttachmentPlayer
|
||||||
|
|
||||||
|
- (instancetype)initWithMediaAdapter:(TSVideoAttachmentAdapter *)mediaAdapter
|
||||||
|
databaseConnection:(YapDatabaseConnection *)databaseConnection
|
||||||
|
{
|
||||||
|
self = [super init];
|
||||||
|
if (!self) {
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
OWSAssert(mediaAdapter);
|
||||||
|
OWSAssert([mediaAdapter isAudio]);
|
||||||
|
OWSAssert(mediaAdapter.attachmentId);
|
||||||
|
OWSAssert(databaseConnection);
|
||||||
|
|
||||||
|
_mediaAdapter = mediaAdapter;
|
||||||
|
|
||||||
|
__block TSAttachment *attachment = nil;
|
||||||
|
[databaseConnection readWithBlock:^(YapDatabaseReadTransaction *transaction) {
|
||||||
|
attachment = [TSAttachment fetchObjectWithUniqueID:mediaAdapter.attachmentId transaction:transaction];
|
||||||
|
}];
|
||||||
|
OWSAssert(attachment);
|
||||||
|
|
||||||
|
if ([attachment isKindOfClass:[TSAttachmentStream class]]) {
|
||||||
|
self.attachmentStream = (TSAttachmentStream *)attachment;
|
||||||
|
}
|
||||||
|
OWSAssert(self.attachmentStream);
|
||||||
|
|
||||||
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
||||||
|
selector:@selector(applicationDidEnterBackground:)
|
||||||
|
name:UIApplicationDidEnterBackgroundNotification
|
||||||
|
object:nil];
|
||||||
|
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)dealloc
|
||||||
|
{
|
||||||
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
||||||
|
|
||||||
|
[self stop];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)applicationDidEnterBackground:(NSNotification *)notification
|
||||||
|
{
|
||||||
|
[self stop];
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - Methods
|
||||||
|
|
||||||
|
- (void)play
|
||||||
|
{
|
||||||
|
OWSAssert(self.attachmentStream);
|
||||||
|
OWSAssert(![self.mediaAdapter isAudioPlaying]);
|
||||||
|
|
||||||
|
[ViewControllerUtils setAudioIgnoresHardwareMuteSwitch:YES];
|
||||||
|
|
||||||
|
[self.audioPlayerPoller invalidate];
|
||||||
|
|
||||||
|
self.mediaAdapter.isAudioPlaying = YES;
|
||||||
|
self.mediaAdapter.isPaused = NO;
|
||||||
|
[self.mediaAdapter setAudioIconToPause];
|
||||||
|
|
||||||
|
if (!self.audioPlayer) {
|
||||||
|
NSError *error;
|
||||||
|
self.audioPlayer = [[AVAudioPlayer alloc] initWithContentsOfURL:self.attachmentStream.mediaURL error:&error];
|
||||||
|
if (error) {
|
||||||
|
DDLogError(@"%@ error: %@", self.tag, error);
|
||||||
|
[self stop];
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.audioPlayer.delegate = self;
|
||||||
|
}
|
||||||
|
|
||||||
|
[self.audioPlayer prepareToPlay];
|
||||||
|
[self.audioPlayer play];
|
||||||
|
self.audioPlayerPoller = [NSTimer scheduledTimerWithTimeInterval:.05
|
||||||
|
target:self
|
||||||
|
selector:@selector(audioPlayerUpdated:)
|
||||||
|
userInfo:nil
|
||||||
|
repeats:YES];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)pause
|
||||||
|
{
|
||||||
|
OWSAssert(self.attachmentStream);
|
||||||
|
|
||||||
|
self.mediaAdapter.isAudioPlaying = NO;
|
||||||
|
self.mediaAdapter.isPaused = YES;
|
||||||
|
[self.audioPlayer pause];
|
||||||
|
[self.audioPlayerPoller invalidate];
|
||||||
|
double current = [self.audioPlayer currentTime] / [_audioPlayer duration];
|
||||||
|
[self.mediaAdapter setAudioProgressFromFloat:(float)current];
|
||||||
|
[self.mediaAdapter setAudioIconToPlay];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)stop
|
||||||
|
{
|
||||||
|
OWSAssert(self.attachmentStream);
|
||||||
|
|
||||||
|
[self.audioPlayer pause];
|
||||||
|
[self.audioPlayerPoller invalidate];
|
||||||
|
[self.mediaAdapter setAudioProgressFromFloat:0];
|
||||||
|
[self.mediaAdapter setDurationOfAudio:_audioPlayer.duration];
|
||||||
|
[self.mediaAdapter setAudioIconToPlay];
|
||||||
|
self.mediaAdapter.isAudioPlaying = NO;
|
||||||
|
self.mediaAdapter.isPaused = NO;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)togglePlayState
|
||||||
|
{
|
||||||
|
OWSAssert(self.attachmentStream);
|
||||||
|
|
||||||
|
if (self.mediaAdapter.isAudioPlaying) {
|
||||||
|
[self pause];
|
||||||
|
} else {
|
||||||
|
[self play];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - Events
|
||||||
|
|
||||||
|
- (void)audioPlayerUpdated:(NSTimer *)timer
|
||||||
|
{
|
||||||
|
OWSAssert(self.audioPlayer);
|
||||||
|
OWSAssert(self.audioPlayerPoller);
|
||||||
|
|
||||||
|
double current = [self.audioPlayer currentTime] / [self.audioPlayer duration];
|
||||||
|
double interval = [self.audioPlayer duration] - [self.audioPlayer currentTime];
|
||||||
|
[self.mediaAdapter setDurationOfAudio:interval];
|
||||||
|
[self.mediaAdapter setAudioProgressFromFloat:(float)current];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag
|
||||||
|
{
|
||||||
|
[self stop];
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - Logging
|
||||||
|
|
||||||
|
+ (NSString *)tag
|
||||||
|
{
|
||||||
|
return [NSString stringWithFormat:@"[%@]", self.class];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (NSString *)tag
|
||||||
|
{
|
||||||
|
return self.class.tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
@end
|
Loading…
Reference in New Issue