2013-12-09 12:36:43 +00:00
|
|
|
/*
|
2017-01-17 13:02:16 +00:00
|
|
|
Copyright 2013-2017 appPlant GmbH
|
|
|
|
|
|
|
|
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.
|
2013-12-09 12:36:43 +00:00
|
|
|
*/
|
2013-10-08 09:00:36 +00:00
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
#import "APPMethodMagic.h"
|
2013-10-08 09:00:36 +00:00
|
|
|
#import "APPBackgroundMode.h"
|
2017-01-17 13:02:16 +00:00
|
|
|
#import <Cordova/CDVAvailability.h>
|
2013-10-08 09:00:36 +00:00
|
|
|
|
|
|
|
@implementation APPBackgroundMode
|
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
#pragma mark -
|
|
|
|
#pragma mark Constants
|
|
|
|
|
2017-01-16 19:44:46 +00:00
|
|
|
NSString* const kAPPBackgroundJsNamespace = @"cordova.plugins.backgroundMode";
|
|
|
|
NSString* const kAPPBackgroundEventActivate = @"activate";
|
|
|
|
NSString* const kAPPBackgroundEventDeactivate = @"deactivate";
|
|
|
|
NSString* const kAPPBackgroundEventFailure = @"failure";
|
2014-12-13 23:04:12 +00:00
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
#pragma mark -
|
2017-01-17 13:02:16 +00:00
|
|
|
#pragma mark Life Cycle
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called by runtime once the Class has been loaded.
|
|
|
|
* Exchange method implementations to hook into their execution.
|
|
|
|
*/
|
|
|
|
+ (void) load
|
|
|
|
{
|
|
|
|
[self swizzleWKWebViewEngine];
|
|
|
|
}
|
2013-10-08 12:36:19 +00:00
|
|
|
|
2013-10-09 09:40:06 +00:00
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Initialize the plugin.
|
|
|
|
*/
|
|
|
|
- (void) pluginInitialize
|
|
|
|
{
|
2017-01-16 19:44:46 +00:00
|
|
|
[self.commandDelegate runInBackground:^{
|
2017-01-18 17:00:57 +00:00
|
|
|
enabled = [self.class isRunningWebKit];
|
2017-01-16 19:44:46 +00:00
|
|
|
[self configureAudioPlayer];
|
|
|
|
[self configureAudioSession];
|
|
|
|
[self observeLifeCycle];
|
|
|
|
}];
|
2014-11-04 13:05:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Register the listener for pause and resume events.
|
2013-10-09 09:40:06 +00:00
|
|
|
*/
|
2014-11-17 13:19:29 +00:00
|
|
|
- (void) observeLifeCycle
|
2013-10-09 09:40:06 +00:00
|
|
|
{
|
2017-01-16 19:44:46 +00:00
|
|
|
NSNotificationCenter* listener = [NSNotificationCenter
|
|
|
|
defaultCenter];
|
2017-01-18 17:00:57 +00:00
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
[listener addObserver:self
|
|
|
|
selector:@selector(keepAwake)
|
|
|
|
name:UIApplicationDidEnterBackgroundNotification
|
|
|
|
object:nil];
|
|
|
|
|
|
|
|
[listener addObserver:self
|
|
|
|
selector:@selector(stopKeepingAwake)
|
|
|
|
name:UIApplicationWillEnterForegroundNotification
|
|
|
|
object:nil];
|
2017-01-18 17:00:57 +00:00
|
|
|
|
|
|
|
if ([self.class isRunningWebKit])
|
|
|
|
return;
|
2014-11-04 13:05:45 +00:00
|
|
|
|
|
|
|
[listener addObserver:self
|
|
|
|
selector:@selector(handleAudioSessionInterruption:)
|
|
|
|
name:AVAudioSessionInterruptionNotification
|
|
|
|
object:nil];
|
2013-10-09 09:40:06 +00:00
|
|
|
}
|
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
#pragma mark -
|
2017-01-16 14:15:15 +00:00
|
|
|
#pragma mark Interface
|
2014-11-04 13:05:45 +00:00
|
|
|
|
2013-10-09 08:50:15 +00:00
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Enable the mode to stay awake
|
|
|
|
* when switching to background for the next time.
|
2013-10-09 08:50:15 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) enable:(CDVInvokedUrlCommand*)command
|
2013-10-09 08:50:15 +00:00
|
|
|
{
|
2017-01-18 17:00:57 +00:00
|
|
|
if (enabled)
|
|
|
|
return;
|
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
enabled = YES;
|
2017-01-16 19:44:46 +00:00
|
|
|
[self execCallback:command];
|
2013-10-09 08:50:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Disable the background mode
|
|
|
|
* and stop being active in background.
|
2013-10-09 08:50:15 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) disable:(CDVInvokedUrlCommand*)command
|
2013-10-09 08:50:15 +00:00
|
|
|
{
|
2017-01-18 17:00:57 +00:00
|
|
|
if (!enabled || [self.class isRunningWebKit])
|
|
|
|
return;
|
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
enabled = NO;
|
|
|
|
[self stopKeepingAwake];
|
2017-01-16 19:44:46 +00:00
|
|
|
[self execCallback:command];
|
2013-10-09 08:50:15 +00:00
|
|
|
}
|
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
#pragma mark -
|
2017-01-16 14:15:15 +00:00
|
|
|
#pragma mark Core
|
2014-11-04 13:05:45 +00:00
|
|
|
|
2013-10-09 08:50:15 +00:00
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Keep the app awake.
|
2013-10-09 08:50:15 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) keepAwake
|
|
|
|
{
|
2017-01-16 19:44:46 +00:00
|
|
|
if (!enabled)
|
|
|
|
return;
|
|
|
|
|
2017-01-18 17:00:57 +00:00
|
|
|
if (![self.class isRunningWebKit]) {
|
|
|
|
[audioPlayer play];
|
|
|
|
}
|
|
|
|
|
2017-01-16 19:44:46 +00:00
|
|
|
[self fireEvent:kAPPBackgroundEventActivate];
|
2013-10-09 09:40:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Let the app going to sleep.
|
2013-10-09 09:40:06 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) stopKeepingAwake
|
|
|
|
{
|
2014-11-17 13:56:26 +00:00
|
|
|
if (TARGET_IPHONE_SIMULATOR) {
|
|
|
|
NSLog(@"BackgroundMode: On simulator apps never pause in background!");
|
|
|
|
}
|
|
|
|
|
2017-01-18 17:00:57 +00:00
|
|
|
if (audioPlayer.isPlaying || [self.class isRunningWebKit]) {
|
2017-01-16 19:44:46 +00:00
|
|
|
[self fireEvent:kAPPBackgroundEventDeactivate];
|
2014-12-13 23:04:12 +00:00
|
|
|
}
|
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
[audioPlayer pause];
|
2013-10-09 09:40:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Configure the audio player.
|
2013-10-09 09:40:06 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) configureAudioPlayer
|
|
|
|
{
|
|
|
|
NSString* path = [[NSBundle mainBundle]
|
|
|
|
pathForResource:@"appbeep" ofType:@"wav"];
|
2014-11-04 13:05:45 +00:00
|
|
|
|
|
|
|
NSURL* url = [NSURL fileURLWithPath:path];
|
2013-10-09 09:40:06 +00:00
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
|
2017-01-16 14:15:15 +00:00
|
|
|
audioPlayer = [[AVAudioPlayer alloc]
|
|
|
|
initWithContentsOfURL:url error:NULL];
|
2014-11-04 13:05:45 +00:00
|
|
|
|
2017-01-16 14:15:15 +00:00
|
|
|
audioPlayer.volume = 0;
|
2014-11-04 13:05:45 +00:00
|
|
|
audioPlayer.numberOfLoops = -1;
|
|
|
|
};
|
2013-10-09 09:40:06 +00:00
|
|
|
|
|
|
|
/**
|
2014-11-04 13:05:45 +00:00
|
|
|
* Configure the audio session.
|
2013-10-09 09:40:06 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) configureAudioSession
|
|
|
|
{
|
2014-11-04 13:05:45 +00:00
|
|
|
AVAudioSession* session = [AVAudioSession
|
|
|
|
sharedInstance];
|
2017-01-16 14:15:15 +00:00
|
|
|
|
2017-01-18 17:00:57 +00:00
|
|
|
if ([self.class isRunningWebKit])
|
|
|
|
return;
|
|
|
|
|
2017-01-15 09:29:51 +00:00
|
|
|
// Don't activate the audio session yet
|
|
|
|
[session setActive:NO error:NULL];
|
2017-01-16 14:15:15 +00:00
|
|
|
|
2014-11-04 13:05:45 +00:00
|
|
|
// Play music even in background and dont stop playing music
|
|
|
|
// even another app starts playing sound
|
|
|
|
[session setCategory:AVAudioSessionCategoryPlayback
|
|
|
|
withOptions:AVAudioSessionCategoryOptionMixWithOthers
|
|
|
|
error:NULL];
|
2017-01-16 14:15:15 +00:00
|
|
|
|
2017-01-15 09:29:51 +00:00
|
|
|
// Active the audio session
|
2014-11-04 13:05:45 +00:00
|
|
|
[session setActive:YES error:NULL];
|
|
|
|
};
|
2013-10-08 09:00:36 +00:00
|
|
|
|
2014-12-13 23:04:12 +00:00
|
|
|
#pragma mark -
|
2017-01-16 14:15:15 +00:00
|
|
|
#pragma mark Helper
|
2014-12-13 23:04:12 +00:00
|
|
|
|
2017-01-16 19:44:46 +00:00
|
|
|
/**
|
|
|
|
* Simply invokes the callback without any parameter.
|
|
|
|
*/
|
|
|
|
- (void) execCallback:(CDVInvokedUrlCommand*)command
|
|
|
|
{
|
|
|
|
CDVPluginResult *result = [CDVPluginResult
|
|
|
|
resultWithStatus:CDVCommandStatus_OK];
|
|
|
|
|
|
|
|
[self.commandDelegate sendPluginResult:result
|
|
|
|
callbackId:command.callbackId];
|
|
|
|
}
|
|
|
|
|
2013-10-09 08:50:15 +00:00
|
|
|
/**
|
2014-11-05 11:08:34 +00:00
|
|
|
* Restart playing sound when interrupted by phone calls.
|
2013-10-09 08:50:15 +00:00
|
|
|
*/
|
2017-01-16 14:15:15 +00:00
|
|
|
- (void) handleAudioSessionInterruption:(NSNotification*)notification
|
|
|
|
{
|
2017-01-16 19:44:46 +00:00
|
|
|
[self fireEvent:kAPPBackgroundEventDeactivate];
|
2014-11-05 11:08:34 +00:00
|
|
|
[self keepAwake];
|
2013-10-08 12:36:19 +00:00
|
|
|
}
|
2013-10-08 09:00:36 +00:00
|
|
|
|
2017-01-18 17:00:57 +00:00
|
|
|
/**
|
|
|
|
* Find out if the app runs inside the webkit powered webview.
|
|
|
|
*/
|
|
|
|
+ (BOOL) isRunningWebKit
|
|
|
|
{
|
|
|
|
return IsAtLeastiOSVersion(@"8.0") && NSClassFromString(@"CDVWKWebViewEngine");
|
|
|
|
}
|
|
|
|
|
2014-12-13 23:04:12 +00:00
|
|
|
/**
|
|
|
|
* Method to fire an event with some parameters in the browser.
|
|
|
|
*/
|
2017-01-16 19:44:46 +00:00
|
|
|
- (void) fireEvent:(NSString*)event
|
2014-12-13 23:04:12 +00:00
|
|
|
{
|
2017-01-16 19:44:46 +00:00
|
|
|
NSString* active =
|
|
|
|
[event isEqualToString:kAPPBackgroundEventActivate] ? @"true" : @"false";
|
2014-12-14 13:23:57 +00:00
|
|
|
|
|
|
|
NSString* flag = [NSString stringWithFormat:@"%@._isActive=%@;",
|
2017-01-16 14:15:15 +00:00
|
|
|
kAPPBackgroundJsNamespace, active];
|
2014-12-14 13:23:57 +00:00
|
|
|
|
2017-01-16 19:44:46 +00:00
|
|
|
NSString* depFn = [NSString stringWithFormat:@"%@.on%@();",
|
|
|
|
kAPPBackgroundJsNamespace, event];
|
|
|
|
|
|
|
|
NSString* fn = [NSString stringWithFormat:@"%@.fireEvent('%@');",
|
|
|
|
kAPPBackgroundJsNamespace, event];
|
2014-12-14 13:23:57 +00:00
|
|
|
|
2017-01-16 14:15:15 +00:00
|
|
|
NSString* js = [NSString stringWithFormat:@"%@%@%@", flag, depFn, fn];
|
2014-12-13 23:04:12 +00:00
|
|
|
|
|
|
|
[self.commandDelegate evalJs:js];
|
|
|
|
}
|
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
#pragma mark -
|
|
|
|
#pragma mark Swizzling
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Swizzle some implementations of CDVWKWebViewEngine.
|
|
|
|
*/
|
|
|
|
+ (void) swizzleWKWebViewEngine
|
|
|
|
{
|
2017-01-18 17:00:57 +00:00
|
|
|
if (![self isRunningWebKit])
|
2017-01-17 13:02:16 +00:00
|
|
|
return;
|
2017-01-18 16:11:46 +00:00
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
Class wkWebViewEngineCls = NSClassFromString(@"CDVWKWebViewEngine");
|
|
|
|
SEL selector = NSSelectorFromString(@"createConfigurationFromSettings:");
|
2017-01-18 16:11:46 +00:00
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
SwizzleSelectorWithBlock_Begin(wkWebViewEngineCls, selector)
|
|
|
|
^(CDVPlugin *self, NSDictionary *settings) {
|
|
|
|
id obj = ((id (*)(id, SEL, NSDictionary*))_imp)(self, _cmd, settings);
|
2017-01-18 16:11:46 +00:00
|
|
|
|
2017-01-17 13:02:16 +00:00
|
|
|
SEL sel = NSSelectorFromString(@"_setAlwaysRunsAtForegroundPriority:");
|
2017-01-18 16:11:46 +00:00
|
|
|
((void (*)(id, SEL, BOOL))[obj methodForSelector:sel])(obj, sel, YES);
|
|
|
|
|
|
|
|
return obj;
|
2017-01-17 13:02:16 +00:00
|
|
|
}
|
|
|
|
SwizzleSelectorWithBlock_End;
|
|
|
|
}
|
|
|
|
|
2014-02-25 23:15:00 +00:00
|
|
|
@end
|