/* * Copyright (C) 2008 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.phone; import android.app.ActionBar; import android.app.Activity; import android.app.AlertDialog; import android.app.Dialog; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.IntentFilter; import android.content.pm.PackageManager; import android.content.pm.PackageManager.NameNotFoundException; import android.content.pm.ResolveInfo; import android.content.res.Resources; import android.os.Bundle; import android.os.Handler; import android.os.HandlerExecutor; import android.os.Looper; import android.os.PersistableBundle; import android.os.UserManager; import android.preference.Preference; import android.preference.PreferenceActivity; import android.preference.PreferenceScreen; import android.preference.SwitchPreference; import android.provider.Settings; import android.telecom.PhoneAccountHandle; import android.telecom.TelecomManager; import android.telephony.CarrierConfigManager; import android.telephony.SubscriptionManager; import android.telephony.TelephonyCallback; import android.telephony.TelephonyManager; import android.telephony.ims.ProvisioningManager; import android.telephony.ims.feature.ImsFeature; import android.util.Log; import android.view.MenuItem; import android.widget.Toast; import com.android.ims.ImsConfig; import com.android.ims.ImsException; import com.android.ims.ImsManager; import com.android.internal.telephony.Phone; import com.android.internal.telephony.PhoneConstants; import com.android.internal.telephony.flags.Flags; import com.android.phone.settings.PhoneAccountSettingsFragment; import com.android.phone.settings.SuppServicesUiUtil; import com.android.phone.settings.VoicemailSettingsActivity; import com.android.phone.settings.fdn.FdnSetting; import java.util.List; /** * Top level "Call settings" UI; see res/xml/call_feature_setting.xml * * This preference screen is the root of the "Call settings" hierarchy available from the Phone * app; the settings here let you control various features related to phone calls (including * voicemail settings, the "Respond via SMS" feature, and others.) It's used only on * voice-capable phone devices. * * Note that this activity is part of the package com.android.phone, even * though you reach it from the "Phone" app (i.e. DialtactsActivity) which * is from the package com.android.contacts. * * For the "Mobile network settings" screen under the main Settings app, * See {@link com.android.settings.network.telephony.MobileNetworkActivity}. */ public class CallFeaturesSetting extends PreferenceActivity implements Preference.OnPreferenceChangeListener { private static final String LOG_TAG = "CallFeaturesSetting"; private static final boolean DBG = (PhoneGlobals.DBG_LEVEL >= 2); // String keys for preference lookup // TODO: Naming these "BUTTON_*" is confusing since they're not actually buttons(!) // TODO: Consider moving these strings to strings.xml, so that they are not duplicated here and // in the layout files. These strings need to be treated carefully; if the setting is // persistent, they are used as the key to store shared preferences and the name should not be // changed unless the settings are also migrated. private static final String VOICEMAIL_SETTING_SCREEN_PREF_KEY = "button_voicemail_category_key"; private static final String BUTTON_FDN_KEY = "button_fdn_key"; private static final String BUTTON_RETRY_KEY = "button_auto_retry_key"; private static final String BUTTON_GSM_UMTS_OPTIONS = "button_gsm_more_expand_key"; private static final String BUTTON_CDMA_OPTIONS = "button_cdma_more_expand_key"; private static final String PHONE_ACCOUNT_SETTINGS_KEY = "phone_account_settings_preference_screen"; private static final String ENABLE_VIDEO_CALLING_KEY = "button_enable_video_calling"; private static final String BUTTON_VP_KEY = "button_voice_privacy_key"; private Phone mPhone; private ImsManager mImsMgr; private SubscriptionInfoHelper mSubscriptionInfoHelper; private TelecomManager mTelecomManager; private TelephonyCallback mTelephonyCallback; private SwitchPreference mButtonAutoRetry; private PreferenceScreen mVoicemailSettingsScreen; private SwitchPreference mEnableVideoCalling; private Preference mButtonWifiCalling; private boolean mDisallowedConfig = false; /* * Click Listeners, handle click based on objects attached to UI. */ private final BroadcastReceiver mReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { log("onReceive: " + intent.getAction()); if (Intent.ACTION_AIRPLANE_MODE_CHANGED.equals(intent.getAction())) { log("ACTION_AIRPLANE_MODE_CHANGED"); boolean isAirplaneModeOn = intent.getBooleanExtra("state", false); handleAirplaneModeChange(isAirplaneModeOn); } } }; private void handleAirplaneModeChange(boolean isAirplaneModeOn) { PersistableBundle b = null; if (mSubscriptionInfoHelper.hasSubId()) { b = PhoneGlobals.getInstance().getCarrierConfigForSubId( mSubscriptionInfoHelper.getSubId()); } else { b = PhoneGlobals.getInstance().getCarrierConfig(); } if (b != null && b.getBoolean( CarrierConfigManager.KEY_DISABLE_SUPPLEMENTARY_SERVICES_IN_AIRPLANE_MODE_BOOL)) { PreferenceScreen preferenceScreen = getPreferenceScreen(); Preference callForwarding = preferenceScreen.findPreference( GsmUmtsCallOptions.CALL_FORWARDING_KEY); Preference callBarring = preferenceScreen.findPreference( GsmUmtsCallOptions.CALL_BARRING_KEY); Preference additional = preferenceScreen.findPreference( GsmUmtsCallOptions.ADDITIONAL_GSM_SETTINGS_KEY); if (callForwarding != null) { callForwarding.setEnabled(!isAirplaneModeOn); } if (callBarring != null) { callBarring.setEnabled(!isAirplaneModeOn); } if (additional != null) { additional.setEnabled(!isAirplaneModeOn); } } } // Click listener for all toggle events @Override public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference) { if (preference == mButtonAutoRetry) { android.provider.Settings.Global.putInt(mPhone.getContext().getContentResolver(), android.provider.Settings.Global.CALL_AUTO_RETRY, mButtonAutoRetry.isChecked() ? 1 : 0); return true; } else if (preference == preferenceScreen.findPreference( GsmUmtsCallOptions.CALL_FORWARDING_KEY)) { return doSsOverUtPrecautions(preference); } else if (preference == preferenceScreen.findPreference( GsmUmtsCallOptions.CALL_BARRING_KEY)) { return doSsOverUtPrecautions(preference); } return false; } private boolean doSsOverUtPrecautions(Preference preference) { PersistableBundle b = null; if (mSubscriptionInfoHelper.hasSubId()) { b = PhoneGlobals.getInstance().getCarrierConfigForSubId( mSubscriptionInfoHelper.getSubId()); } else { b = PhoneGlobals.getInstance().getCarrierConfig(); } String configKey; if (preference.getKey().equals(GsmUmtsCallOptions.CALL_FORWARDING_KEY)) { configKey = CarrierConfigManager.KEY_CALL_FORWARDING_OVER_UT_WARNING_BOOL; } else { configKey = CarrierConfigManager.KEY_CALL_BARRING_OVER_UT_WARNING_BOOL; } if (b != null && b.getBoolean(configKey) && mPhone != null && SuppServicesUiUtil.isSsOverUtPrecautions(this, mPhone)) { SuppServicesUiUtil.showBlockingSuppServicesDialog(this, mPhone, preference.getKey()).show(); return true; } return false; } /** * Implemented to support onPreferenceChangeListener to look for preference * changes. * * @param preference is the preference to be changed * @param objValue should be the value of the selection, NOT its localized * display value. */ @Override public boolean onPreferenceChange(Preference preference, Object objValue) { if (DBG) log("onPreferenceChange: \"" + preference + "\" changed to \"" + objValue + "\""); if (preference == mEnableVideoCalling) { if (mImsMgr.isEnhanced4gLteModeSettingEnabledByUser()) { mImsMgr.setVtSetting((boolean) objValue); } else { AlertDialog.Builder builder = FrameworksUtils.makeAlertDialogBuilder(this); DialogInterface.OnClickListener networkSettingsClickListener = new Dialog.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { Intent intent = new Intent(); ComponentName mobileNetworkSettingsComponent = new ComponentName( getString(R.string.mobile_network_settings_package), getString(R.string.mobile_network_settings_class)); intent.setComponent(mobileNetworkSettingsComponent); startActivity(intent); } }; builder.setMessage(getResourcesForSubId().getString( R.string.enable_video_calling_dialog_msg)) .setNeutralButton(getResourcesForSubId().getString( R.string.enable_video_calling_dialog_settings), networkSettingsClickListener) .setPositiveButton(android.R.string.ok, null) .show(); return false; } } // Always let the preference setting proceed. return true; } @Override protected void onCreate(Bundle icicle) { super.onCreate(icicle); if (DBG) log("onCreate: Intent is " + getIntent()); // Make sure we are running as an admin user. UserManager userManager = (UserManager) getSystemService(Context.USER_SERVICE); if (!userManager.isAdminUser()) { Toast.makeText(this, R.string.call_settings_admin_user_only, Toast.LENGTH_SHORT).show(); finish(); return; } // Check if mobile network configs are restricted. if (Flags.ensureAccessToCallSettingsIsRestricted() && userManager.hasUserRestriction(UserManager.DISALLOW_CONFIG_MOBILE_NETWORKS)) { mDisallowedConfig = true; Log.i(LOG_TAG, "Mobile network configs are restricted, disabling mobile network " + "settings"); } mSubscriptionInfoHelper = new SubscriptionInfoHelper(this, getIntent()); mPhone = mSubscriptionInfoHelper.getPhone(); mSubscriptionInfoHelper.setActionBarTitle( getActionBar(), getResourcesForSubId(), R.string.call_settings_with_label); mTelecomManager = getSystemService(TelecomManager.class); mTelephonyCallback = new CallFeaturesTelephonyCallback(); ActionBar actionBar = getActionBar(); if (actionBar != null) { // sets the talkback voice prompt to say "Back" instead of "Navigate Up" actionBar.setHomeActionContentDescription(R.string.ota_call_end); } } private void updateImsManager(Phone phone) { log("updateImsManager :: phone.getContext()=" + phone.getContext() + " phone.getPhoneId()=" + phone.getPhoneId()); mImsMgr = ImsManager.getInstance(phone.getContext(), phone.getPhoneId()); if (mImsMgr == null) { log("updateImsManager :: Could not get ImsManager instance!"); } else { log("updateImsManager :: mImsMgr=" + mImsMgr); } } private void listenPhoneState(boolean listen) { TelephonyManager telephonyManager = getSystemService(TelephonyManager.class) .createForSubscriptionId(mPhone.getSubId()); if (listen) { telephonyManager.registerTelephonyCallback( new HandlerExecutor(new Handler(Looper.getMainLooper())), mTelephonyCallback); } else { telephonyManager.unregisterTelephonyCallback(mTelephonyCallback); } } private final class CallFeaturesTelephonyCallback extends TelephonyCallback implements TelephonyCallback.CallStateListener { @Override public void onCallStateChanged(int state) { if (DBG) log("PhoneStateListener onCallStateChanged: state is " + state); boolean isCallStateIdle = state == TelephonyManager.CALL_STATE_IDLE; if (mEnableVideoCalling != null) { mEnableVideoCalling.setEnabled(isCallStateIdle); } if (mButtonWifiCalling != null) { mButtonWifiCalling.setEnabled(isCallStateIdle); } } } private final ProvisioningManager.Callback mProvisioningCallback = new ProvisioningManager.Callback() { @Override public void onProvisioningIntChanged(int item, int value) { if (item == ImsConfig.ConfigConstants.VOICE_OVER_WIFI_SETTING_ENABLED || item == ImsConfig.ConfigConstants.VLT_SETTING_ENABLED || item == ImsConfig.ConfigConstants.LVC_SETTING_ENABLED) { updateVtWfc(); } } }; @Override protected void onPause() { super.onPause(); listenPhoneState(false); unregisterReceiver(mReceiver); // Remove callback for provisioning changes. try { if (mImsMgr != null) { mImsMgr.getConfigInterface().removeConfigCallback( mProvisioningCallback.getBinder()); } } catch (ImsException e) { Log.w(LOG_TAG, "onPause: Unable to remove callback for provisioning changes"); } } @Override protected void onResume() { super.onResume(); updateImsManager(mPhone); listenPhoneState(true); PreferenceScreen preferenceScreen = getPreferenceScreen(); if (preferenceScreen != null) { preferenceScreen.removeAll(); } addPreferencesFromResource(R.xml.call_feature_setting); TelephonyManager telephonyManager = getSystemService(TelephonyManager.class) .createForSubscriptionId(mPhone.getSubId()); // Note: The PhoneAccountSettingsActivity accessible via the // android.telecom.action.CHANGE_PHONE_ACCOUNTS intent is accessible directly from // the AOSP Dialer settings page on multi-sim devices. // Where a device does NOT make the PhoneAccountSettingsActivity directly accessible from // its Dialer app, this check must be modified in the device's AOSP branch to ensure that // the PhoneAccountSettingsActivity is always accessible. if (telephonyManager.isMultiSimEnabled()) { Preference phoneAccountSettingsPreference = findPreference(PHONE_ACCOUNT_SETTINGS_KEY); getPreferenceScreen().removePreference(phoneAccountSettingsPreference); } PreferenceScreen prefSet = getPreferenceScreen(); mVoicemailSettingsScreen = (PreferenceScreen) findPreference(VOICEMAIL_SETTING_SCREEN_PREF_KEY); mVoicemailSettingsScreen.setIntent(mSubscriptionInfoHelper.getIntent( VoicemailSettingsActivity.class)); maybeHideVoicemailSettings(); mButtonAutoRetry = (SwitchPreference) findPreference(BUTTON_RETRY_KEY); mEnableVideoCalling = (SwitchPreference) findPreference(ENABLE_VIDEO_CALLING_KEY); mButtonWifiCalling = findPreference(getResourcesForSubId().getString( R.string.wifi_calling_settings_key)); PersistableBundle carrierConfig = PhoneGlobals.getInstance().getCarrierConfigForSubId(mPhone.getSubId()); if (carrierConfig.getBoolean(CarrierConfigManager.KEY_AUTO_RETRY_ENABLED_BOOL)) { mButtonAutoRetry.setOnPreferenceChangeListener(this); int autoretry = Settings.Global.getInt( getContentResolver(), Settings.Global.CALL_AUTO_RETRY, 0); mButtonAutoRetry.setChecked(autoretry != 0); } else { prefSet.removePreference(mButtonAutoRetry); mButtonAutoRetry = null; } Preference cdmaOptions = prefSet.findPreference(BUTTON_CDMA_OPTIONS); Preference gsmOptions = prefSet.findPreference(BUTTON_GSM_UMTS_OPTIONS); Preference fdnButton = prefSet.findPreference(BUTTON_FDN_KEY); fdnButton.setIntent(mSubscriptionInfoHelper.getIntent(FdnSetting.class)); if (carrierConfig.getBoolean(CarrierConfigManager.KEY_WORLD_PHONE_BOOL)) { cdmaOptions.setIntent(mSubscriptionInfoHelper.getIntent(CdmaCallOptions.class)); gsmOptions.setIntent(mSubscriptionInfoHelper.getIntent(GsmUmtsCallOptions.class)); } else { // Remove GSM options and repopulate the preferences in this Activity if phone type is // GSM. prefSet.removePreference(gsmOptions); int phoneType = mPhone.getPhoneType(); if (carrierConfig.getBoolean(CarrierConfigManager.KEY_HIDE_CARRIER_NETWORK_SETTINGS_BOOL)) { prefSet.removePreference(fdnButton); } else { if (phoneType == PhoneConstants.PHONE_TYPE_CDMA) { // For now, just keep CdmaCallOptions as one entity. Eventually CDMA should // follow the same pattern as GSM below, where VP and Call forwarding are // populated here and Call waiting is populated in another "Additional Settings" // submenu for CDMA. prefSet.removePreference(fdnButton); cdmaOptions.setSummary(null); cdmaOptions.setTitle(R.string.additional_gsm_call_settings); cdmaOptions.setIntent(mSubscriptionInfoHelper.getIntent(CdmaCallOptions.class)); } else if (phoneType == PhoneConstants.PHONE_TYPE_GSM) { prefSet.removePreference(cdmaOptions); if (mPhone.getIccCard() == null || !mPhone.getIccCard().getIccFdnAvailable()) { prefSet.removePreference(fdnButton); } if (carrierConfig.getBoolean( CarrierConfigManager.KEY_ADDITIONAL_CALL_SETTING_BOOL)) { addPreferencesFromResource(R.xml.gsm_umts_call_options); GsmUmtsCallOptions.init(prefSet, mSubscriptionInfoHelper); } } else { throw new IllegalStateException("Unexpected phone type: " + phoneType); } } } updateVtWfc(); // Register callback for provisioning changes. try { if (mImsMgr != null) { mImsMgr.getConfigInterface().addConfigCallback(mProvisioningCallback); } } catch (ImsException e) { Log.w(LOG_TAG, "onResume: Unable to register callback for provisioning changes."); } IntentFilter intentFilter = new IntentFilter(Intent.ACTION_AIRPLANE_MODE_CHANGED); registerReceiver(mReceiver, intentFilter); } private void updateVtWfc() { PreferenceScreen prefSet = getPreferenceScreen(); TelephonyManager telephonyManager = getSystemService(TelephonyManager.class) .createForSubscriptionId(mPhone.getSubId()); PersistableBundle carrierConfig = PhoneGlobals.getInstance().getCarrierConfigForSubId(mPhone.getSubId()); boolean useWfcHomeModeForRoaming = carrierConfig.getBoolean( CarrierConfigManager.KEY_USE_WFC_HOME_NETWORK_MODE_IN_ROAMING_NETWORK_BOOL, false); boolean isDataEnabled; isDataEnabled = mPhone.getDataSettingsManager().isDataEnabled(); if (mImsMgr.isVtEnabledByPlatform() && mImsMgr.isVtProvisionedOnDevice() && (carrierConfig.getBoolean( CarrierConfigManager.KEY_IGNORE_DATA_ENABLED_CHANGED_FOR_VIDEO_CALLS) || isDataEnabled) && !mDisallowedConfig) { boolean currentValue = mImsMgr.isEnhanced4gLteModeSettingEnabledByUser() ? mImsMgr.isVtEnabledByUser() : false; mEnableVideoCalling.setChecked(currentValue); mEnableVideoCalling.setOnPreferenceChangeListener(this); prefSet.addPreference(mEnableVideoCalling); } else { prefSet.removePreference(mEnableVideoCalling); } final PhoneAccountHandle simCallManager = mTelecomManager.getSimCallManagerForSubscription( mPhone.getSubId()); if (simCallManager != null) { Intent intent = PhoneAccountSettingsFragment.buildPhoneAccountConfigureIntent( this, simCallManager); if (intent != null) { PackageManager pm = mPhone.getContext().getPackageManager(); List resolutions = pm.queryIntentActivities(intent, 0); if (!resolutions.isEmpty()) { mButtonWifiCalling.setTitle(resolutions.get(0).loadLabel(pm)); mButtonWifiCalling.setSummary(null); mButtonWifiCalling.setIntent(intent); prefSet.addPreference(mButtonWifiCalling); } else { prefSet.removePreference(mButtonWifiCalling); } } else { prefSet.removePreference(mButtonWifiCalling); } } else if (!mImsMgr.isWfcEnabledByPlatform() || !mImsMgr.isWfcProvisionedOnDevice()) { prefSet.removePreference(mButtonWifiCalling); } else { String title = getResourcesForSubId().getString(R.string.wifi_calling); mButtonWifiCalling.setTitle(title); int resId = com.android.internal.R.string.wifi_calling_off_summary; if (mImsMgr.isWfcEnabledByUser()) { boolean isRoaming = telephonyManager.isNetworkRoaming(); // Also check carrier config for roaming mode int wfcMode = mImsMgr.getWfcMode(isRoaming && !useWfcHomeModeForRoaming); switch (wfcMode) { case ImsConfig.WfcModeFeatureValueConstants.WIFI_ONLY: resId = com.android.internal.R.string.wfc_mode_wifi_only_summary; break; case ImsConfig.WfcModeFeatureValueConstants.CELLULAR_PREFERRED: resId = com.android.internal.R.string.wfc_mode_cellular_preferred_summary; break; case ImsConfig.WfcModeFeatureValueConstants.WIFI_PREFERRED: resId = com.android.internal.R.string.wfc_mode_wifi_preferred_summary; break; default: if (DBG) log("Unexpected WFC mode value: " + wfcMode); } } mButtonWifiCalling.setSummary(getResourcesForSubId().getString(resId)); Intent intent = mButtonWifiCalling.getIntent(); if (intent != null) { intent.putExtra(Settings.EXTRA_SUB_ID, mPhone.getSubId()); } prefSet.addPreference(mButtonWifiCalling); } try { if (mImsMgr.getImsServiceState() != ImsFeature.STATE_READY) { log("Feature state not ready so remove vt and wfc settings for " + " phone =" + mPhone.getPhoneId()); prefSet.removePreference(mButtonWifiCalling); prefSet.removePreference(mEnableVideoCalling); } } catch (ImsException ex) { log("Exception when trying to get ImsServiceStatus: " + ex); prefSet.removePreference(mButtonWifiCalling); prefSet.removePreference(mEnableVideoCalling); } } /** * Hides the top level voicemail settings entry point if the default dialer contains a * particular manifest metadata key. This is required when the default dialer wants to display * its own version of voicemail settings. */ private void maybeHideVoicemailSettings() { String defaultDialer = getSystemService(TelecomManager.class).getDefaultDialerPackage(); if (defaultDialer == null) { return; } try { Bundle metadata = getPackageManager() .getApplicationInfo(defaultDialer, PackageManager.GET_META_DATA).metaData; if (metadata == null) { return; } if (!metadata .getBoolean(TelephonyManager.METADATA_HIDE_VOICEMAIL_SETTINGS_MENU, false)) { if (DBG) { log("maybeHideVoicemailSettings(): not disabled by default dialer"); } return; } getPreferenceScreen().removePreference(mVoicemailSettingsScreen); if (DBG) { log("maybeHideVoicemailSettings(): disabled by default dialer"); } } catch (NameNotFoundException e) { // do nothing if (DBG) { log("maybeHideVoicemailSettings(): not controlled by default dialer"); } } } @Override protected void onNewIntent(Intent newIntent) { setIntent(newIntent); mSubscriptionInfoHelper = new SubscriptionInfoHelper(this, getIntent()); mPhone = mSubscriptionInfoHelper.getPhone(); mSubscriptionInfoHelper.setActionBarTitle( getActionBar(), getResourcesForSubId(), R.string.call_settings_with_label); } private static void log(String msg) { Log.d(LOG_TAG, msg); } @Override public boolean onOptionsItemSelected(MenuItem item) { final int itemId = item.getItemId(); if (itemId == android.R.id.home) { // See ActionBar#setDisplayHomeAsUpEnabled() onBackPressed(); return true; } return super.onOptionsItemSelected(item); } /** * Finish current Activity and go up to the top level Settings ({@link CallFeaturesSetting}). * This is useful for implementing "HomeAsUp" capability for second-level Settings. */ public static void goUpToTopLevelSetting( Activity activity, SubscriptionInfoHelper subscriptionInfoHelper) { Intent intent = subscriptionInfoHelper.getIntent(CallFeaturesSetting.class); intent.setAction(Intent.ACTION_MAIN); intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); activity.startActivity(intent); activity.finish(); } private Resources getResourcesForSubId() { if (mPhone != null) { return SubscriptionManager.getResourcesForSubId(mPhone.getContext(), mPhone.getSubId()); } else { return getResources(); } } }