Merge "Add a confirmation dialog when forgetting BT device" into oc-dr1-dev am: 21423623bd
am: c608cfcceb
Change-Id: Ia3a4a17dd0c253b10be2e1bcfbad4609d0ed1d88
This commit is contained in:
@@ -43,14 +43,19 @@ public class BluetoothDetailsButtonsController extends BluetoothDetailsControlle
|
|||||||
mIsConnected = device.isConnected();
|
mIsConnected = device.isConnected();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void onForgetButtonPressed() {
|
||||||
|
ForgetDeviceDialogFragment fragment =
|
||||||
|
ForgetDeviceDialogFragment.newInstance(mCachedDevice.getAddress());
|
||||||
|
fragment.show(mFragment.getFragmentManager(), ForgetDeviceDialogFragment.TAG);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void init(PreferenceScreen screen) {
|
protected void init(PreferenceScreen screen) {
|
||||||
mActionButtons = (LayoutPreference) screen.findPreference(getPreferenceKey());
|
mActionButtons = (LayoutPreference) screen.findPreference(getPreferenceKey());
|
||||||
Button rightButton = (Button) mActionButtons.findViewById(R.id.right_button);
|
Button rightButton = (Button) mActionButtons.findViewById(R.id.right_button);
|
||||||
rightButton.setText(R.string.forget);
|
rightButton.setText(R.string.forget);
|
||||||
rightButton.setOnClickListener((view) -> {
|
rightButton.setOnClickListener((view) -> {
|
||||||
mCachedDevice.unpair();
|
onForgetButtonPressed();
|
||||||
mFragment.getActivity().finish();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -0,0 +1,84 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2017 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 android.app.Activity;
|
||||||
|
import android.app.AlertDialog;
|
||||||
|
import android.app.Dialog;
|
||||||
|
import android.bluetooth.BluetoothDevice;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.DialogInterface;
|
||||||
|
import android.os.Bundle;
|
||||||
|
|
||||||
|
import com.android.internal.annotations.VisibleForTesting;
|
||||||
|
import com.android.internal.logging.nano.MetricsProto;
|
||||||
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.core.instrumentation.InstrumentedDialogFragment;
|
||||||
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
||||||
|
import com.android.settingslib.bluetooth.LocalBluetoothManager;
|
||||||
|
|
||||||
|
/** 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;
|
||||||
|
|
||||||
|
public static ForgetDeviceDialogFragment newInstance(String deviceAddress) {
|
||||||
|
Bundle args = new Bundle(1);
|
||||||
|
args.putString(KEY_DEVICE_ADDRESS, deviceAddress);
|
||||||
|
ForgetDeviceDialogFragment dialog = new ForgetDeviceDialogFragment();
|
||||||
|
dialog.setArguments(args);
|
||||||
|
return dialog;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
CachedBluetoothDevice getDevice(Context context) {
|
||||||
|
String deviceAddress = getArguments().getString(KEY_DEVICE_ADDRESS);
|
||||||
|
LocalBluetoothManager manager = Utils.getLocalBtManager(context);
|
||||||
|
BluetoothDevice device = manager.getBluetoothAdapter().getRemoteDevice(deviceAddress);
|
||||||
|
return manager.getCachedDeviceManager().findDevice(device);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getMetricsCategory() {
|
||||||
|
return MetricsProto.MetricsEvent.DIALOG_BLUETOOTH_PAIRED_DEVICE_FORGET;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dialog onCreateDialog(Bundle inState) {
|
||||||
|
DialogInterface.OnClickListener onConfirm = (dialog, which) -> {
|
||||||
|
mDevice.unpair();
|
||||||
|
Activity activity = getActivity();
|
||||||
|
if (activity != null) {
|
||||||
|
activity.finish();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Context context = getContext();
|
||||||
|
mDevice = getDevice(context);
|
||||||
|
AlertDialog dialog = new AlertDialog.Builder(context)
|
||||||
|
.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()));
|
||||||
|
return dialog;
|
||||||
|
}
|
||||||
|
}
|
@@ -18,25 +18,30 @@ package com.android.settings.bluetooth;
|
|||||||
|
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.mockito.Matchers.anyString;
|
||||||
import static org.mockito.Matchers.eq;
|
import static org.mockito.Matchers.eq;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import android.app.FragmentManager;
|
||||||
|
import android.app.FragmentTransaction;
|
||||||
import android.widget.Button;
|
import android.widget.Button;
|
||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
|
||||||
import com.android.settings.TestConfig;
|
import com.android.settings.TestConfig;
|
||||||
import com.android.settings.applications.LayoutPreference;
|
import com.android.settings.applications.LayoutPreference;
|
||||||
|
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||||
import com.android.settings.testutils.shadow.SettingsShadowBluetoothDevice;
|
import com.android.settings.testutils.shadow.SettingsShadowBluetoothDevice;
|
||||||
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.ArgumentCaptor;
|
||||||
import org.robolectric.annotation.Config;
|
import org.robolectric.annotation.Config;
|
||||||
|
|
||||||
@RunWith(SettingsRobolectricTestRunner.class)
|
@RunWith(SettingsRobolectricTestRunner.class)
|
||||||
@Config(manifest = TestConfig.MANIFEST_PATH, sdk = TestConfig.SDK_VERSION,
|
@Config(manifest = TestConfig.MANIFEST_PATH, sdk = TestConfig.SDK_VERSION,
|
||||||
shadows=SettingsShadowBluetoothDevice.class)
|
shadows = SettingsShadowBluetoothDevice.class)
|
||||||
public class BluetoothDetailsButtonsControllerTest extends BluetoothDetailsControllerTestBase {
|
public class BluetoothDetailsButtonsControllerTest extends BluetoothDetailsControllerTestBase {
|
||||||
private BluetoothDetailsButtonsController mController;
|
private BluetoothDetailsButtonsController mController;
|
||||||
private LayoutPreference mLayoutPreference;
|
private LayoutPreference mLayoutPreference;
|
||||||
@@ -125,13 +130,21 @@ public class BluetoothDetailsButtonsControllerTest extends BluetoothDetailsContr
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void forget() {
|
public void forgetDialog() {
|
||||||
showScreen(mController);
|
showScreen(mController);
|
||||||
|
FragmentManager fragmentManager = mock(FragmentManager.class);
|
||||||
|
when(mFragment.getFragmentManager()).thenReturn(fragmentManager);
|
||||||
|
FragmentTransaction ft = mock(FragmentTransaction.class);
|
||||||
|
when(fragmentManager.beginTransaction()).thenReturn(ft);
|
||||||
mRightButton.callOnClick();
|
mRightButton.callOnClick();
|
||||||
verify(mCachedDevice).unpair();
|
|
||||||
verify(mActivity).finish();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
ArgumentCaptor<ForgetDeviceDialogFragment> dialogCaptor =
|
||||||
|
ArgumentCaptor.forClass(ForgetDeviceDialogFragment.class);
|
||||||
|
verify(ft).add(dialogCaptor.capture(), anyString());
|
||||||
|
|
||||||
|
ForgetDeviceDialogFragment dialogFragment = dialogCaptor.getValue();
|
||||||
|
assertThat(dialogFragment).isNotNull();
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void startsOutBusy() {
|
public void startsOutBusy() {
|
||||||
|
@@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2017 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 com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.Mockito.doReturn;
|
||||||
|
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.app.Activity;
|
||||||
|
import android.app.AlertDialog;
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.android.settings.TestConfig;
|
||||||
|
import com.android.settings.testutils.FakeFeatureFactory;
|
||||||
|
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||||
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
||||||
|
import com.android.settingslib.bluetooth.LocalBluetoothManager;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.Answers;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.MockitoAnnotations;
|
||||||
|
import org.robolectric.Robolectric;
|
||||||
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
import org.robolectric.annotation.Config;
|
||||||
|
import org.robolectric.shadows.ShadowDialog;
|
||||||
|
|
||||||
|
@RunWith(SettingsRobolectricTestRunner.class)
|
||||||
|
@Config(manifest = TestConfig.MANIFEST_PATH, sdk = TestConfig.SDK_VERSION)
|
||||||
|
public class ForgetDeviceDialogFragmentTest {
|
||||||
|
|
||||||
|
@Mock(answer = Answers.RETURNS_DEEP_STUBS)
|
||||||
|
private CachedBluetoothDevice mCachedDevice;
|
||||||
|
|
||||||
|
private ForgetDeviceDialogFragment mFragment;
|
||||||
|
private Context mContext;
|
||||||
|
private Activity mActivity;
|
||||||
|
private AlertDialog mDialog;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
MockitoAnnotations.initMocks(this);
|
||||||
|
mContext = spy(RuntimeEnvironment.application);
|
||||||
|
FakeFeatureFactory.setupForTest(mContext);
|
||||||
|
String deviceAddress = "55:66:77:88:99:AA";
|
||||||
|
when(mCachedDevice.getAddress()).thenReturn(deviceAddress);
|
||||||
|
mFragment = spy(ForgetDeviceDialogFragment.newInstance(deviceAddress));
|
||||||
|
doReturn(mCachedDevice).when(mFragment).getDevice(any());
|
||||||
|
mActivity = Robolectric.setupActivity(Activity.class);
|
||||||
|
mActivity.getFragmentManager().beginTransaction().add(mFragment, null).commit();
|
||||||
|
mDialog = (AlertDialog) ShadowDialog.getLatestDialog();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void cancelDialog() {
|
||||||
|
mDialog.getButton(AlertDialog.BUTTON_NEGATIVE).performClick();
|
||||||
|
verify(mCachedDevice, never()).unpair();
|
||||||
|
assertThat(mActivity.isFinishing()).isFalse();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void confirmDialog() {
|
||||||
|
mDialog.getButton(AlertDialog.BUTTON_POSITIVE).performClick();
|
||||||
|
verify(mCachedDevice).unpair();
|
||||||
|
assertThat(mActivity.isFinishing()).isTrue();
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user