237 lines
9.1 KiB
Java
237 lines
9.1 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.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<Integer, Preference> 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 = 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<Integer, Preference> 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();
|
|
}
|
|
}
|