Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 60ff81fa authored by Evan Chen's avatar Evan Chen
Browse files

Remove all associaitons when forget a device in Settings

The associations will be removed one forget the device.

Test: manually test
Bug: 365613753
Flag: com.android.settings.flags.enable_remove_association_bt_unpair
Change-Id: Ic2224952b6f8e776ffcf07ce4fa6953a98475490
parent b865f8a6
Loading
Loading
Loading
Loading
+8 −0
Original line number Diff line number Diff line
@@ -54,3 +54,11 @@ flag {
      purpose: PURPOSE_BUGFIX
  }
}

flag {
  name: "enable_remove_association_bt_unpair"
  is_exported: true
  namespace: "companion_device_manager"
  description: "Allow to disassociate when to forget a BT pair device"
  bug: "365613753"
 }
+6 −0
Original line number Diff line number Diff line
@@ -419,6 +419,12 @@
    <string name="bluetooth_unpair_dialog_body" product="tablet">Your tablet will no longer be paired with <xliff:g id="device_name">%1$s</xliff:g></string>
    <!--  Bluetooth device details. The body of a confirmation dialog for unpairing a paired device. -->
    <string name="bluetooth_unpair_dialog_body" product="device">Your device will no longer be paired with <xliff:g id="device_name">%1$s</xliff:g></string>
    <!-- Bluetooth device details. The body of a confirmation dialog for unpairing a paired device if there's any associations associated with this device [CHAR_LIMIT=NONE] -->
    <string name="bluetooth_unpair_dialog_with_associations_body" product="default">Your phone will no longer be paired with <xliff:g id="device_name">%1$s</xliff:g>.\u0020<xliff:g id="app_name">%2$s</xliff:g> will no longer manage the device</string>
    <!-- Bluetooth device details. The body of a confirmation dialog for unpairing a paired device if there's any associations associated with this device [CHAR_LIMIT=NONE] -->
    <string name="bluetooth_unpair_dialog_with_associations_body" product="tablet">Your tablet will no longer be paired with <xliff:g id="device_name">%1$s</xliff:g>.\u0020<xliff:g id="app_name">%2$s</xliff:g> will no longer manage the device</string>
    <!-- Bluetooth device details. The body of a confirmation dialog for unpairing a paired device if there's any associations associated with this device [CHAR_LIMIT=NONE] -->
    <string name="bluetooth_unpair_dialog_with_associations_body" product="device">Your device will no longer be paired with <xliff:g id="device_name">%1$s</xliff:g>.\u0020<xliff:g id="app_name">%2$s</xliff:g> will no longer manage the device</string>
    <!--  Virtual device details. The body of a confirmation dialog for unpairing a paired device. [CHAR LIMIT=none] -->
    <string name="virtual_device_forget_dialog_body" product="default"><xliff:g id="device_name">%1$s</xliff:g> will no longer be connected to this phone. If you continue, some apps and app streaming may stop working.</string>
    <!--  Virtual device details. The body of a confirmation dialog for unpairing a paired device. [CHAR LIMIT=none] -->
+94 −9
Original line number Diff line number Diff line
@@ -16,31 +16,52 @@

package com.android.settings.bluetooth;

import static com.android.internal.util.CollectionUtils.filter;

import android.app.Activity;
import android.app.Dialog;
import android.app.settings.SettingsEnums;
import android.bluetooth.BluetoothDevice;
import android.companion.AssociationInfo;
import android.companion.CompanionDeviceManager;
import android.content.Context;
import android.content.DialogInterface;
import android.content.pm.PackageManager;
import android.icu.text.ListFormatter;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.VisibleForTesting;
import androidx.appcompat.app.AlertDialog;

import com.android.settings.R;
import com.android.settings.core.instrumentation.InstrumentedDialogFragment;
import com.android.settings.flags.Flags;
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
import com.android.settingslib.bluetooth.LocalBluetoothManager;

import com.google.common.base.Objects;

import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Set;

/** Implements an AlertDialog for confirming that a user wishes to unpair or "forget" a paired
 *  device*/
public class ForgetDeviceDialogFragment extends InstrumentedDialogFragment {
    public static final String TAG = "ForgetBluetoothDevice";
    private static final String KEY_DEVICE_ADDRESS = "device_address";

