- For devices that have opted in with overlay config, enabling a 2nd SIM and modifying 'Switch mobile data automatically' will update cross-SIM calling enablement. - These updates will be logged as settings events. Bug: 260266794 Test: Live tests for cross-SIM calling across both subs. Change-Id: I64d25ca406877ca5f061064e0a3b23f54c125638
241 lines
8.9 KiB
Java
241 lines
8.9 KiB
Java
/*
|
|
* Copyright (C) 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.settings.network.telephony;
|
|
|
|
import static androidx.lifecycle.Lifecycle.Event.ON_PAUSE;
|
|
import static androidx.lifecycle.Lifecycle.Event.ON_RESUME;
|
|
|
|
import android.app.settings.SettingsEnums;
|
|
import android.content.Context;
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
import android.os.PersistableBundle;
|
|
import android.telephony.CarrierConfigManager;
|
|
import android.telephony.SubscriptionManager;
|
|
import android.telephony.TelephonyManager;
|
|
import android.telephony.ims.ImsException;
|
|
import android.telephony.ims.ImsManager;
|
|
import android.telephony.ims.ImsMmTelManager;
|
|
import android.util.Log;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.lifecycle.LifecycleObserver;
|
|
import androidx.lifecycle.OnLifecycleEvent;
|
|
import androidx.preference.Preference;
|
|
import androidx.preference.PreferenceScreen;
|
|
import androidx.preference.SwitchPreference;
|
|
|
|
import com.android.internal.annotations.VisibleForTesting;
|
|
import com.android.settings.R;
|
|
import com.android.settings.datausage.DataUsageUtils;
|
|
import com.android.settings.network.MobileDataContentObserver;
|
|
import com.android.settings.network.ProxySubscriptionManager;
|
|
import com.android.settings.network.SubscriptionsChangeListener;
|
|
import com.android.settings.network.ims.WifiCallingQueryImsState;
|
|
import com.android.settings.overlay.FeatureFactory;
|
|
import com.android.settingslib.core.instrumentation.MetricsFeatureProvider;
|
|
|
|
/**
|
|
* Controls whether switch mobile data to the non-default SIM if the non-default SIM has better
|
|
* availability.
|
|
*
|
|
* This is used for temporarily allowing data on the non-default data SIM when on-default SIM
|
|
* has better availability on DSDS devices, where better availability means strong
|
|
* signal/connectivity.
|
|
* If this feature is enabled, data will be temporarily enabled on the non-default data SIM,
|
|
* including during any voice calls.
|
|
*/
|
|
public class AutoDataSwitchPreferenceController extends TelephonyTogglePreferenceController
|
|
implements LifecycleObserver,
|
|
SubscriptionsChangeListener.SubscriptionsChangeListenerClient {
|
|
private static final String LOG_TAG = "AutoDataSwitchPrefCtrl";
|
|
|
|
private SwitchPreference mPreference;
|
|
private SubscriptionsChangeListener mChangeListener;
|
|
private TelephonyManager mManager;
|
|
private MobileDataContentObserver mMobileDataContentObserver;
|
|
private PreferenceScreen mScreen;
|
|
|
|
private final MetricsFeatureProvider mMetricsFeatureProvider;
|
|
|
|
public AutoDataSwitchPreferenceController(Context context,
|
|
String preferenceKey) {
|
|
super(context, preferenceKey);
|
|
mMetricsFeatureProvider = FeatureFactory.getFactory(context).getMetricsFeatureProvider();
|
|
}
|
|
|
|
void init(int subId) {
|
|
this.mSubId = subId;
|
|
mManager = mContext.getSystemService(TelephonyManager.class).createForSubscriptionId(subId);
|
|
}
|
|
|
|
@OnLifecycleEvent(ON_RESUME)
|
|
public void onResume() {
|
|
if (mChangeListener == null) {
|
|
mChangeListener = new SubscriptionsChangeListener(mContext, this);
|
|
}
|
|
mChangeListener.start();
|
|
if (mMobileDataContentObserver == null) {
|
|
mMobileDataContentObserver = new MobileDataContentObserver(
|
|
new Handler(Looper.getMainLooper()));
|
|
mMobileDataContentObserver.setOnMobileDataChangedListener(() -> refreshPreference());
|
|
}
|
|
mMobileDataContentObserver.register(mContext, mSubId);
|
|
}
|
|
|
|
@OnLifecycleEvent(ON_PAUSE)
|
|
public void onPause() {
|
|
if (mChangeListener != null) {
|
|
mChangeListener.stop();
|
|
}
|
|
if (mMobileDataContentObserver != null) {
|
|
mMobileDataContentObserver.unRegister(mContext);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void displayPreference(PreferenceScreen screen) {
|
|
super.displayPreference(screen);
|
|
mPreference = screen.findPreference(getPreferenceKey());
|
|
mScreen = screen;
|
|
}
|
|
|
|
@Override
|
|
public boolean isChecked() {
|
|
return mManager != null && mManager.isMobileDataPolicyEnabled(
|
|
TelephonyManager.MOBILE_DATA_POLICY_AUTO_DATA_SWITCH);
|
|
}
|
|
|
|
private int getOtherSubId(@NonNull int[] subIds) {
|
|
if (subIds.length > 1) {
|
|
for (int subId : subIds) {
|
|
if (subId != mSubId) {
|
|
return subId;
|
|
}
|
|
}
|
|
}
|
|
return SubscriptionManager.INVALID_SUBSCRIPTION_ID;
|
|
}
|
|
|
|
private boolean isEnabled(int subId) {
|
|
if (subId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) {
|
|
return false;
|
|
}
|
|
TelephonyManager telephonyManager = mContext.getSystemService(
|
|
TelephonyManager.class).createForSubscriptionId(subId);
|
|
return telephonyManager != null && telephonyManager.isMobileDataPolicyEnabled(
|
|
TelephonyManager.MOBILE_DATA_POLICY_AUTO_DATA_SWITCH);
|
|
}
|
|
|
|
@Override
|
|
public boolean setChecked(boolean isChecked) {
|
|
mManager.setMobileDataPolicyEnabled(
|
|
TelephonyManager.MOBILE_DATA_POLICY_AUTO_DATA_SWITCH,
|
|
isChecked);
|
|
if (mContext.getResources().getBoolean(
|
|
R.bool.config_auto_data_switch_enables_cross_sim_calling)) {
|
|
trySetCrossSimCalling(mContext, getActiveSubscriptionIdList(), isChecked /* enabled */);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
protected boolean hasMobileData() {
|
|
return DataUsageUtils.hasMobileData(mContext);
|
|
}
|
|
|
|
private boolean isCrossSimCallingAllowedByPlatform(Context context, int subId) {
|
|
if ((new WifiCallingQueryImsState(context, subId)).isWifiCallingSupported()) {
|
|
PersistableBundle bundle = getCarrierConfigForSubId(subId);
|
|
return (bundle != null) && bundle.getBoolean(
|
|
CarrierConfigManager.KEY_CARRIER_CROSS_SIM_IMS_AVAILABLE_BOOL,
|
|
false /*default*/);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected ImsMmTelManager getImsMmTelManager(Context context, int subId) {
|
|
ImsManager imsMgr = context.getSystemService(ImsManager.class);
|
|
return (imsMgr == null) ? null : imsMgr.getImsMmTelManager(subId);
|
|
}
|
|
|
|
private void trySetCrossSimCallingPerSub(Context context, int subId, boolean enabled) {
|
|
try {
|
|
getImsMmTelManager(context, subId).setCrossSimCallingEnabled(enabled);
|
|
} catch (ImsException | IllegalArgumentException | NullPointerException exception) {
|
|
Log.w(LOG_TAG, "failed to change cross SIM calling configuration to " + enabled
|
|
+ " for subID " + subId + "with exception: ", exception);
|
|
}
|
|
}
|
|
|
|
private void trySetCrossSimCalling(Context context, int[] subIds, boolean enabled) {
|
|
mMetricsFeatureProvider.action(mContext,
|
|
SettingsEnums.ACTION_UPDATE_CROSS_SIM_CALLING_ON_AUTO_DATA_SWITCH_EVENT, enabled);
|
|
for (int subId : subIds) {
|
|
if (isCrossSimCallingAllowedByPlatform(context, subId)) {
|
|
trySetCrossSimCallingPerSub(context, subId, enabled);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public int getAvailabilityStatus(int subId) {
|
|
if (!SubscriptionManager.isValidSubscriptionId(subId)
|
|
|| SubscriptionManager.getDefaultDataSubscriptionId() == subId
|
|
|| (!hasMobileData())) {
|
|
return CONDITIONALLY_UNAVAILABLE;
|
|
}
|
|
return AVAILABLE;
|
|
}
|
|
|
|
@Override
|
|
public void updateState(Preference preference) {
|
|
super.updateState(preference);
|
|
if (preference == null) {
|
|
return;
|
|
}
|
|
preference.setVisible(isAvailable());
|
|
}
|
|
|
|
@Override
|
|
public void onAirplaneModeChanged(boolean airplaneModeEnabled) {}
|
|
|
|
@Override
|
|
public void onSubscriptionsChanged() {
|
|
updateState(mPreference);
|
|
}
|
|
|
|
private int[] getActiveSubscriptionIdList() {
|
|
return ProxySubscriptionManager.getInstance(mContext).getActiveSubscriptionIdList();
|
|
}
|
|
|
|
/**
|
|
* Trigger displaying preference when Mobile data content changed.
|
|
*/
|
|
@VisibleForTesting
|
|
public void refreshPreference() {
|
|
if (mContext.getResources().getBoolean(
|
|
R.bool.config_auto_data_switch_enables_cross_sim_calling)) {
|
|
int[] subIds = getActiveSubscriptionIdList();
|
|
trySetCrossSimCalling(mContext, subIds, isEnabled(getOtherSubId(subIds)));
|
|
}
|
|
if (mScreen != null) {
|
|
super.displayPreference(mScreen);
|
|
}
|
|
}
|
|
}
|