/* * 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.network; import static androidx.lifecycle.Lifecycle.Event.ON_PAUSE; import static androidx.lifecycle.Lifecycle.Event.ON_RESUME; import android.content.Context; import android.content.Intent; import android.provider.Settings; import android.telephony.SubscriptionInfo; import android.telephony.SubscriptionManager; import android.telephony.TelephonyManager; import androidx.collection.ArrayMap; import androidx.lifecycle.Lifecycle; import androidx.lifecycle.LifecycleObserver; import androidx.lifecycle.OnLifecycleEvent; import androidx.preference.Preference; import androidx.preference.PreferenceGroup; import androidx.preference.PreferenceScreen; import com.android.settings.R; import com.android.settings.network.telephony.MobileNetworkActivity; import com.android.settingslib.core.AbstractPreferenceController; import java.util.Map; /** * This manages a set of Preferences it places into a PreferenceGroup owned by some parent * controller class - one for each available subscription. This controller is only considered * available if there are 2 or more subscriptions. */ public class SubscriptionsPreferenceController extends AbstractPreferenceController implements LifecycleObserver, SubscriptionsChangeListener.SubscriptionsChangeListenerClient { private static final String TAG = "SubscriptionsPrefCntrlr"; private UpdateListener mUpdateListener; private String mPreferenceGroupKey; private PreferenceGroup mPreferenceGroup; private SubscriptionManager mManager; private SubscriptionsChangeListener mSubscriptionsListener; // Map of subscription id to Preference private Map mSubscriptionPreferences; private int mStartOrder; /** * This interface lets a parent of this class know that some change happened - this could * either be because overall availability changed, or because we've added/removed/updated some * preferences. */ public interface UpdateListener { void onChildrenUpdated(); } /** * @param context the context for the UI where we're placing these preferences * @param lifecycle for listening to lifecycle events for the UI * @param updateListener called to let our parent controller know that our availability has * changed, or that one or more of the preferences we've placed in the * PreferenceGroup has changed * @param preferenceGroupKey the key used to lookup the PreferenceGroup where Preferences will * be placed * @param startOrder the order that should be given to the first Preference placed into * the PreferenceGroup; the second will use startOrder+1, third will * use startOrder+2, etc. - this is useful for when the parent wants * to have other preferences in the same PreferenceGroup and wants * a specific ordering relative to this controller's prefs. */ public SubscriptionsPreferenceController(Context context, Lifecycle lifecycle, UpdateListener updateListener, String preferenceGroupKey, int startOrder) { super(context); mUpdateListener = updateListener; mPreferenceGroupKey = preferenceGroupKey; mStartOrder = startOrder; mManager = context.getSystemService(SubscriptionManager.class); mSubscriptionPreferences = new ArrayMap<>(); mSubscriptionsListener = new SubscriptionsChangeListener(context, this); lifecycle.addObserver(this); } @OnLifecycleEvent(ON_RESUME) public void onResume() { mSubscriptionsListener.start(); update(); } @OnLifecycleEvent(ON_PAUSE) public void onPause() { mSubscriptionsListener.stop(); } @Override public void displayPreference(PreferenceScreen screen) { mPreferenceGroup = (PreferenceGroup) screen.findPreference(mPreferenceGroupKey); update(); } private void update() { if (mPreferenceGroup == null) { return; } if (mSubscriptionsListener.isAirplaneModeOn()) { for (Preference pref : mSubscriptionPreferences.values()) { mPreferenceGroup.removePreference(pref); } mSubscriptionPreferences.clear(); mUpdateListener.onChildrenUpdated(); return; } final Map existingPrefs = mSubscriptionPreferences; mSubscriptionPreferences = new ArrayMap<>(); int order = mStartOrder; for (SubscriptionInfo info : SubscriptionUtil.getAvailableSubscriptions(mManager)) { final int subId = info.getSubscriptionId(); Preference pref = existingPrefs.remove(subId); if (pref == null) { pref = new Preference(mPreferenceGroup.getContext()); mPreferenceGroup.addPreference(pref); } pref.setTitle(info.getDisplayName()); pref.setSummary(getSummary(subId)); pref.setIcon(R.drawable.ic_network_cell); pref.setOrder(order++); pref.setOnPreferenceClickListener(clickedPref -> { final Intent intent = new Intent(mContext, MobileNetworkActivity.class); intent.putExtra(Settings.EXTRA_SUB_ID, subId); mContext.startActivity(intent); return true; }); mSubscriptionPreferences.put(subId, pref); } // Remove any old preferences that no longer map to a subscription. for (Preference pref : existingPrefs.values()) { mPreferenceGroup.removePreference(pref); } mUpdateListener.onChildrenUpdated(); } /** * The summary can have either 1 or 2 lines depending on which services (calls, SMS, data) this * subscription is the default for. * * If this subscription is the default for calls and/or SMS, we add a line to show that. * * If this subscription is the default for data, we add a line with detail about * whether the data connection is active. * * If a subscription isn't the default for anything, we just say it is available. */ protected String getSummary(int subId) { final int callsDefaultSubId = SubscriptionManager.getDefaultVoiceSubscriptionId(); final int smsDefaultSubId = SubscriptionManager.getDefaultSmsSubscriptionId(); final int dataDefaultSubId = SubscriptionManager.getDefaultDataSubscriptionId(); String line1 = null; if (subId == callsDefaultSubId && subId == smsDefaultSubId) { line1 = mContext.getString(R.string.default_for_calls_and_sms); } else if (subId == callsDefaultSubId) { line1 = mContext.getString(R.string.default_for_calls); } else if (subId == smsDefaultSubId) { line1 = mContext.getString(R.string.default_for_sms); } String line2 = null; if (subId == dataDefaultSubId) { final TelephonyManager telMgrForSub = mContext.getSystemService( TelephonyManager.class).createForSubscriptionId(subId); final int dataState = telMgrForSub.getDataState(); if (dataState == TelephonyManager.DATA_CONNECTED) { line2 = mContext.getString(R.string.mobile_data_active); } else if (!telMgrForSub.isDataEnabled()) { line2 = mContext.getString(R.string.mobile_data_off); } else { line2 = mContext.getString(R.string.default_for_mobile_data); } } if (line1 != null && line2 != null) { return String.join(System.lineSeparator(), line1, line2); } else if (line1 != null) { return line1; } else if (line2 != null) { return line2; } else { return mContext.getString(R.string.subscription_available); } } /** * * @return true if there are at least 2 available subscriptions. */ @Override public boolean isAvailable() { if (mSubscriptionsListener.isAirplaneModeOn()) { return false; } return SubscriptionUtil.getAvailableSubscriptions(mManager).size() >= 2; } @Override public String getPreferenceKey() { return null; } @Override public void onAirplaneModeChanged(boolean airplaneModeEnabled) { update(); } @Override public void onSubscriptionsChanged() { update(); } }