/* * Copyright (C) 2016 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.datausage; import android.app.settings.SettingsEnums; import android.content.Context; import android.content.DialogInterface; import android.database.ContentObserver; import android.net.NetworkTemplate; import android.net.Uri; import android.os.Handler; import android.os.Looper; import android.os.Parcel; import android.os.Parcelable; import android.provider.Settings.Global; import android.telephony.SubscriptionInfo; import android.telephony.SubscriptionManager; import android.telephony.TelephonyManager; import android.util.AttributeSet; import android.util.Log; import android.view.View; import android.widget.Checkable; import androidx.annotation.VisibleForTesting; import androidx.appcompat.app.AlertDialog.Builder; import androidx.core.content.res.TypedArrayUtils; import androidx.preference.PreferenceViewHolder; import com.android.settings.R; import com.android.settings.Utils; import com.android.settings.overlay.FeatureFactory; import com.android.settingslib.CustomDialogPreferenceCompat; import java.util.List; public class CellDataPreference extends CustomDialogPreferenceCompat implements TemplatePreference { private static final String TAG = "CellDataPreference"; public int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID; public boolean mChecked; public boolean mMultiSimDialog; private TelephonyManager mTelephonyManager; @VisibleForTesting SubscriptionManager mSubscriptionManager; public CellDataPreference(Context context, AttributeSet attrs) { super(context, attrs, TypedArrayUtils.getAttr(context, androidx.preference.R.attr.switchPreferenceStyle, android.R.attr.switchPreferenceStyle)); } @Override protected void onRestoreInstanceState(Parcelable s) { CellDataState state = (CellDataState) s; super.onRestoreInstanceState(state.getSuperState()); mTelephonyManager = TelephonyManager.from(getContext()); mChecked = state.mChecked; mMultiSimDialog = state.mMultiSimDialog; if (mSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) { mSubId = state.mSubId; setKey(getKey() + mSubId); } notifyChanged(); } @Override protected Parcelable onSaveInstanceState() { CellDataState state = new CellDataState(super.onSaveInstanceState()); state.mChecked = mChecked; state.mMultiSimDialog = mMultiSimDialog; state.mSubId = mSubId; return state; } @Override public void onAttached() { super.onAttached(); mDataStateListener.setListener(true, mSubId, getContext()); if (mSubscriptionManager!= null) { mSubscriptionManager.addOnSubscriptionsChangedListener(mOnSubscriptionsChangeListener); } } @Override public void onDetached() { mDataStateListener.setListener(false, mSubId, getContext()); if (mSubscriptionManager!= null) { mSubscriptionManager.removeOnSubscriptionsChangedListener( mOnSubscriptionsChangeListener); } super.onDetached(); } @Override public void setTemplate(NetworkTemplate template, int subId, NetworkServices services) { if (subId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) { throw new IllegalArgumentException("CellDataPreference needs a SubscriptionInfo"); } mSubscriptionManager = SubscriptionManager.from(getContext()); mTelephonyManager = TelephonyManager.from(getContext()); mSubscriptionManager.addOnSubscriptionsChangedListener(mOnSubscriptionsChangeListener); if (mSubId == SubscriptionManager.INVALID_SUBSCRIPTION_ID) { mSubId = subId; setKey(getKey() + subId); } updateEnabled(); updateChecked(); } private void updateChecked() { setChecked(mTelephonyManager.getDataEnabled(mSubId)); } private void updateEnabled() { // If this subscription is not active, for example, SIM card is taken out, we disable // the button. setEnabled(mSubscriptionManager.getActiveSubscriptionInfo(mSubId) != null); } @Override protected void performClick(View view) { final Context context = getContext(); FeatureFactory.getFactory(context).getMetricsFeatureProvider() .action(context, SettingsEnums.ACTION_CELL_DATA_TOGGLE, !mChecked); final SubscriptionInfo currentSir = mSubscriptionManager.getActiveSubscriptionInfo( mSubId); final SubscriptionInfo nextSir = mSubscriptionManager.getDefaultDataSubscriptionInfo(); if (mChecked) { // If the device is single SIM or is enabling data on the active data SIM then forgo // the pop-up. if (!Utils.showSimCardTile(getContext()) || (nextSir != null && currentSir != null && currentSir.getSubscriptionId() == nextSir.getSubscriptionId())) { setMobileDataEnabled(false); if (nextSir != null && currentSir != null && currentSir.getSubscriptionId() == nextSir.getSubscriptionId()) { disableDataForOtherSubscriptions(mSubId); } return; } // disabling data; show confirmation dialog which eventually // calls setMobileDataEnabled() once user confirms. mMultiSimDialog = false; super.performClick(view); } else { // If we are showing the Sim Card tile then we are a Multi-Sim device. if (Utils.showSimCardTile(getContext())) { mMultiSimDialog = true; if (nextSir != null && currentSir != null && currentSir.getSubscriptionId() == nextSir.getSubscriptionId()) { setMobileDataEnabled(true); disableDataForOtherSubscriptions(mSubId); return; } super.performClick(view); } else { setMobileDataEnabled(true); } } } private void setMobileDataEnabled(boolean enabled) { if (DataUsageSummary.LOGD) Log.d(TAG, "setMobileDataEnabled(" + enabled + "," + mSubId + ")"); mTelephonyManager.setDataEnabled(mSubId, enabled); setChecked(enabled); } private void setChecked(boolean checked) { if (mChecked == checked) return; mChecked = checked; notifyChanged(); } @Override public void onBindViewHolder(PreferenceViewHolder holder) { super.onBindViewHolder(holder); View switchView = holder.findViewById(android.R.id.switch_widget); switchView.setClickable(false); ((Checkable) switchView).setChecked(mChecked); } @Override protected void onPrepareDialogBuilder(Builder builder, DialogInterface.OnClickListener listener) { if (mMultiSimDialog) { showMultiSimDialog(builder, listener); } else { showDisableDialog(builder, listener); } } private void showDisableDialog(Builder builder, DialogInterface.OnClickListener listener) { builder.setTitle(null) .setMessage(R.string.data_usage_disable_mobile) .setPositiveButton(android.R.string.ok, listener) .setNegativeButton(android.R.string.cancel, null); } private void showMultiSimDialog(Builder builder, DialogInterface.OnClickListener listener) { final SubscriptionInfo currentSir = mSubscriptionManager.getActiveSubscriptionInfo(mSubId); final SubscriptionInfo nextSir = mSubscriptionManager.getDefaultDataSubscriptionInfo(); final String previousName = (nextSir == null) ? getContext().getResources().getString(R.string.sim_selection_required_pref) : nextSir.getDisplayName().toString(); builder.setTitle(R.string.sim_change_data_title); builder.setMessage(getContext().getString(R.string.sim_change_data_message, String.valueOf(currentSir != null ? currentSir.getDisplayName() : null), previousName)); builder.setPositiveButton(R.string.okay, listener); builder.setNegativeButton(R.string.cancel, null); } private void disableDataForOtherSubscriptions(int subId) { List subInfoList = mSubscriptionManager .getActiveSubscriptionInfoList(true); if (subInfoList != null) { for (SubscriptionInfo subInfo : subInfoList) { if (subInfo.getSubscriptionId() != subId) { mTelephonyManager.setDataEnabled(subInfo.getSubscriptionId(), false); } } } } @Override protected void onClick(DialogInterface dialog, int which) { if (which != DialogInterface.BUTTON_POSITIVE) { return; } if (mMultiSimDialog) { mSubscriptionManager.setDefaultDataSubId(mSubId); setMobileDataEnabled(true); disableDataForOtherSubscriptions(mSubId); } else { // TODO: extend to modify policy enabled flag. setMobileDataEnabled(false); } } @VisibleForTesting final SubscriptionManager.OnSubscriptionsChangedListener mOnSubscriptionsChangeListener = new SubscriptionManager.OnSubscriptionsChangedListener() { @Override public void onSubscriptionsChanged() { if (DataUsageSummary.LOGD) { Log.d(TAG, "onSubscriptionsChanged"); } updateEnabled(); } }; private final DataStateListener mDataStateListener = new DataStateListener() { @Override public void onChange(boolean selfChange) { updateChecked(); } }; public abstract static class DataStateListener extends ContentObserver { public DataStateListener() { super(new Handler(Looper.getMainLooper())); } public void setListener(boolean listening, int subId, Context context) { if (listening) { Uri uri = Global.getUriFor(Global.MOBILE_DATA); if (TelephonyManager.getDefault().getSimCount() != 1) { uri = Global.getUriFor(Global.MOBILE_DATA + subId); } context.getContentResolver().registerContentObserver(uri, false, this); } else { context.getContentResolver().unregisterContentObserver(this); } } } public static class CellDataState extends BaseSavedState { public int mSubId; public boolean mChecked; public boolean mMultiSimDialog; public CellDataState(Parcelable base) { super(base); } public CellDataState(Parcel source) { super(source); mChecked = source.readByte() != 0; mMultiSimDialog = source.readByte() != 0; mSubId = source.readInt(); } @Override public void writeToParcel(Parcel dest, int flags) { super.writeToParcel(dest, flags); dest.writeByte((byte) (mChecked ? 1 : 0)); dest.writeByte((byte) (mMultiSimDialog ? 1 : 0)); dest.writeInt(mSubId); } public static final Creator CREATOR = new Creator() { @Override public CellDataState createFromParcel(Parcel source) { return new CellDataState(source); } @Override public CellDataState[] newArray(int size) { return new CellDataState[size]; } }; } }