    private CachedBluetoothDevice mDevice;

    @VisibleForTesting
    CachedBluetoothDevice mDevice;
    @VisibleForTesting
    CompanionDeviceManager mCompanionDeviceManager;
    @VisibleForTesting
    PackageManager mPackageManager;
    public static ForgetDeviceDialogFragment newInstance(String deviceAddress) {
        Bundle args = new Bundle(1);
        args.putString(KEY_DEVICE_ADDRESS, deviceAddress);
@@ -63,29 +84,93 @@ public class ForgetDeviceDialogFragment extends InstrumentedDialogFragment {
    }

    @Override
    public Dialog onCreateDialog(Bundle inState) {
        Context context = getContext();
    public void onAttach(@NonNull Context context) {
        super.onAttach(context);
        mCompanionDeviceManager = context.getSystemService(CompanionDeviceManager.class);
        mPackageManager = context.getPackageManager();
        mDevice = getDevice(context);
    }

    @NonNull
    @Override
    public Dialog onCreateDialog(@Nullable Bundle inState) {
        if (mDevice == null) {
            Log.e(TAG, "onCreateDialog: Device is null.");
            return null;
            throw new IllegalStateException("Device must not be null when creating dialog.");
        }
        List<AssociationInfo> associationInfos = getAssociations(mDevice.getAddress());
        Set<String> packageNames = new HashSet<>();
        if (Flags.enableRemoveAssociationBtUnpair()) {
            for (AssociationInfo ai : associationInfos) {
                CharSequence appLabel = getAppLabel(ai.getPackageName());
                if (!TextUtils.isEmpty(appLabel)) {
                    packageNames.add(appLabel.toString());
                }
            }
        }

        DialogInterface.OnClickListener onConfirm = (dialog, which) -> {
            // 1. Unpair the device.
            mDevice.unpair();
            // 2. Remove the associations if any.
            if (Flags.enableRemoveAssociationBtUnpair()) {
                for (AssociationInfo ai : associationInfos) {
                    mCompanionDeviceManager.disassociate(ai.getId());
                }
            }

            Activity activity = getActivity();
            if (activity != null) {
                activity.finish();
            }
        };
        AlertDialog dialog = new AlertDialog.Builder(context)

        AlertDialog dialog = new AlertDialog.Builder(getActivity())
                .setPositiveButton(R.string.bluetooth_unpair_dialog_forget_confirm_button,
                        onConfirm)
                .setNegativeButton(android.R.string.cancel, null)
                .create();

        dialog.setTitle(R.string.bluetooth_unpair_dialog_title);
        dialog.setMessage(context.getString(R.string.bluetooth_unpair_dialog_body,
                mDevice.getName()));
        String message = buildUnpairMessage(
                getActivity(), mDevice, associationInfos, packageNames.stream().toList());
        dialog.setMessage(message);

        return dialog;
    }

    private List<AssociationInfo> getAssociations(String address) {
        return filter(
                mCompanionDeviceManager.getAllAssociations(),
                a -> Objects.equal(address, a.getDeviceMacAddressAsString()));
    }

    private String buildUnpairMessage(Context context, CachedBluetoothDevice device,
            List<AssociationInfo> associationInfos, List<String> packageNames) {
        if (Flags.enableRemoveAssociationBtUnpair() && !associationInfos.isEmpty()) {
            String appNamesString = getAppNamesString(packageNames.stream().toList());
            return context.getString(R.string.bluetooth_unpair_dialog_with_associations_body,
                    device.getName(), appNamesString);
        } else {
            return context.getString(R.string.bluetooth_unpair_dialog_body, device.getName());
        }
    }

    private String getAppNamesString(List<String> appNames) {
        if (appNames == null || appNames.isEmpty()) {
            return "";
        }

        ListFormatter formatter = ListFormatter.getInstance(Locale.getDefault());
        return formatter.format(appNames);
    }

    private CharSequence getAppLabel(String packageName) {
        try {
            return mPackageManager.getApplicationLabel(
                    mPackageManager.getApplicationInfo(packageName, 0));
        } catch (PackageManager.NameNotFoundException e) {
            Log.e(TAG, "Package Not Found", e);
            return "";
        }
    }
}
+122 −7
Original line number Diff line number Diff line
@@ -20,24 +20,36 @@ import static com.google.common.truth.Truth.assertThat;

import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.never;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;

import android.bluetooth.BluetoothDevice;
import android.companion.AssociationInfo;
import android.companion.CompanionDeviceManager;
import android.content.Context;
import android.content.DialogInterface;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.net.MacAddress;
import android.os.Bundle;
import android.platform.test.annotations.RequiresFlagsEnabled;
import android.platform.test.flag.junit.CheckFlagsRule;
import android.platform.test.flag.junit.DeviceFlagsValueProvider;

import androidx.appcompat.app.AlertDialog;
import androidx.fragment.app.FragmentActivity;

import com.android.settings.R;
import com.android.settings.testutils.FakeFeatureFactory;
import com.android.settings.flags.Flags;
import com.android.settings.testutils.shadow.ShadowAlertDialogCompat;
import com.android.settingslib.bluetooth.CachedBluetoothDevice;

import org.junit.Before;
import org.junit.Ignore;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Answers;
@@ -48,41 +60,64 @@ import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowDialog;
import org.robolectric.shadows.ShadowLooper;
import org.robolectric.shadows.androidx.fragment.FragmentController;

@Ignore
import java.util.ArrayList;
import java.util.List;

@RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowAlertDialogCompat.class})
@Config(shadows = {
        com.android.settings.testutils.shadow.ShadowFragment.class,
        ShadowAlertDialogCompat.class,
})
public class ForgetDeviceDialogFragmentTest {

