blob: cd8949bcf713154c2e305b91948ef50ecd3f3ae2 [file] [log] [blame]
/*
* Copyright 2019 The Android Open Source Project
*
* Licensed 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.
*/
package com.android.server.bluetooth;
import android.annotation.RequiresPermission;
import android.content.Context;
import android.content.pm.PackageManager;
import android.content.res.Resources;
import android.database.ContentObserver;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.provider.Settings;
import android.util.Log;
import com.android.internal.annotations.VisibleForTesting;
/**
* The BluetoothAirplaneModeListener handles system airplane mode change callback and checks
* whether we need to inform BluetoothManagerService on this change.
*
* The information of airplane mode turns on would not be passed to the BluetoothManagerService
* when Bluetooth is on and Bluetooth is in one of the following situations:
* 1. Bluetooth A2DP is connected.
* 2. Bluetooth Hearing Aid profile is connected.
* 3. Bluetooth LE Audio is connected
*/
public class BluetoothAirplaneModeListener {
private static final String TAG = "BluetoothAirplaneModeListener";
@VisibleForTesting static final String TOAST_COUNT = "bluetooth_airplane_toast_count";
// keeps track of whether wifi should remain on in airplane mode
public static final String WIFI_APM_STATE = "wifi_apm_state";
// keeps track of whether wifi and bt remains on notification was shown
public static final String APM_WIFI_BT_NOTIFICATION = "apm_wifi_bt_notification";
// keeps track of whether bt remains on notification was shown
public static final String APM_BT_NOTIFICATION = "apm_bt_notification";
// keeps track of whether airplane mode enhancement feature is enabled
public static final String APM_ENHANCEMENT = "apm_enhancement_enabled";
// keeps track of whether user changed bt state in airplane mode
public static final String APM_USER_TOGGLED_BLUETOOTH = "apm_user_toggled_bluetooth";
// keeps track of whether bt should remain on in airplane mode
public static final String BLUETOOTH_APM_STATE = "bluetooth_apm_state";
// keeps track of what the default value for bt should be in airplane mode
public static final String BT_DEFAULT_APM_STATE = "bt_default_apm_state";
// keeps track of whether user enabling bt notification was shown
public static final String APM_BT_ENABLED_NOTIFICATION = "apm_bt_enabled_notification";
private static final int MSG_AIRPLANE_MODE_CHANGED = 0;
public static final int NOTIFICATION_NOT_SHOWN = 0;
public static final int NOTIFICATION_SHOWN = 1;
public static final int UNUSED = 0;
public static final int USED = 1;
@VisibleForTesting static final int MAX_TOAST_COUNT = 10; // 10 times
private final BluetoothManagerService mBluetoothManager;
private final BluetoothAirplaneModeHandler mHandler;
private final Context mContext;
private BluetoothModeChangeHelper mAirplaneHelper;
private BluetoothNotificationManager mNotificationManager;
@VisibleForTesting int mToastCount = 0;
BluetoothAirplaneModeListener(BluetoothManagerService service, Looper looper, Context context,
BluetoothNotificationManager notificationManager) {
mBluetoothManager = service;
mNotificationManager = notificationManager;
mContext = context;
mHandler = new BluetoothAirplaneModeHandler(looper);
context.getContentResolver().registerContentObserver(
Settings.Global.getUriFor(Settings.Global.AIRPLANE_MODE_ON), true,
mAirplaneModeObserver);
}
private final ContentObserver mAirplaneModeObserver = new ContentObserver(null) {
@Override
public void onChange(boolean unused) {
// Post from system main thread to android_io thread.
Message msg = mHandler.obtainMessage(MSG_AIRPLANE_MODE_CHANGED);
mHandler.sendMessage(msg);
}
};
private class BluetoothAirplaneModeHandler extends Handler {
BluetoothAirplaneModeHandler(Looper looper) {
super(looper);
}
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case MSG_AIRPLANE_MODE_CHANGED:
handleAirplaneModeChange();
break;
default:
Log.e(TAG, "Invalid message: " + msg.what);
break;
}
}
}
/**
* Call after boot complete
*/
@VisibleForTesting
void start(BluetoothModeChangeHelper helper) {
Log.i(TAG, "start");
mAirplaneHelper = helper;
mToastCount = mAirplaneHelper.getSettingsInt(TOAST_COUNT);
}
@VisibleForTesting
boolean shouldPopToast() {
if (mToastCount >= MAX_TOAST_COUNT) {
return false;
}
mToastCount++;
mAirplaneHelper.setSettingsInt(TOAST_COUNT, mToastCount);
return true;
}
@VisibleForTesting
@RequiresPermission(android.Manifest.permission.BLUETOOTH_PRIVILEGED)
void handleAirplaneModeChange() {
if (shouldSkipAirplaneModeChange()) {
Log.i(TAG, "Ignore airplane mode change");
// Airplane mode enabled when Bluetooth is being used for audio/headering aid.
// Bluetooth is not disabled in such case, only state is changed to
// BLUETOOTH_ON_AIRPLANE mode.
mAirplaneHelper.setSettingsInt(Settings.Global.BLUETOOTH_ON,
BluetoothManagerService.BLUETOOTH_ON_AIRPLANE);
if (!isApmEnhancementEnabled() || !isBluetoothToggledOnApm()) {
if (shouldPopToast()) {
mAirplaneHelper.showToastMessage();
}
} else {
if (isWifiEnabledOnApm() && isFirstTimeNotification(APM_WIFI_BT_NOTIFICATION)) {
try {
sendApmNotification("bluetooth_and_wifi_stays_on_title",
"bluetooth_and_wifi_stays_on_message",
APM_WIFI_BT_NOTIFICATION);
} catch (Exception e) {
Log.e(TAG, "APM enhancement BT and Wi-Fi stays on notification not shown");
}
} else if (!isWifiEnabledOnApm() && isFirstTimeNotification(APM_BT_NOTIFICATION)) {
try {
sendApmNotification("bluetooth_stays_on_title",
"bluetooth_stays_on_message",
APM_BT_NOTIFICATION);
} catch (Exception e) {
Log.e(TAG, "APM enhancement BT stays on notification not shown");
}
}
}
return;
}
if (mAirplaneHelper != null) {
mAirplaneHelper.onAirplaneModeChanged(mBluetoothManager);
}
}
@VisibleForTesting
boolean shouldSkipAirplaneModeChange() {
if (mAirplaneHelper == null) {
return false;
}
boolean apmEnhancementUsed = isApmEnhancementEnabled() && isBluetoothToggledOnApm();
// APM feature disabled or user has not used the feature yet by changing BT state in APM
// BT will only remain on in APM when media profile is connected
if (!apmEnhancementUsed && mAirplaneHelper.isBluetoothOn()
&& mAirplaneHelper.isAirplaneModeOn()
&& mAirplaneHelper.isMediaProfileConnected()) {
return true;
}
// APM feature enabled and user has used the feature by changing BT state in APM
// BT will only remain on in APM based on user's last action in APM
if (apmEnhancementUsed && mAirplaneHelper.isBluetoothOn()
&& mAirplaneHelper.isAirplaneModeOn()
&& mAirplaneHelper.isBluetoothOnAPM()) {
return true;
}
// APM feature enabled and user has not used the feature yet by changing BT state in APM
// BT will only remain on in APM if the default value is set to on
if (isApmEnhancementEnabled() && !isBluetoothToggledOnApm()
&& mAirplaneHelper.isBluetoothOn()
&& mAirplaneHelper.isAirplaneModeOn()
&& mAirplaneHelper.isBluetoothOnAPM()) {
return true;
}
return false;
}
private boolean isApmEnhancementEnabled() {
return mAirplaneHelper.getSettingsInt(APM_ENHANCEMENT) == 1;
}
private boolean isBluetoothToggledOnApm() {
return mAirplaneHelper.getSettingsSecureInt(APM_USER_TOGGLED_BLUETOOTH, UNUSED) == USED;
}
private boolean isWifiEnabledOnApm() {
return mAirplaneHelper.getSettingsInt(Settings.Global.WIFI_ON) != 0
&& mAirplaneHelper.getSettingsSecureInt(WIFI_APM_STATE, 0) == 1;
}
private boolean isFirstTimeNotification(String name) {
return mAirplaneHelper.getSettingsSecureInt(
name, NOTIFICATION_NOT_SHOWN) == NOTIFICATION_NOT_SHOWN;
}
/**
* Helper method to send APM notification
*/
public void sendApmNotification(String titleId, String messageId, String notificationState)
throws PackageManager.NameNotFoundException {
String btPackageName = mAirplaneHelper.getBluetoothPackageName();
if (btPackageName == null) {
Log.e(TAG, "Unable to find Bluetooth package name with "
+ "APM notification resources");
return;
}
Resources resources = mContext.getPackageManager()
.getResourcesForApplication(btPackageName);
int title = resources.getIdentifier(titleId, "string", btPackageName);
int message = resources.getIdentifier(messageId, "string", btPackageName);
mNotificationManager.sendApmNotification(
resources.getString(title), resources.getString(message));
mAirplaneHelper.setSettingsSecureInt(notificationState,
NOTIFICATION_SHOWN);
}
}