Before this CL, we only compare whether the value of function is
equal to accessory to ensure usb is in accessory mode. But in some
cases there are different accessory combinations, like
"accessory,audio source". It will make the condition return false
when encountering accessory combinations.
This CL will check whether the function will include accessory mode
to fix this issue.
Bug: 162451162
Test: make -j42 RunSettingsRoboTests
Change-Id: I7c80f02de5340799e292949608e19b86b187b982
Merged-In: I7c80f02de5340799e292949608e19b86b187b982
(cherry picked from commit 7d71081d45
)
185 lines
7.0 KiB
Java
185 lines
7.0 KiB
Java
/*
|
|
* Copyright (C) 2018 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.connecteddevice.usb;
|
|
|
|
import static android.hardware.usb.UsbPortStatus.DATA_ROLE_DEVICE;
|
|
import static android.net.ConnectivityManager.TETHERING_USB;
|
|
|
|
import android.content.Context;
|
|
import android.hardware.usb.UsbManager;
|
|
import android.net.ConnectivityManager;
|
|
import android.util.Log;
|
|
|
|
import androidx.annotation.VisibleForTesting;
|
|
import androidx.preference.PreferenceCategory;
|
|
import androidx.preference.PreferenceScreen;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.Utils;
|
|
import com.android.settingslib.widget.RadioButtonPreference;
|
|
|
|
import java.util.LinkedHashMap;
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* This class controls the radio buttons for choosing between different USB functions.
|
|
*/
|
|
public class UsbDetailsFunctionsController extends UsbDetailsController
|
|
implements RadioButtonPreference.OnClickListener {
|
|
|
|
private static final String TAG = "UsbFunctionsCtrl";
|
|
private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
|
|
|
|
static final Map<Long, Integer> FUNCTIONS_MAP = new LinkedHashMap<>();
|
|
|
|
static {
|
|
FUNCTIONS_MAP.put(UsbManager.FUNCTION_MTP, R.string.usb_use_file_transfers);
|
|
FUNCTIONS_MAP.put(UsbManager.FUNCTION_RNDIS, R.string.usb_use_tethering);
|
|
FUNCTIONS_MAP.put(UsbManager.FUNCTION_MIDI, R.string.usb_use_MIDI);
|
|
FUNCTIONS_MAP.put(UsbManager.FUNCTION_PTP, R.string.usb_use_photo_transfers);
|
|
FUNCTIONS_MAP.put(UsbManager.FUNCTION_NONE, R.string.usb_use_charging_only);
|
|
}
|
|
|
|
private PreferenceCategory mProfilesContainer;
|
|
private ConnectivityManager mConnectivityManager;
|
|
@VisibleForTesting
|
|
OnStartTetheringCallback mOnStartTetheringCallback;
|
|
@VisibleForTesting
|
|
long mPreviousFunction;
|
|
|
|
public UsbDetailsFunctionsController(Context context, UsbDetailsFragment fragment,
|
|
UsbBackend backend) {
|
|
super(context, fragment, backend);
|
|
mConnectivityManager = context.getSystemService(ConnectivityManager.class);
|
|
mOnStartTetheringCallback = new OnStartTetheringCallback();
|
|
mPreviousFunction = mUsbBackend.getCurrentFunctions();
|
|
}
|
|
|
|
@Override
|
|
public void displayPreference(PreferenceScreen screen) {
|
|
super.displayPreference(screen);
|
|
mProfilesContainer = screen.findPreference(getPreferenceKey());
|
|
}
|
|
|
|
/**
|
|
* Gets a switch preference for the particular option, creating it if needed.
|
|
*/
|
|
private RadioButtonPreference getProfilePreference(String key, int titleId) {
|
|
RadioButtonPreference pref = mProfilesContainer.findPreference(key);
|
|
if (pref == null) {
|
|
pref = new RadioButtonPreference(mProfilesContainer.getContext());
|
|
pref.setKey(key);
|
|
pref.setTitle(titleId);
|
|
pref.setOnClickListener(this);
|
|
mProfilesContainer.addPreference(pref);
|
|
}
|
|
return pref;
|
|
}
|
|
|
|
@Override
|
|
protected void refresh(boolean connected, long functions, int powerRole, int dataRole) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "refresh() connected : " + connected + ", functions : " + functions
|
|
+ ", powerRole : " + powerRole + ", dataRole : " + dataRole);
|
|
}
|
|
if (!connected || dataRole != DATA_ROLE_DEVICE) {
|
|
mProfilesContainer.setEnabled(false);
|
|
} else {
|
|
// Functions are only available in device mode
|
|
mProfilesContainer.setEnabled(true);
|
|
}
|
|
RadioButtonPreference pref;
|
|
for (long option : FUNCTIONS_MAP.keySet()) {
|
|
int title = FUNCTIONS_MAP.get(option);
|
|
pref = getProfilePreference(UsbBackend.usbFunctionsToString(option), title);
|
|
// Only show supported options
|
|
if (mUsbBackend.areFunctionsSupported(option)) {
|
|
if (isAccessoryMode(functions)) {
|
|
pref.setChecked(UsbManager.FUNCTION_MTP == option);
|
|
} else {
|
|
pref.setChecked(functions == option);
|
|
}
|
|
} else {
|
|
mProfilesContainer.removePreference(pref);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onRadioButtonClicked(RadioButtonPreference preference) {
|
|
final long function = UsbBackend.usbFunctionsFromString(preference.getKey());
|
|
final long previousFunction = mUsbBackend.getCurrentFunctions();
|
|
if (DEBUG) {
|
|
Log.d(TAG, "onRadioButtonClicked() function : " + function + ", toString() : "
|
|
+ UsbManager.usbFunctionsToString(function) + ", previousFunction : "
|
|
+ previousFunction + ", toString() : "
|
|
+ UsbManager.usbFunctionsToString(previousFunction));
|
|
}
|
|
if (function != previousFunction && !Utils.isMonkeyRunning()
|
|
&& !shouldIgnoreClickEvent(function, previousFunction)) {
|
|
mPreviousFunction = previousFunction;
|
|
|
|
//Update the UI in advance to make it looks smooth
|
|
final RadioButtonPreference prevPref =
|
|
(RadioButtonPreference) mProfilesContainer.findPreference(
|
|
UsbBackend.usbFunctionsToString(mPreviousFunction));
|
|
if (prevPref != null) {
|
|
prevPref.setChecked(false);
|
|
preference.setChecked(true);
|
|
}
|
|
|
|
if (function == UsbManager.FUNCTION_RNDIS) {
|
|
// We need to have entitlement check for usb tethering, so use API in
|
|
// ConnectivityManager.
|
|
mConnectivityManager.startTethering(TETHERING_USB, true /* showProvisioningUi */,
|
|
mOnStartTetheringCallback);
|
|
} else {
|
|
mUsbBackend.setCurrentFunctions(function);
|
|
}
|
|
}
|
|
}
|
|
|
|
private boolean shouldIgnoreClickEvent(long function, long previousFunction) {
|
|
return isAccessoryMode(previousFunction) && function == UsbManager.FUNCTION_MTP;
|
|
}
|
|
|
|
private boolean isAccessoryMode(long function) {
|
|
return (function & UsbManager.FUNCTION_ACCESSORY) != 0;
|
|
}
|
|
|
|
@Override
|
|
public boolean isAvailable() {
|
|
return !Utils.isMonkeyRunning();
|
|
}
|
|
|
|
@Override
|
|
public String getPreferenceKey() {
|
|
return "usb_details_functions";
|
|
}
|
|
|
|
@VisibleForTesting
|
|
final class OnStartTetheringCallback extends
|
|
ConnectivityManager.OnStartTetheringCallback {
|
|
|
|
@Override
|
|
public void onTetheringFailed() {
|
|
super.onTetheringFailed();
|
|
mUsbBackend.setCurrentFunctions(mPreviousFunction);
|
|
}
|
|
}
|
|
}
|