    private static final String DEVICE_NAME = "Nightshade";
    private static final String PACKAGE_NAME = "com.android.test";
    private static final CharSequence APP_NAME = "test";

    @Mock(answer = Answers.RETURNS_DEEP_STUBS)
    private CachedBluetoothDevice mCachedDevice;
    @Mock
    private BluetoothDevice mBluetoothDevice;
    @Mock
    private CompanionDeviceManager mCompanionDeviceManager;
    @Mock
    private PackageManager mPackageManager;

    private ForgetDeviceDialogFragment mFragment;
    private FragmentActivity mActivity;
    private AlertDialog mDialog;
    private Context mContext;
    private List<AssociationInfo> mAssociations;

    @Rule
    public final CheckFlagsRule mCheckFlagsRule = DeviceFlagsValueProvider.createCheckFlagsRule();

    @Before
    public void setUp() {
        MockitoAnnotations.initMocks(this);

        mContext = RuntimeEnvironment.application;
        FakeFeatureFactory.setupForTest();
        String deviceAddress = "55:66:77:88:99:AA";
        mAssociations = new ArrayList<>();
        mFragment = spy(ForgetDeviceDialogFragment.newInstance(deviceAddress));
        mContext = spy(RuntimeEnvironment.application);
        mFragment.mCompanionDeviceManager = mCompanionDeviceManager;
        mFragment.mPackageManager = mPackageManager;
        mFragment.mDevice = mCachedDevice;
        mActivity = Robolectric.setupActivity(FragmentActivity.class);

        when(mFragment.getActivity()).thenReturn(mActivity);
        when(mCachedDevice.getAddress()).thenReturn(deviceAddress);
        when(mCachedDevice.getIdentityAddress()).thenReturn(deviceAddress);
        when(mCachedDevice.getDevice()).thenReturn(mBluetoothDevice);
        when(mCachedDevice.getName()).thenReturn(DEVICE_NAME);
        mFragment = spy(ForgetDeviceDialogFragment.newInstance(deviceAddress));
        when(mCompanionDeviceManager.getAllAssociations()).thenReturn(mAssociations);
        doReturn(mCachedDevice).when(mFragment).getDevice(any());
        mActivity = Robolectric.setupActivity(FragmentActivity.class);
    }

