blob: 92d5f917afc26b946710e2673780f1222b859eff [file] [log] [blame]
/*
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing,
software distributed under the License is distributed on an
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
KIND, either express or implied. See the License for the
specific language governing permissions and limitations
under the License.
*/
#import "CDVSound.h"
#import "CDVFile.h"
#import <AVFoundation/AVFoundation.h>
#include <math.h>
#define DOCUMENTS_SCHEME_PREFIX @"documents://"
#define HTTP_SCHEME_PREFIX @"http://"
#define HTTPS_SCHEME_PREFIX @"https://"
#define CDVFILE_PREFIX @"cdvfile://"
#define RECORDING_WAV @"wav"
@implementation CDVSound
@synthesize soundCache, avSession, currMediaId;
// Maps a url for a resource path for recording
- (NSURL*)urlForRecording:(NSString*)resourcePath
{
NSURL* resourceURL = nil;
NSString* filePath = nil;
NSString* docsPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[0];
// first check for correct extension
if ([[resourcePath pathExtension] caseInsensitiveCompare:RECORDING_WAV] != NSOrderedSame) {
resourceURL = nil;
NSLog(@"Resource for recording must have %@ extension", RECORDING_WAV);
} else if ([resourcePath hasPrefix:DOCUMENTS_SCHEME_PREFIX]) {
// try to find Documents:// resources
filePath = [resourcePath stringByReplacingOccurrencesOfString:DOCUMENTS_SCHEME_PREFIX withString:[NSString stringWithFormat:@"%@/", docsPath]];
NSLog(@"Will use resource '%@' from the documents folder with path = %@", resourcePath, filePath);
} else if ([resourcePath hasPrefix:CDVFILE_PREFIX]) {
CDVFile *filePlugin = [self.commandDelegate getCommandInstance:@"File"];
CDVFilesystemURL *url = [CDVFilesystemURL fileSystemURLWithString:resourcePath];
filePath = [filePlugin filesystemPathForURL:url];
if (filePath == nil) {
resourceURL = [NSURL URLWithString:resourcePath];
}
} else {
// if resourcePath is not from FileSystem put in tmp dir, else attempt to use provided resource path
NSString* tmpPath = [NSTemporaryDirectory()stringByStandardizingPath];
BOOL isTmp = [resourcePath rangeOfString:tmpPath].location != NSNotFound;
BOOL isDoc = [resourcePath rangeOfString:docsPath].location != NSNotFound;
if (!isTmp && !isDoc) {
// put in temp dir
filePath = [NSString stringWithFormat:@"%@/%@", tmpPath, resourcePath];
} else {
filePath = resourcePath;
}
}
if (filePath != nil) {
// create resourceURL
resourceURL = [NSURL fileURLWithPath:filePath];
}
return resourceURL;
}
// Maps a url for a resource path for playing
// "Naked" resource paths are assumed to be from the www folder as its base
- (NSURL*)urlForPlaying:(NSString*)resourcePath
{
NSURL* resourceURL = nil;
NSString* filePath = nil;
// first try to find HTTP:// or Documents:// resources
if ([resourcePath hasPrefix:HTTP_SCHEME_PREFIX] || [resourcePath hasPrefix:HTTPS_SCHEME_PREFIX]) {
// if it is a http url, use it
NSLog(@"Will use resource '%@' from the Internet.", resourcePath);
resourceURL = [NSURL URLWithString:resourcePath];
} else if ([resourcePath hasPrefix:DOCUMENTS_SCHEME_PREFIX]) {
NSString* docsPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[0];
filePath = [resourcePath stringByReplacingOccurrencesOfString:DOCUMENTS_SCHEME_PREFIX withString:[NSString stringWithFormat:@"%@/", docsPath]];
NSLog(@"Will use resource '%@' from the documents folder with path = %@", resourcePath, filePath);
} else if ([resourcePath hasPrefix:CDVFILE_PREFIX]) {
CDVFile *filePlugin = [self.commandDelegate getCommandInstance:@"File"];
CDVFilesystemURL *url = [CDVFilesystemURL fileSystemURLWithString:resourcePath];
filePath = [filePlugin filesystemPathForURL:url];
if (filePath == nil) {
resourceURL = [NSURL URLWithString:resourcePath];
}
} else {
// attempt to find file path in www directory or LocalFileSystem.TEMPORARY directory
filePath = [self.commandDelegate pathForResource:resourcePath];
if (filePath == nil) {
// see if this exists in the documents/temp directory from a previous recording
NSString* testPath = [NSString stringWithFormat:@"%@/%@", [NSTemporaryDirectory()stringByStandardizingPath], resourcePath];
if ([[NSFileManager defaultManager] fileExistsAtPath:testPath]) {
// inefficient as existence will be checked again below but only way to determine if file exists from previous recording
filePath = testPath;
NSLog(@"Will attempt to use file resource from LocalFileSystem.TEMPORARY directory");
} else {
// attempt to use path provided
filePath = resourcePath;
NSLog(@"Will attempt to use file resource '%@'", filePath);
}
} else {
NSLog(@"Found resource '%@' in the web folder.", filePath);
}
}
// if the resourcePath resolved to a file path, check that file exists
if (filePath != nil) {
// create resourceURL
resourceURL = [NSURL fileURLWithPath:filePath];
// try to access file
NSFileManager* fMgr = [NSFileManager defaultManager];
if (![fMgr fileExistsAtPath:filePath]) {
resourceURL = nil;
NSLog(@"Unknown resource '%@'", resourcePath);
}
}
return resourceURL;
}
// Creates or gets the cached audio file resource object
- (CDVAudioFile*)audioFileForResource:(NSString*)resourcePath withId:(NSString*)mediaId doValidation:(BOOL)bValidate forRecording:(BOOL)bRecord
{
BOOL bError = NO;
CDVMediaError errcode = MEDIA_ERR_NONE_SUPPORTED;
NSString* errMsg = @"";
NSString* jsString = nil;
CDVAudioFile* audioFile = nil;
NSURL* resourceURL = nil;
if ([self soundCache] == nil) {
[self setSoundCache:[NSMutableDictionary dictionaryWithCapacity:1]];
} else {
audioFile = [[self soundCache] objectForKey:mediaId];
}
if (audioFile == nil) {
// validate resourcePath and create
if ((resourcePath == nil) || ![resourcePath isKindOfClass:[NSString class]] || [resourcePath isEqualToString:@""]) {
bError = YES;
errcode = MEDIA_ERR_ABORTED;
errMsg = @"invalid media src argument";
} else {
audioFile = [[CDVAudioFile alloc] init];
audioFile.resourcePath = resourcePath;
audioFile.resourceURL = nil; // validate resourceURL when actually play or record
[[self soundCache] setObject:audioFile forKey:mediaId];
}
}
if (bValidate && (audioFile.resourceURL == nil)) {
if (bRecord) {
resourceURL = [self urlForRecording:resourcePath];
} else {
resourceURL = [self urlForPlaying:resourcePath];
}
if (resourceURL == nil) {
bError = YES;
errcode = MEDIA_ERR_ABORTED;
errMsg = [NSString stringWithFormat:@"Cannot use audio file from resource '%@'", resourcePath];
} else {
audioFile.resourceURL = resourceURL;
}
}
if (bError) {
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:errcode message:errMsg]];
[self.commandDelegate evalJs:jsString];
}
return audioFile;
}
// returns whether or not audioSession is available - creates it if necessary
- (BOOL)hasAudioSession
{
BOOL bSession = YES;
if (!self.avSession) {
NSError* error = nil;
self.avSession = [AVAudioSession sharedInstance];
if (error) {
// is not fatal if can't get AVAudioSession , just log the error
NSLog(@"error creating audio session: %@", [[error userInfo] description]);
self.avSession = nil;
bSession = NO;
}
}
return bSession;
}
// helper function to create a error object string
- (NSString*)createMediaErrorWithCode:(CDVMediaError)code message:(NSString*)message
{
NSMutableDictionary* errorDict = [NSMutableDictionary dictionaryWithCapacity:2];
[errorDict setObject:[NSNumber numberWithUnsignedInteger:code] forKey:@"code"];
[errorDict setObject:message ? message:@"" forKey:@"message"];
NSData* jsonData = [NSJSONSerialization dataWithJSONObject:errorDict options:0 error:nil];
return [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
}
- (void)create:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
NSString* resourcePath = [command argumentAtIndex:1];
CDVAudioFile* audioFile = [self audioFileForResource:resourcePath withId:mediaId doValidation:YES forRecording:NO];
if (audioFile == nil) {
NSString* errorMessage = [NSString stringWithFormat:@"Failed to initialize Media file with path %@", resourcePath];
NSString* jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_ABORTED message:errorMessage]];
[self.commandDelegate evalJs:jsString];
} else {
NSURL* resourceUrl = audioFile.resourceURL;
if (![resourceUrl isFileURL] && ![resourcePath hasPrefix:CDVFILE_PREFIX]) {
// First create an AVPlayerItem
AVPlayerItem* playerItem = [AVPlayerItem playerItemWithURL:resourceUrl];
// Subscribe to the AVPlayerItem's DidPlayToEndTime notification.
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(itemDidFinishPlaying:) name:AVPlayerItemDidPlayToEndTimeNotification object:playerItem];
// Subscribe to the AVPlayerItem's PlaybackStalledNotification notification.
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(itemStalledPlaying:) name:AVPlayerItemPlaybackStalledNotification object:playerItem];
// Pass the AVPlayerItem to a new player
avPlayer = [[AVPlayer alloc] initWithPlayerItem:playerItem];
//avPlayer = [[AVPlayer alloc] initWithURL:resourceUrl];
}
self.currMediaId = mediaId;
CDVPluginResult* result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK];
[self.commandDelegate sendPluginResult:result callbackId:command.callbackId];
}
}
- (void)setVolume:(CDVInvokedUrlCommand*)command
{
NSString* callbackId = command.callbackId;
#pragma unused(callbackId)
NSString* mediaId = [command argumentAtIndex:0];
NSNumber* volume = [command argumentAtIndex:1 withDefault:[NSNumber numberWithFloat:1.0]];
if ([self soundCache] != nil) {
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
if (audioFile != nil) {
audioFile.volume = volume;
if (audioFile.player) {
audioFile.player.volume = [volume floatValue];
}
[[self soundCache] setObject:audioFile forKey:mediaId];
}
}
// don't care for any callbacks
}
- (void)setRate:(CDVInvokedUrlCommand*)command
{
NSString* callbackId = command.callbackId;
#pragma unused(callbackId)
NSString* mediaId = [command argumentAtIndex:0];
NSNumber* rate = [command argumentAtIndex:1 withDefault:[NSNumber numberWithFloat:1.0]];
if ([self soundCache] != nil) {
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
if (audioFile != nil) {
audioFile.rate = rate;
if (audioFile.player) {
audioFile.player.enableRate = YES;
audioFile.player.rate = [rate floatValue];
}
if (avPlayer.currentItem && avPlayer.currentItem.asset){
float customRate = [rate floatValue];
[avPlayer setRate:customRate];
}
[[self soundCache] setObject:audioFile forKey:mediaId];
}
}
// don't care for any callbacks
}
- (void)startPlayingAudio:(CDVInvokedUrlCommand*)command
{
[self.commandDelegate runInBackground:^{
NSString* callbackId = command.callbackId;
#pragma unused(callbackId)
NSString* mediaId = [command argumentAtIndex:0];
NSString* resourcePath = [command argumentAtIndex:1];
NSDictionary* options = [command argumentAtIndex:2 withDefault:nil];
BOOL bError = NO;
NSString* jsString = nil;
CDVAudioFile* audioFile = [self audioFileForResource:resourcePath withId:mediaId doValidation:YES forRecording:NO];
if ((audioFile != nil) && (audioFile.resourceURL != nil)) {
if (audioFile.player == nil) {
bError = [self prepareToPlay:audioFile withId:mediaId];
}
if (!bError) {
//self.currMediaId = audioFile.player.mediaId;
self.currMediaId = mediaId;
// audioFile.player != nil or player was successfully created
// get the audioSession and set the category to allow Playing when device is locked or ring/silent switch engaged
if ([self hasAudioSession]) {
NSError* __autoreleasing err = nil;
NSNumber* playAudioWhenScreenIsLocked = [options objectForKey:@"playAudioWhenScreenIsLocked"];
BOOL bPlayAudioWhenScreenIsLocked = YES;
if (playAudioWhenScreenIsLocked != nil) {
bPlayAudioWhenScreenIsLocked = [playAudioWhenScreenIsLocked boolValue];
}
NSString* sessionCategory = bPlayAudioWhenScreenIsLocked ? AVAudioSessionCategoryPlayback : AVAudioSessionCategorySoloAmbient;
[self.avSession setCategory:sessionCategory error:&err];
if (![self.avSession setActive:YES error:&err]) {
// other audio with higher priority that does not allow mixing could cause this to fail
NSLog(@"Unable to play audio: %@", [err localizedFailureReason]);
bError = YES;
}
}
if (!bError) {
NSLog(@"Playing audio sample '%@'", audioFile.resourcePath);
double duration = 0;
if (avPlayer.currentItem && avPlayer.currentItem.asset) {
CMTime time = avPlayer.currentItem.asset.duration;
duration = CMTimeGetSeconds(time);
if (isnan(duration)) {
NSLog(@"Duration is infifnite, setting it to -1");
duration = -1;
}
if (audioFile.rate != nil){
float customRate = [audioFile.rate floatValue];
NSLog(@"Playing stream with AVPlayer & custom rate");
[avPlayer setRate:customRate];
} else {
NSLog(@"Playing stream with AVPlayer & default rate");
[avPlayer play];
}
} else {
NSNumber* loopOption = [options objectForKey:@"numberOfLoops"];
NSInteger numberOfLoops = 0;
if (loopOption != nil) {
numberOfLoops = [loopOption intValue] - 1;
}
audioFile.player.numberOfLoops = numberOfLoops;
if (audioFile.player.isPlaying) {
[audioFile.player stop];
audioFile.player.currentTime = 0;
}
if (audioFile.volume != nil) {
audioFile.player.volume = [audioFile.volume floatValue];
}
audioFile.player.enableRate = YES;
if (audioFile.rate != nil) {
audioFile.player.rate = [audioFile.rate floatValue];
}
[audioFile.player play];
duration = round(audioFile.player.duration * 1000) / 1000;
}
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%.3f);\n%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_DURATION, duration, @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_RUNNING];
[self.commandDelegate evalJs:jsString];
}
}
if (bError) {
/* I don't see a problem playing previously recorded audio so removing this section - BG
NSError* error;
// try loading it one more time, in case the file was recorded previously
audioFile.player = [[ AVAudioPlayer alloc ] initWithContentsOfURL:audioFile.resourceURL error:&error];
if (error != nil) {
NSLog(@"Failed to initialize AVAudioPlayer: %@\n", error);
audioFile.player = nil;
} else {
NSLog(@"Playing audio sample '%@'", audioFile.resourcePath);
audioFile.player.numberOfLoops = numberOfLoops;
[audioFile.player play];
} */
// error creating the session or player
// jsString = [NSString stringWithFormat: @"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, MEDIA_ERR_NONE_SUPPORTED];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_NONE_SUPPORTED message:nil]];
[self.commandDelegate evalJs:jsString];
}
}
// else audioFile was nil - error already returned from audioFile for resource
return;
}];
}
- (BOOL)prepareToPlay:(CDVAudioFile*)audioFile withId:(NSString*)mediaId
{
BOOL bError = NO;
NSError* __autoreleasing playerError = nil;
// create the player
NSURL* resourceURL = audioFile.resourceURL;
if ([resourceURL isFileURL]) {
audioFile.player = [[CDVAudioPlayer alloc] initWithContentsOfURL:resourceURL error:&playerError];
} else {
/*
NSMutableURLRequest* request = [NSMutableURLRequest requestWithURL:resourceURL];
NSString* userAgent = [self.commandDelegate userAgent];
if (userAgent) {
[request setValue:userAgent forHTTPHeaderField:@"User-Agent"];
}
NSURLResponse* __autoreleasing response = nil;
NSData* data = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&playerError];
if (playerError) {
NSLog(@"Unable to download audio from: %@", [resourceURL absoluteString]);
} else {
// bug in AVAudioPlayer when playing downloaded data in NSData - we have to download the file and play from disk
CFUUIDRef uuidRef = CFUUIDCreate(kCFAllocatorDefault);
CFStringRef uuidString = CFUUIDCreateString(kCFAllocatorDefault, uuidRef);
NSString* filePath = [NSString stringWithFormat:@"%@/%@", [NSTemporaryDirectory()stringByStandardizingPath], uuidString];
CFRelease(uuidString);
CFRelease(uuidRef);
[data writeToFile:filePath atomically:YES];
NSURL* fileURL = [NSURL fileURLWithPath:filePath];
audioFile.player = [[CDVAudioPlayer alloc] initWithContentsOfURL:fileURL error:&playerError];
}
*/
}
if (playerError != nil) {
NSLog(@"Failed to initialize AVAudioPlayer: %@\n", [playerError localizedDescription]);
audioFile.player = nil;
if (self.avSession) {
[self.avSession setActive:NO error:nil];
}
bError = YES;
} else {
audioFile.player.mediaId = mediaId;
audioFile.player.delegate = self;
if (avPlayer == nil)
bError = ![audioFile.player prepareToPlay];
}
return bError;
}
- (void)stopPlayingAudio:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if ((audioFile != nil) && (audioFile.player != nil)) {
NSLog(@"Stopped playing audio sample '%@'", audioFile.resourcePath);
[audioFile.player stop];
audioFile.player.currentTime = 0;
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
}
if (avPlayer.currentItem && avPlayer.currentItem.asset) {
NSLog(@"Stopped playing audio sample '%@'", audioFile.resourcePath);
[avPlayer seekToTime: kCMTimeZero
toleranceBefore: kCMTimeZero
toleranceAfter: kCMTimeZero
completionHandler: ^(BOOL finished){
if (finished) [avPlayer pause];
}];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
}
// ignore if no media playing
if (jsString) {
[self.commandDelegate evalJs:jsString];
}
}
- (void)pausePlayingAudio:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
NSString* jsString = nil;
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
if ((audioFile != nil) && ((audioFile.player != nil) || (avPlayer != nil))) {
NSLog(@"Paused playing audio sample '%@'", audioFile.resourcePath);
if (audioFile.player != nil) {
[audioFile.player pause];
} else if (avPlayer != nil) {
[avPlayer pause];
}
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_PAUSED];
}
// ignore if no media playing
if (jsString) {
[self.commandDelegate evalJs:jsString];
}
}
- (void)seekToAudio:(CDVInvokedUrlCommand*)command
{
// args:
// 0 = Media id
// 1 = seek to location in milliseconds
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
double position = [[command argumentAtIndex:1] doubleValue];
double posInSeconds = position / 1000;
NSString* jsString;
if ((audioFile != nil) && (audioFile.player != nil)) {
if (posInSeconds >= audioFile.player.duration) {
// The seek is past the end of file. Stop media and reset to beginning instead of seeking past the end.
[audioFile.player stop];
audioFile.player.currentTime = 0;
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%.3f);\n%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_POSITION, 0.0, @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
// NSLog(@"seekToEndJsString=%@",jsString);
} else {
audioFile.player.currentTime = posInSeconds;
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%f);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_POSITION, posInSeconds];
// NSLog(@"seekJsString=%@",jsString);
}
} else if (avPlayer != nil) {
int32_t timeScale = avPlayer.currentItem.asset.duration.timescale;
CMTime timeToSeek = CMTimeMakeWithSeconds(posInSeconds, timeScale);
BOOL isPlaying = (avPlayer.rate > 0 && !avPlayer.error);
BOOL isReadyToSeek = (avPlayer.status == AVPlayerStatusReadyToPlay) && (avPlayer.currentItem.status == AVPlayerItemStatusReadyToPlay);
// CB-10535:
// When dealing with remote files, we can get into a situation where we start playing before AVPlayer has had the time to buffer the file to be played.
// To avoid the app crashing in such a situation, we only seek if both the player and the player item are ready to play. If not ready, we send an error back to JS land.
if(isReadyToSeek) {
[avPlayer seekToTime: timeToSeek
toleranceBefore: kCMTimeZero
toleranceAfter: kCMTimeZero
completionHandler: ^(BOOL finished) {
if (isPlaying) [avPlayer play];
}];
} else {
CDVMediaError errcode = MEDIA_ERR_ABORTED;
NSString* errMsg = @"AVPlayerItem cannot service a seek request with a completion handler until its status is AVPlayerItemStatusReadyToPlay.";
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:errcode message:errMsg]];
}
}
[self.commandDelegate evalJs:jsString];
}
- (void)release:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
//NSString* mediaId = self.currMediaId;
if (mediaId != nil) {
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
if (audioFile != nil) {
if (audioFile.player && [audioFile.player isPlaying]) {
[audioFile.player stop];
}
if (audioFile.recorder && [audioFile.recorder isRecording]) {
[audioFile.recorder stop];
}
if (avPlayer != nil) {
[avPlayer pause];
avPlayer = nil;
}
if (self.avSession) {
[self.avSession setActive:NO error:nil];
self.avSession = nil;
}
[[self soundCache] removeObjectForKey:mediaId];
NSLog(@"Media with id %@ released", mediaId);
}
}
}
- (void)getCurrentPositionAudio:(CDVInvokedUrlCommand*)command
{
NSString* callbackId = command.callbackId;
NSString* mediaId = [command argumentAtIndex:0];
#pragma unused(mediaId)
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
double position = -1;
if ((audioFile != nil) && (audioFile.player != nil) && [audioFile.player isPlaying]) {
position = round(audioFile.player.currentTime * 1000) / 1000;
}
if (avPlayer) {
CMTime time = [avPlayer currentTime];
position = CMTimeGetSeconds(time);
}
CDVPluginResult* result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK messageAsDouble:position];
NSString* jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%.3f);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_POSITION, position];
[self.commandDelegate evalJs:jsString];
[self.commandDelegate sendPluginResult:result callbackId:callbackId];
}
- (void)startRecordingAudio:(CDVInvokedUrlCommand*)command
{
NSString* callbackId = command.callbackId;
#pragma unused(callbackId)
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [self audioFileForResource:[command argumentAtIndex:1] withId:mediaId doValidation:YES forRecording:YES];
__block NSString* jsString = nil;
__block NSString* errorMsg = @"";
if ((audioFile != nil) && (audioFile.resourceURL != nil)) {
__weak CDVSound* weakSelf = self;
void (^startRecording)(void) = ^{
NSError* __autoreleasing error = nil;
if (audioFile.recorder != nil) {
[audioFile.recorder stop];
audioFile.recorder = nil;
}
// get the audioSession and set the category to allow recording when device is locked or ring/silent switch engaged
if ([weakSelf hasAudioSession]) {
if (![weakSelf.avSession.category isEqualToString:AVAudioSessionCategoryPlayAndRecord]) {
[weakSelf.avSession setCategory:AVAudioSessionCategoryRecord error:nil];
}
if (![weakSelf.avSession setActive:YES error:&error]) {
// other audio with higher priority that does not allow mixing could cause this to fail
errorMsg = [NSString stringWithFormat:@"Unable to record audio: %@", [error localizedFailureReason]];
// jsString = [NSString stringWithFormat: @"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, MEDIA_ERR_ABORTED];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [weakSelf createMediaErrorWithCode:MEDIA_ERR_ABORTED message:errorMsg]];
[weakSelf.commandDelegate evalJs:jsString];
return;
}
}
// create a new recorder for each start record
NSDictionary *audioSettings = @{AVFormatIDKey: @(kAudioFormatMPEG4AAC),
AVSampleRateKey: @(44100),
AVNumberOfChannelsKey: @(1),
AVEncoderAudioQualityKey: @(AVAudioQualityMedium)
};
audioFile.recorder = [[CDVAudioRecorder alloc] initWithURL:audioFile.resourceURL settings:nil error:&error];
bool recordingSuccess = NO;
if (error == nil) {
audioFile.recorder.delegate = weakSelf;
audioFile.recorder.mediaId = mediaId;
audioFile.recorder.meteringEnabled = YES;
recordingSuccess = [audioFile.recorder record];
if (recordingSuccess) {
NSLog(@"Started recording audio sample '%@'", audioFile.resourcePath);
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_RUNNING];
[weakSelf.commandDelegate evalJs:jsString];
}
}
if ((error != nil) || (recordingSuccess == NO)) {
if (error != nil) {
errorMsg = [NSString stringWithFormat:@"Failed to initialize AVAudioRecorder: %@\n", [error localizedFailureReason]];
} else {
errorMsg = @"Failed to start recording using AVAudioRecorder";
}
audioFile.recorder = nil;
if (weakSelf.avSession) {
[weakSelf.avSession setActive:NO error:nil];
}
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [weakSelf createMediaErrorWithCode:MEDIA_ERR_ABORTED message:errorMsg]];
[weakSelf.commandDelegate evalJs:jsString];
}
};
SEL rrpSel = NSSelectorFromString(@"requestRecordPermission:");
if ([self hasAudioSession] && [self.avSession respondsToSelector:rrpSel])
{
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
[self.avSession performSelector:rrpSel withObject:^(BOOL granted){
if (granted) {
startRecording();
} else {
NSString* msg = @"Error creating audio session, microphone permission denied.";
NSLog(@"%@", msg);
audioFile.recorder = nil;
if (weakSelf.avSession) {
[weakSelf.avSession setActive:NO error:nil];
}
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_ABORTED message:msg]];
[weakSelf.commandDelegate evalJs:jsString];
}
}];
#pragma clang diagnostic pop
} else {
startRecording();
}
} else {
// file did not validate
NSString* errorMsg = [NSString stringWithFormat:@"Could not record audio at '%@'", audioFile.resourcePath];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_ABORTED message:errorMsg]];
[self.commandDelegate evalJs:jsString];
}
}
- (void)stopRecordingAudio:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if ((audioFile != nil) && (audioFile.recorder != nil)) {
NSLog(@"Stopped recording audio sample '%@'", audioFile.resourcePath);
[audioFile.recorder stop];
// no callback - that will happen in audioRecorderDidFinishRecording
}
// ignore if no media recording
if (jsString) {
[self.commandDelegate evalJs:jsString];
}
}
- (void)audioRecorderDidFinishRecording:(AVAudioRecorder*)recorder successfully:(BOOL)flag
{
CDVAudioRecorder* aRecorder = (CDVAudioRecorder*)recorder;
NSString* mediaId = aRecorder.mediaId;
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if (audioFile != nil) {
NSLog(@"Finished recording audio sample '%@'", audioFile.resourcePath);
}
if (flag) {
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
} else {
// jsString = [NSString stringWithFormat: @"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, MEDIA_ERR_DECODE];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_DECODE message:nil]];
}
if (self.avSession) {
[self.avSession setActive:NO error:nil];
}
[self.commandDelegate evalJs:jsString];
}
- (void)audioPlayerDidFinishPlaying:(AVAudioPlayer*)player successfully:(BOOL)flag
{
//commented as unused
CDVAudioPlayer* aPlayer = (CDVAudioPlayer*)player;
NSString* mediaId = aPlayer.mediaId;
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if (audioFile != nil) {
NSLog(@"Finished playing audio sample '%@'", audioFile.resourcePath);
}
if (flag) {
audioFile.player.currentTime = 0;
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
} else {
// jsString = [NSString stringWithFormat: @"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, MEDIA_ERR_DECODE];
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%@);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_ERROR, [self createMediaErrorWithCode:MEDIA_ERR_DECODE message:nil]];
}
if (self.avSession) {
[self.avSession setActive:NO error:nil];
}
[self.commandDelegate evalJs:jsString];
}
-(void)itemDidFinishPlaying:(NSNotification *) notification {
// Will be called when AVPlayer finishes playing playerItem
NSString* mediaId = self.currMediaId;
NSString* jsString = nil;
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_STOPPED];
if (self.avSession) {
[self.avSession setActive:NO error:nil];
}
[self.commandDelegate evalJs:jsString];
}
-(void)itemStalledPlaying:(NSNotification *) notification {
// Will be called when playback stalls due to buffer empty
NSLog(@"Stalled playback");
}
- (void)onMemoryWarning
{
[[self soundCache] removeAllObjects];
[self setSoundCache:nil];
[self setAvSession:nil];
[super onMemoryWarning];
}
- (void)dealloc
{
[[self soundCache] removeAllObjects];
}
- (void)onReset
{
for (CDVAudioFile* audioFile in [[self soundCache] allValues]) {
if (audioFile != nil) {
if (audioFile.player != nil) {
[audioFile.player stop];
audioFile.player.currentTime = 0;
}
if (audioFile.recorder != nil) {
[audioFile.recorder stop];
}
}
}
[[self soundCache] removeAllObjects];
}
- (void)getCurrentAmplitudeAudio:(CDVInvokedUrlCommand*)command
{
NSString* callbackId = command.callbackId;
NSString* mediaId = [command argumentAtIndex:0];
#pragma unused(mediaId)
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
float amplitude = 0; // The linear 0.0 .. 1.0 value
if ((audioFile != nil) && (audioFile.recorder != nil) && [audioFile.recorder isRecording]) {
[audioFile.recorder updateMeters];
float minDecibels = -60.0f; // Or use -60dB, which I measured in a silent room.
float decibels = [audioFile.recorder averagePowerForChannel:0];
if (decibels < minDecibels) {
amplitude = 0.0f;
} else if (decibels >= 0.0f) {
amplitude = 1.0f;
} else {
float root = 2.0f;
float minAmp = powf(10.0f, 0.05f * minDecibels);
float inverseAmpRange = 1.0f / (1.0f - minAmp);
float amp = powf(10.0f, 0.05f * decibels);
float adjAmp = (amp - minAmp) * inverseAmpRange;
amplitude = powf(adjAmp, 1.0f / root);
}
}
CDVPluginResult* result = [CDVPluginResult resultWithStatus:CDVCommandStatus_OK messageAsDouble:amplitude];
[self.commandDelegate sendPluginResult:result callbackId:callbackId];
}
- (void)resumeRecordingAudio:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if ((audioFile != nil) && (audioFile.recorder != nil)) {
NSLog(@"Resumed recording audio sample '%@'", audioFile.resourcePath);
[audioFile.recorder record];
// no callback - that will happen in audioRecorderDidFinishRecording
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_RUNNING];
}
// ignore if no media recording
if (jsString) {
[self.commandDelegate evalJs:jsString];
}
}
- (void)pauseRecordingAudio:(CDVInvokedUrlCommand*)command
{
NSString* mediaId = [command argumentAtIndex:0];
CDVAudioFile* audioFile = [[self soundCache] objectForKey:mediaId];
NSString* jsString = nil;
if ((audioFile != nil) && (audioFile.recorder != nil)) {
NSLog(@"Paused recording audio sample '%@'", audioFile.resourcePath);
[audioFile.recorder pause];
// no callback - that will happen in audioRecorderDidFinishRecording
// no callback - that will happen in audioRecorderDidFinishRecording
jsString = [NSString stringWithFormat:@"%@(\"%@\",%d,%d);", @"cordova.require('cordova-plugin-media.Media').onStatus", mediaId, MEDIA_STATE, MEDIA_PAUSED];
}
// ignore if no media recording
if (jsString) {
[self.commandDelegate evalJs:jsString];
}
}
@end
@implementation CDVAudioFile
@synthesize resourcePath;
@synthesize resourceURL;
@synthesize player, volume, rate;
@synthesize recorder;
@end
@implementation CDVAudioPlayer
@synthesize mediaId;
@end
@implementation CDVAudioRecorder
@synthesize mediaId;
@end