/* * Copyright (C) 2011 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.settings.bluetooth; import static android.os.Process.BLUETOOTH_UID; import android.app.settings.SettingsEnums; import android.bluetooth.BluetoothDevice; import android.bluetooth.BluetoothProfile; import android.content.Context; import android.content.DialogInterface; import android.content.pm.ActivityInfo; import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.content.pm.PackageManager.NameNotFoundException; import android.os.UserHandle; import android.provider.DeviceConfig; import android.provider.Settings; import android.text.TextUtils; import android.util.Log; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.annotation.VisibleForTesting; import androidx.appcompat.app.AlertDialog; import com.android.settings.R; import com.android.settings.core.SettingsUIDeviceConfig; import com.android.settings.overlay.FeatureFactory; import com.android.settingslib.bluetooth.BluetoothUtils; import com.android.settingslib.bluetooth.BluetoothUtils.ErrorListener; import com.android.settingslib.bluetooth.LocalBluetoothManager; import com.android.settingslib.bluetooth.LocalBluetoothManager.BluetoothManagerCallback; /** * Utils is a helper class that contains constants for various * Android resource IDs, debug logging flags, and static methods * for creating dialogs. */ public final class Utils { private static final String TAG = "BluetoothUtils"; static final boolean V = BluetoothUtils.V; // verbose logging static final boolean D = BluetoothUtils.D; // regular logging private Utils() { } public static int getConnectionStateSummary(int connectionState) { switch (connectionState) { case BluetoothProfile.STATE_CONNECTED: return R.string.bluetooth_connected; case BluetoothProfile.STATE_CONNECTING: return R.string.bluetooth_connecting; case BluetoothProfile.STATE_DISCONNECTED: return R.string.bluetooth_disconnected; case BluetoothProfile.STATE_DISCONNECTING: return R.string.bluetooth_disconnecting; default: return 0; } } // Create (or recycle existing) and show disconnect dialog. static AlertDialog showDisconnectDialog(Context context, AlertDialog dialog, DialogInterface.OnClickListener disconnectListener, CharSequence title, CharSequence message) { if (dialog == null) { dialog = new AlertDialog.Builder(context) .setPositiveButton(android.R.string.ok, disconnectListener) .setNegativeButton(android.R.string.cancel, null) .create(); } else { if (dialog.isShowing()) { dialog.dismiss(); } // use disconnectListener for the correct profile(s) CharSequence okText = context.getText(android.R.string.ok); dialog.setButton(DialogInterface.BUTTON_POSITIVE, okText, disconnectListener); } dialog.setTitle(title); dialog.setMessage(message); dialog.show(); return dialog; } @VisibleForTesting static void showConnectingError(Context context, String name, LocalBluetoothManager manager) { FeatureFactory.getFactory(context).getMetricsFeatureProvider().visible(context, SettingsEnums.PAGE_UNKNOWN, SettingsEnums.ACTION_SETTINGS_BLUETOOTH_CONNECT_ERROR, 0); showError(context, name, R.string.bluetooth_connecting_error_message, manager); } static void showError(Context context, String name, int messageResId) { showError(context, name, messageResId, getLocalBtManager(context)); } private static void showError(Context context, String name, int messageResId, LocalBluetoothManager manager) { String message = context.getString(messageResId, name); Context activity = manager.getForegroundActivity(); if (manager.isForegroundActivity()) { try { new AlertDialog.Builder(activity) .setTitle(R.string.bluetooth_error_title) .setMessage(message) .setPositiveButton(android.R.string.ok, null) .show(); } catch (Exception e) { Log.e(TAG, "Cannot show error dialog.", e); } } else { Toast.makeText(context, message, Toast.LENGTH_SHORT).show(); } } public static LocalBluetoothManager getLocalBtManager(Context context) { return LocalBluetoothManager.getInstance(context, mOnInitCallback); } public static String createRemoteName(Context context, BluetoothDevice device) { String mRemoteName = device != null ? device.getAlias() : null; if (mRemoteName == null) { mRemoteName = context.getString(R.string.unknown); } return mRemoteName; } private static final ErrorListener mErrorListener = new ErrorListener() { @Override public void onShowError(Context context, String name, int messageResId) { showError(context, name, messageResId); } }; private static final BluetoothManagerCallback mOnInitCallback = new BluetoothManagerCallback() { @Override public void onBluetoothManagerInitialized(Context appContext, LocalBluetoothManager bluetoothManager) { BluetoothUtils.setErrorListener(mErrorListener); } }; public static boolean isBluetoothScanningEnabled(Context context) { return Settings.Global.getInt(context.getContentResolver(), Settings.Global.BLE_SCAN_ALWAYS_AVAILABLE, 0) == 1; } /** * Check if the Bluetooth device supports advanced details header * * @param bluetoothDevice the BluetoothDevice to get metadata * @return true if it supports advanced details header, false otherwise. */ public static boolean isAdvancedDetailsHeader(@NonNull BluetoothDevice bluetoothDevice) { final boolean advancedEnabled = DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SETTINGS_UI, SettingsUIDeviceConfig.BT_ADVANCED_HEADER_ENABLED, true); if (!advancedEnabled) { Log.d(TAG, "isAdvancedDetailsHeader: advancedEnabled is false"); return false; } // The metadata is for Android R final boolean untetheredHeadset = BluetoothUtils.getBooleanMetaData(bluetoothDevice, BluetoothDevice.METADATA_IS_UNTETHERED_HEADSET); if (untetheredHeadset) { Log.d(TAG, "isAdvancedDetailsHeader: untetheredHeadset is true"); return true; } // The metadata is for Android S final String deviceType = BluetoothUtils.getStringMetaData(bluetoothDevice, BluetoothDevice.METADATA_DEVICE_TYPE); if (TextUtils.equals(deviceType, BluetoothDevice.DEVICE_TYPE_UNTETHERED_HEADSET) || TextUtils.equals(deviceType, BluetoothDevice.DEVICE_TYPE_WATCH) || TextUtils.equals(deviceType, BluetoothDevice.DEVICE_TYPE_DEFAULT)) { Log.d(TAG, "isAdvancedDetailsHeader: deviceType is " + deviceType); return true; } return false; } /** * Returns the Bluetooth Package name */ public static String findBluetoothPackageName(Context context) throws NameNotFoundException { // this activity will always be in the package where the rest of Bluetooth lives final String sentinelActivity = "com.android.bluetooth.opp.BluetoothOppLauncherActivity"; PackageManager packageManager = context.createContextAsUser(UserHandle.SYSTEM, 0) .getPackageManager(); String[] allPackages = packageManager.getPackagesForUid(BLUETOOTH_UID); String matchedPackage = null; for (String candidatePackage : allPackages) { PackageInfo packageInfo; try { packageInfo = packageManager.getPackageInfo( candidatePackage, PackageManager.GET_ACTIVITIES | PackageManager.MATCH_ANY_USER | PackageManager.MATCH_UNINSTALLED_PACKAGES | PackageManager.MATCH_DISABLED_COMPONENTS); } catch (NameNotFoundException e) { // rethrow throw e; } if (packageInfo.activities == null) { continue; } for (ActivityInfo activity : packageInfo.activities) { if (sentinelActivity.equals(activity.name)) { if (matchedPackage == null) { matchedPackage = candidatePackage; } else { throw new NameNotFoundException("multiple main bluetooth packages found"); } } } } if (matchedPackage != null) { return matchedPackage; } throw new NameNotFoundException("Could not find main bluetooth package"); } }