2014-10-26 15:47:18 +00:00
|
|
|
/*
|
2017-01-01 21:40:41 +00:00
|
|
|
Copyright 2013-2017 appPlant GmbH
|
2014-10-26 15:47:18 +00:00
|
|
|
|
|
|
|
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.
|
2014-11-04 16:59:33 +00:00
|
|
|
*/
|
2014-10-26 15:47:18 +00:00
|
|
|
|
|
|
|
package de.appplant.cordova.plugin.background;
|
|
|
|
|
|
|
|
import android.app.Activity;
|
2019-02-04 19:18:34 +00:00
|
|
|
import android.content.*;
|
|
|
|
import android.os.Bundle;
|
2014-10-26 15:47:18 +00:00
|
|
|
import android.os.IBinder;
|
|
|
|
|
2016-08-17 09:14:47 +00:00
|
|
|
import org.apache.cordova.CallbackContext;
|
2019-02-04 19:18:34 +00:00
|
|
|
import org.apache.cordova.CordovaInterface;
|
2016-08-17 09:14:47 +00:00
|
|
|
import org.apache.cordova.CordovaPlugin;
|
2019-02-04 19:18:34 +00:00
|
|
|
import org.apache.cordova.CordovaWebView;
|
2016-08-17 09:14:47 +00:00
|
|
|
import org.json.JSONArray;
|
|
|
|
import org.json.JSONObject;
|
|
|
|
|
2017-02-10 12:10:46 +00:00
|
|
|
import de.appplant.cordova.plugin.background.ForegroundService.ForegroundBinder;
|
|
|
|
|
2017-02-06 14:59:57 +00:00
|
|
|
import static android.content.Context.BIND_AUTO_CREATE;
|
2019-02-03 14:58:40 +00:00
|
|
|
import static de.appplant.cordova.plugin.background.BackgroundExt.clearKeyguardFlags;
|
2017-02-06 14:59:57 +00:00
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
public class BackgroundMode extends CordovaPlugin {
|
|
|
|
|
2014-12-13 23:04:12 +00:00
|
|
|
// Event types for callbacks
|
2019-02-03 14:58:40 +00:00
|
|
|
private enum Event { ACTIVATE, DEACTIVATE, FAILURE }
|
2014-12-13 23:04:12 +00:00
|
|
|
|
2014-12-14 12:23:10 +00:00
|
|
|
// Plugin namespace
|
2019-02-03 14:58:40 +00:00
|
|
|
private static final String JS_NAMESPACE = "cordova.plugins.backgroundMode";
|
2014-12-14 12:23:10 +00:00
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
// Flag indicates if the app is in background or foreground
|
|
|
|
private boolean inBackground = false;
|
|
|
|
|
|
|
|
// Flag indicates if the plugin is enabled or disabled
|
2014-11-12 22:04:01 +00:00
|
|
|
private boolean isDisabled = true;
|
2014-10-26 15:47:18 +00:00
|
|
|
|
2014-11-07 18:55:24 +00:00
|
|
|
// Flag indicates if the service is bind
|
|
|
|
private boolean isBind = false;
|
|
|
|
|
2014-12-14 12:38:36 +00:00
|
|
|
// Default settings for the notification
|
|
|
|
private static JSONObject defaultSettings = new JSONObject();
|
|
|
|
|
2017-01-01 15:58:08 +00:00
|
|
|
// Service that keeps the app awake
|
|
|
|
private ForegroundService service;
|
2015-06-03 18:45:26 +00:00
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
// Used to (un)bind the service to with the activity
|
2019-02-03 14:58:40 +00:00
|
|
|
private final ServiceConnection connection = new ServiceConnection()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
@Override
|
2019-02-03 14:58:40 +00:00
|
|
|
public void onServiceConnected (ComponentName name, IBinder service)
|
|
|
|
{
|
2017-02-10 12:10:46 +00:00
|
|
|
ForegroundBinder binder = (ForegroundBinder) service;
|
2017-01-01 15:58:08 +00:00
|
|
|
BackgroundMode.this.service = binder.getService();
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2019-02-03 14:58:40 +00:00
|
|
|
public void onServiceDisconnected (ComponentName name)
|
|
|
|
{
|
2017-01-30 08:41:45 +00:00
|
|
|
fireEvent(Event.FAILURE, "'service disconnected'");
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-02-04 19:18:34 +00:00
|
|
|
@Override
|
|
|
|
public void initialize(CordovaInterface cordova, CordovaWebView webView) {
|
|
|
|
super.initialize(cordova, webView);
|
|
|
|
IntentFilter filter = new IntentFilter();
|
|
|
|
filter.addAction("com.backgroundmode.close");
|
|
|
|
cordova.getActivity().registerReceiver(receiver, filter);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private BroadcastReceiver receiver = new BroadcastReceiver() {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onReceive(Context context, Intent intent) {
|
|
|
|
cordova.getActivity().finish();
|
|
|
|
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
/**
|
|
|
|
* Executes the request.
|
|
|
|
*
|
|
|
|
* @param action The action to execute.
|
|
|
|
* @param args The exec() arguments.
|
|
|
|
* @param callback The callback context used when
|
|
|
|
* calling back into JavaScript.
|
|
|
|
*
|
2017-01-16 09:17:25 +00:00
|
|
|
* @return Returning false results in a "MethodNotFound" error.
|
2014-10-26 15:47:18 +00:00
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public boolean execute (String action, JSONArray args,
|
2019-02-03 14:58:40 +00:00
|
|
|
CallbackContext callback)
|
|
|
|
{
|
|
|
|
boolean validAction = true;
|
|
|
|
|
|
|
|
switch (action)
|
|
|
|
{
|
|
|
|
case "configure":
|
|
|
|
configure(args.optJSONObject(0), args.optBoolean(1));
|
|
|
|
break;
|
|
|
|
case "enable":
|
|
|
|
enableMode();
|
|
|
|
break;
|
|
|
|
case "disable":
|
|
|
|
disableMode();
|
|
|
|
break;
|
|
|
|
case "optimizations":
|
|
|
|
case "background":
|
|
|
|
case "foreground":
|
|
|
|
case "tasklist":
|
|
|
|
case "dimmed":
|
|
|
|
case "wakeup":
|
|
|
|
case "unlock":
|
|
|
|
new BackgroundExt(this).executeAsync(action, callback);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
validAction = false;
|
2017-01-01 17:48:06 +00:00
|
|
|
}
|
2017-02-06 14:59:57 +00:00
|
|
|
|
2019-02-03 14:58:40 +00:00
|
|
|
if (validAction) {
|
2017-02-06 14:59:57 +00:00
|
|
|
callback.success();
|
2019-02-03 14:58:40 +00:00
|
|
|
} else {
|
|
|
|
callback.error("Invalid action: " + action);
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-02-03 14:58:40 +00:00
|
|
|
return validAction;
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when the system is about to start resuming a previous activity.
|
|
|
|
*
|
2017-01-16 09:17:25 +00:00
|
|
|
* @param multitasking Flag indicating if multitasking is turned on for app.
|
2014-10-26 15:47:18 +00:00
|
|
|
*/
|
|
|
|
@Override
|
2019-02-03 14:58:40 +00:00
|
|
|
public void onPause(boolean multitasking)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
inBackground = true;
|
|
|
|
startService();
|
|
|
|
} finally {
|
|
|
|
clearKeyguardFlags(cordova.getActivity());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when the activity is no longer visible to the user.
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public void onStop () {
|
|
|
|
clearKeyguardFlags(cordova.getActivity());
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when the activity will start interacting with the user.
|
|
|
|
*
|
2017-01-16 09:17:25 +00:00
|
|
|
* @param multitasking Flag indicating if multitasking is turned on for app.
|
2014-10-26 15:47:18 +00:00
|
|
|
*/
|
|
|
|
@Override
|
2019-02-03 14:58:40 +00:00
|
|
|
public void onResume (boolean multitasking)
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
inBackground = false;
|
|
|
|
stopService();
|
|
|
|
}
|
|
|
|
|
2014-11-12 21:28:25 +00:00
|
|
|
/**
|
|
|
|
* Called when the activity will be destroyed.
|
|
|
|
*/
|
2014-10-26 15:47:18 +00:00
|
|
|
@Override
|
2019-02-03 14:58:40 +00:00
|
|
|
public void onDestroy()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
stopService();
|
2018-01-09 19:51:50 +00:00
|
|
|
android.os.Process.killProcess(android.os.Process.myPid());
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enable the background mode.
|
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void enableMode()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
isDisabled = false;
|
|
|
|
|
|
|
|
if (inBackground) {
|
|
|
|
startService();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Disable the background mode.
|
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void disableMode()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
stopService();
|
|
|
|
isDisabled = true;
|
|
|
|
}
|
|
|
|
|
2017-01-01 17:48:06 +00:00
|
|
|
/**
|
|
|
|
* Update the default settings and configure the notification.
|
|
|
|
*
|
|
|
|
* @param settings The settings
|
|
|
|
* @param update A truthy value means to update the running service.
|
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void configure(JSONObject settings, boolean update)
|
|
|
|
{
|
2017-01-01 17:48:06 +00:00
|
|
|
if (update) {
|
|
|
|
updateNotification(settings);
|
|
|
|
} else {
|
|
|
|
setDefaultSettings(settings);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-04 16:59:33 +00:00
|
|
|
/**
|
2014-12-14 12:38:36 +00:00
|
|
|
* Update the default settings for the notification.
|
|
|
|
*
|
2017-01-01 17:48:06 +00:00
|
|
|
* @param settings The new default settings
|
2014-12-14 12:38:36 +00:00
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void setDefaultSettings(JSONObject settings)
|
|
|
|
{
|
2014-12-14 12:38:36 +00:00
|
|
|
defaultSettings = settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-02-03 14:58:40 +00:00
|
|
|
* Returns the settings for the new/updated notification.
|
2014-12-14 12:38:36 +00:00
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
static JSONObject getSettings () {
|
2014-12-14 12:38:36 +00:00
|
|
|
return defaultSettings;
|
|
|
|
}
|
|
|
|
|
2014-11-17 13:19:29 +00:00
|
|
|
/**
|
|
|
|
* Update the notification.
|
2016-08-17 09:52:31 +00:00
|
|
|
*
|
2017-01-01 17:48:06 +00:00
|
|
|
* @param settings The config settings
|
2014-11-17 13:19:29 +00:00
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void updateNotification(JSONObject settings)
|
|
|
|
{
|
2014-11-17 13:19:29 +00:00
|
|
|
if (isBind) {
|
2017-01-01 15:58:08 +00:00
|
|
|
service.updateNotification(settings);
|
2014-11-04 16:59:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
/**
|
|
|
|
* Bind the activity to a background service and put them into foreground
|
|
|
|
* state.
|
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void startService()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
Activity context = cordova.getActivity();
|
|
|
|
|
2014-12-14 12:38:36 +00:00
|
|
|
if (isDisabled || isBind)
|
2014-10-26 15:47:18 +00:00
|
|
|
return;
|
|
|
|
|
2017-02-06 14:59:57 +00:00
|
|
|
Intent intent = new Intent(context, ForegroundService.class);
|
2016-08-17 09:14:47 +00:00
|
|
|
|
2014-12-13 23:04:12 +00:00
|
|
|
try {
|
2017-02-06 14:59:57 +00:00
|
|
|
context.bindService(intent, connection, BIND_AUTO_CREATE);
|
2014-12-13 23:04:12 +00:00
|
|
|
fireEvent(Event.ACTIVATE, null);
|
2014-12-14 12:38:36 +00:00
|
|
|
context.startService(intent);
|
2014-12-13 23:04:12 +00:00
|
|
|
} catch (Exception e) {
|
2017-01-30 08:41:45 +00:00
|
|
|
fireEvent(Event.FAILURE, String.format("'%s'", e.getMessage()));
|
2014-12-13 23:04:12 +00:00
|
|
|
}
|
2014-11-07 18:55:24 +00:00
|
|
|
|
|
|
|
isBind = true;
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bind the activity to a background service and put them into foreground
|
|
|
|
* state.
|
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void stopService()
|
|
|
|
{
|
2014-10-26 15:47:18 +00:00
|
|
|
Activity context = cordova.getActivity();
|
2017-02-06 14:59:57 +00:00
|
|
|
Intent intent = new Intent(context, ForegroundService.class);
|
2014-10-26 15:47:18 +00:00
|
|
|
|
2019-02-03 14:58:40 +00:00
|
|
|
if (!isBind) return;
|
2014-11-07 18:55:24 +00:00
|
|
|
|
2014-12-14 12:38:36 +00:00
|
|
|
fireEvent(Event.DEACTIVATE, null);
|
|
|
|
context.unbindService(connection);
|
2014-10-26 15:47:18 +00:00
|
|
|
context.stopService(intent);
|
2014-11-07 18:55:24 +00:00
|
|
|
|
|
|
|
isBind = false;
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|
2014-12-13 23:04:12 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Fire vent with some parameters inside the web view.
|
|
|
|
*
|
2017-01-01 17:48:06 +00:00
|
|
|
* @param event The name of the event
|
|
|
|
* @param params Optional arguments for the event
|
2014-12-13 23:04:12 +00:00
|
|
|
*/
|
2019-02-03 14:58:40 +00:00
|
|
|
private void fireEvent (Event event, String params)
|
|
|
|
{
|
2017-02-10 19:52:42 +00:00
|
|
|
String eventName = event.name().toLowerCase();
|
|
|
|
Boolean active = event == Event.ACTIVATE;
|
2014-12-14 13:23:57 +00:00
|
|
|
|
2017-02-10 19:52:42 +00:00
|
|
|
String str = String.format("%s._setActive(%b)",
|
2014-12-14 13:23:57 +00:00
|
|
|
JS_NAMESPACE, active);
|
|
|
|
|
2017-02-10 19:52:42 +00:00
|
|
|
str = String.format("%s;%s.on%s(%s)",
|
|
|
|
str, JS_NAMESPACE, eventName, params);
|
2014-12-13 23:04:12 +00:00
|
|
|
|
2017-02-10 19:52:42 +00:00
|
|
|
str = String.format("%s;%s.fireEvent('%s',%s);",
|
|
|
|
str, JS_NAMESPACE, eventName, params);
|
2017-01-01 15:58:08 +00:00
|
|
|
|
2017-02-10 19:52:42 +00:00
|
|
|
final String js = str;
|
2014-12-14 13:23:57 +00:00
|
|
|
|
2019-02-03 14:58:40 +00:00
|
|
|
cordova.getActivity().runOnUiThread(() -> webView.loadUrl("javascript:" + js));
|
2014-12-13 23:04:12 +00:00
|
|
|
}
|
2017-01-01 17:48:06 +00:00
|
|
|
|
2014-10-26 15:47:18 +00:00
|
|
|
}
|