    @Ignore("b/253386225")
    @Test
    public void cancelDialog() {
        initDialog();
@@ -92,6 +127,7 @@ public class ForgetDeviceDialogFragmentTest {
        assertThat(mActivity.isFinishing()).isFalse();
    }

    @Ignore("b/253386225")
    @Test
    public void confirmDialog() {
        initDialog();
@@ -101,6 +137,7 @@ public class ForgetDeviceDialogFragmentTest {
        assertThat(mActivity.isFinishing()).isTrue();
    }

    @Ignore("b/253386225")
    @Test
    public void createDialog_normalDevice_showNormalMessage() {
        when(mBluetoothDevice.getMetadata(BluetoothDevice.METADATA_IS_UNTETHERED_HEADSET))
@@ -115,8 +152,86 @@ public class ForgetDeviceDialogFragmentTest {
                mContext.getString(R.string.bluetooth_unpair_dialog_body, DEVICE_NAME));
    }

    @Test
    @RequiresFlagsEnabled(Flags.FLAG_ENABLE_REMOVE_ASSOCIATION_BT_UNPAIR)
    public void cancelDialog_with_association() {
        addAssociation();
        final AlertDialog dialog = (AlertDialog) mFragment.onCreateDialog(Bundle.EMPTY);
        dialog.show();
        dialog.getButton(DialogInterface.BUTTON_NEGATIVE).performClick();
        ShadowLooper.idleMainLooper();

        verify(mCachedDevice, never()).unpair();
        verify(mCompanionDeviceManager, never()).disassociate(1);
        assertThat(mActivity.isFinishing()).isFalse();
    }

    @Test
    @RequiresFlagsEnabled(Flags.FLAG_ENABLE_REMOVE_ASSOCIATION_BT_UNPAIR)
    public void confirmDialog_with_association() {
        addAssociation();
        final AlertDialog dialog = (AlertDialog) mFragment.onCreateDialog(Bundle.EMPTY);
        dialog.show();
        dialog.getButton(DialogInterface.BUTTON_POSITIVE).performClick();
        ShadowLooper.idleMainLooper();

        verify(mCachedDevice).unpair();
        verify(mCompanionDeviceManager).disassociate(1);

        assertThat(mActivity.isFinishing()).isTrue();
    }

    @Test
    @RequiresFlagsEnabled(Flags.FLAG_ENABLE_REMOVE_ASSOCIATION_BT_UNPAIR)
    public void createDialog_showMessage_with_association() {
        addAssociation();
        final AlertDialog dialog = (AlertDialog) mFragment.onCreateDialog(Bundle.EMPTY);
        dialog.show();
        ShadowLooper.idleMainLooper();

        ShadowAlertDialogCompat shadowDialog = ShadowAlertDialogCompat.shadowOf(dialog);
        assertThat(shadowDialog.getMessage().toString()).isEqualTo(
                mContext.getString(
                        R.string.bluetooth_unpair_dialog_with_associations_body,
                        DEVICE_NAME, APP_NAME)
        );
    }

    private void initDialog() {
        mActivity.getSupportFragmentManager().beginTransaction().add(mFragment, null).commit();
        mDialog = (AlertDialog) ShadowDialog.getLatestDialog();
    }

    private void addAssociation() {
        setupLabelAndInfo(PACKAGE_NAME, APP_NAME);
        final AssociationInfo association = new AssociationInfo(
                1,
                /* userId */ 0,
                PACKAGE_NAME,
                MacAddress.fromString(mCachedDevice.getAddress()),
                /* displayName */ null,
                /* deviceProfile */ "",
                /* associatedDevice */ null,
                /* selfManaged */ false,
                /* notifyOnDeviceNearby */ true,
                /* revoked */ false,
                /* pending */ false,
                /* timeApprovedMs */ System.currentTimeMillis(),
                /* lastTimeConnected */ Long.MAX_VALUE,
                /* systemDataSyncFlags */ -1,
                /* deviceIcon */ null,
                /* deviceId */ null);

        mAssociations.add(association);
    }

    private void setupLabelAndInfo(String packageName, CharSequence appName) {
        ApplicationInfo appInfo = mock(ApplicationInfo.class);
        try {
            when(mPackageManager.getApplicationInfo(packageName, 0)).thenReturn(appInfo);
            when(mPackageManager.getApplicationLabel(appInfo)).thenReturn(appName);
        } catch (PackageManager.NameNotFoundException e) {
            throw new RuntimeException(e);
        }
    }
}