Added similar test for WiFi to be consistent. Added null tests in BluetoothEnabler. Change-Id: Ia6e7b150a1bc060c7ce0b4db12ab3f6c958af104
567 lines
22 KiB
Java
567 lines
22 KiB
Java
/*
|
|
* 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.settings;
|
|
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.pm.ActivityInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.os.Bundle;
|
|
import android.preference.PreferenceActivity;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
import android.view.LayoutInflater;
|
|
import android.view.View;
|
|
import android.view.View.OnClickListener;
|
|
import android.view.ViewGroup;
|
|
import android.widget.ArrayAdapter;
|
|
import android.widget.Button;
|
|
import android.widget.ImageView;
|
|
import android.widget.ListAdapter;
|
|
import android.widget.Switch;
|
|
import android.widget.TextView;
|
|
|
|
import com.android.settings.bluetooth.BluetoothEnabler;
|
|
import com.android.settings.wifi.WifiEnabler;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
|
|
/**
|
|
* Top-level settings activity to handle single pane and double pane UI layout.
|
|
*/
|
|
public class Settings extends PreferenceActivity implements ButtonBarHandler {
|
|
|
|
private static final String LOG_TAG = "Settings";
|
|
private static final String META_DATA_KEY_HEADER_ID =
|
|
"com.android.settings.TOP_LEVEL_HEADER_ID";
|
|
private static final String META_DATA_KEY_FRAGMENT_CLASS =
|
|
"com.android.settings.FRAGMENT_CLASS";
|
|
private static final String META_DATA_KEY_PARENT_TITLE =
|
|
"com.android.settings.PARENT_FRAGMENT_TITLE";
|
|
private static final String META_DATA_KEY_PARENT_FRAGMENT_CLASS =
|
|
"com.android.settings.PARENT_FRAGMENT_CLASS";
|
|
|
|
private static final String SAVE_KEY_CURRENT_HEADER = "com.android.settings.CURRENT_HEADER";
|
|
private static final String SAVE_KEY_PARENT_HEADER = "com.android.settings.PARENT_HEADER";
|
|
|
|
private String mFragmentClass;
|
|
private int mTopLevelHeaderId;
|
|
private Header mFirstHeader;
|
|
private Header mCurrentHeader;
|
|
private Header mParentHeader;
|
|
private boolean mInLocalHeaderSwitch;
|
|
|
|
// TODO: Update Call Settings based on airplane mode state.
|
|
|
|
protected HashMap<Integer, Integer> mHeaderIndexMap = new HashMap<Integer, Integer>();
|
|
private List<Header> mHeaders;
|
|
|
|
@Override
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
getMetaData();
|
|
mInLocalHeaderSwitch = true;
|
|
super.onCreate(savedInstanceState);
|
|
mInLocalHeaderSwitch = false;
|
|
|
|
if (!onIsHidingHeaders() && onIsMultiPane()) {
|
|
highlightHeader();
|
|
// Force the title so that it doesn't get overridden by a direct launch of
|
|
// a specific settings screen.
|
|
setTitle(R.string.settings_label);
|
|
}
|
|
|
|
// Retrieve any saved state
|
|
if (savedInstanceState != null) {
|
|
mCurrentHeader = savedInstanceState.getParcelable(SAVE_KEY_CURRENT_HEADER);
|
|
mParentHeader = savedInstanceState.getParcelable(SAVE_KEY_PARENT_HEADER);
|
|
}
|
|
|
|
// If the current header was saved, switch to it
|
|
if (savedInstanceState != null && mCurrentHeader != null) {
|
|
//switchToHeaderLocal(mCurrentHeader);
|
|
showBreadCrumbs(mCurrentHeader.title, null);
|
|
}
|
|
|
|
if (mParentHeader != null) {
|
|
setParentTitle(mParentHeader.title, null, new OnClickListener() {
|
|
public void onClick(View v) {
|
|
switchToParent(mParentHeader.fragment);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onSaveInstanceState(Bundle outState) {
|
|
super.onSaveInstanceState(outState);
|
|
|
|
// Save the current fragment, if it is the same as originally launched
|
|
if (mCurrentHeader != null) {
|
|
outState.putParcelable(SAVE_KEY_CURRENT_HEADER, mCurrentHeader);
|
|
}
|
|
if (mParentHeader != null) {
|
|
outState.putParcelable(SAVE_KEY_PARENT_HEADER, mParentHeader);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
|
|
ListAdapter listAdapter = getListAdapter();
|
|
if (listAdapter instanceof HeaderAdapter) {
|
|
((HeaderAdapter) listAdapter).resume();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onPause() {
|
|
super.onPause();
|
|
|
|
ListAdapter listAdapter = getListAdapter();
|
|
if (listAdapter instanceof HeaderAdapter) {
|
|
((HeaderAdapter) listAdapter).pause();
|
|
}
|
|
}
|
|
|
|
private void switchToHeaderLocal(Header header) {
|
|
mInLocalHeaderSwitch = true;
|
|
switchToHeader(header);
|
|
mInLocalHeaderSwitch = false;
|
|
}
|
|
|
|
@Override
|
|
public void switchToHeader(Header header) {
|
|
if (!mInLocalHeaderSwitch) {
|
|
mCurrentHeader = null;
|
|
mParentHeader = null;
|
|
}
|
|
super.switchToHeader(header);
|
|
}
|
|
|
|
/**
|
|
* Switch to parent fragment and store the grand parent's info
|
|
* @param className name of the activity wrapper for the parent fragment.
|
|
*/
|
|
private void switchToParent(String className) {
|
|
final ComponentName cn = new ComponentName(this, className);
|
|
try {
|
|
final PackageManager pm = getPackageManager();
|
|
final ActivityInfo parentInfo = pm.getActivityInfo(cn, PackageManager.GET_META_DATA);
|
|
|
|
if (parentInfo != null && parentInfo.metaData != null) {
|
|
String fragmentClass = parentInfo.metaData.getString(META_DATA_KEY_FRAGMENT_CLASS);
|
|
CharSequence fragmentTitle = parentInfo.loadLabel(pm);
|
|
Header parentHeader = new Header();
|
|
parentHeader.fragment = fragmentClass;
|
|
parentHeader.title = fragmentTitle;
|
|
mCurrentHeader = parentHeader;
|
|
|
|
switchToHeaderLocal(parentHeader);
|
|
|
|
mParentHeader = new Header();
|
|
mParentHeader.fragment
|
|
= parentInfo.metaData.getString(META_DATA_KEY_PARENT_FRAGMENT_CLASS);
|
|
mParentHeader.title = parentInfo.metaData.getString(META_DATA_KEY_PARENT_TITLE);
|
|
}
|
|
} catch (NameNotFoundException nnfe) {
|
|
Log.w(LOG_TAG, "Could not find parent activity : " + className);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onNewIntent(Intent intent) {
|
|
super.onNewIntent(intent);
|
|
|
|
// If it is not launched from history, then reset to top-level
|
|
if ((intent.getFlags() & Intent.FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY) == 0
|
|
&& mFirstHeader != null) {
|
|
switchToHeaderLocal(mFirstHeader);
|
|
}
|
|
}
|
|
|
|
private void highlightHeader() {
|
|
if (mTopLevelHeaderId != 0) {
|
|
Integer index = mHeaderIndexMap.get(mTopLevelHeaderId);
|
|
if (index != null) {
|
|
getListView().setItemChecked(index, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public Intent getIntent() {
|
|
Intent superIntent = super.getIntent();
|
|
String startingFragment = getStartingFragmentClass(superIntent);
|
|
if (startingFragment != null && !onIsMultiPane()) {
|
|
Intent modIntent = new Intent(superIntent);
|
|
modIntent.putExtra(EXTRA_SHOW_FRAGMENT, startingFragment);
|
|
Bundle args = superIntent.getExtras();
|
|
if (args != null) {
|
|
args = new Bundle(args);
|
|
} else {
|
|
args = new Bundle();
|
|
}
|
|
args.putParcelable("intent", superIntent);
|
|
modIntent.putExtra(EXTRA_SHOW_FRAGMENT_ARGUMENTS, superIntent.getExtras());
|
|
return modIntent;
|
|
}
|
|
return superIntent;
|
|
}
|
|
|
|
/**
|
|
* Checks if the component name in the intent is different from the Settings class and
|
|
* returns the class name to load as a fragment.
|
|
*/
|
|
protected String getStartingFragmentClass(Intent intent) {
|
|
if (mFragmentClass != null) return mFragmentClass;
|
|
|
|
String intentClass = intent.getComponent().getClassName();
|
|
if (intentClass.equals(getClass().getName())) return null;
|
|
|
|
if ("com.android.settings.ManageApplications".equals(intentClass)
|
|
|| "com.android.settings.RunningServices".equals(intentClass)
|
|
|| "com.android.settings.applications.StorageUse".equals(intentClass)) {
|
|
// Old names of manage apps.
|
|
intentClass = com.android.settings.applications.ManageApplications.class.getName();
|
|
}
|
|
|
|
return intentClass;
|
|
}
|
|
|
|
/**
|
|
* Override initial header when an activity-alias is causing Settings to be launched
|
|
* for a specific fragment encoded in the android:name parameter.
|
|
*/
|
|
@Override
|
|
public Header onGetInitialHeader() {
|
|
String fragmentClass = getStartingFragmentClass(super.getIntent());
|
|
if (fragmentClass != null) {
|
|
Header header = new Header();
|
|
header.fragment = fragmentClass;
|
|
header.title = getTitle();
|
|
header.fragmentArguments = getIntent().getExtras();
|
|
mCurrentHeader = header;
|
|
return header;
|
|
}
|
|
|
|
// Find first non-category header
|
|
int position = 0;
|
|
while (position < mHeaders.size()) {
|
|
Header header = mHeaders.get(position);
|
|
if (HeaderAdapter.getHeaderType(header) != HeaderAdapter.HEADER_TYPE_CATEGORY)
|
|
return header;
|
|
position++;
|
|
}
|
|
|
|
Log.e(LOG_TAG, "Unable to find a non-category header");
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public Intent onBuildStartFragmentIntent(String fragmentName, Bundle args,
|
|
int titleRes, int shortTitleRes) {
|
|
Intent intent = super.onBuildStartFragmentIntent(fragmentName, args,
|
|
titleRes, shortTitleRes);
|
|
intent.setClass(this, SubSettings.class);
|
|
return intent;
|
|
}
|
|
|
|
/**
|
|
* Populate the activity with the top-level headers.
|
|
*/
|
|
@Override
|
|
public void onBuildHeaders(List<Header> headers) {
|
|
loadHeadersFromResource(R.xml.settings_headers, headers);
|
|
|
|
updateHeaderList(headers);
|
|
|
|
mHeaders = headers;
|
|
}
|
|
|
|
private void updateHeaderList(List<Header> target) {
|
|
int i = 0;
|
|
while (i < target.size()) {
|
|
Header header = target.get(i);
|
|
// Ids are integers, so downcasting
|
|
int id = (int) header.id;
|
|
if (id == R.id.dock_settings) {
|
|
if (!needsDockSettings())
|
|
target.remove(header);
|
|
} else if (id == R.id.operator_settings || id == R.id.manufacturer_settings) {
|
|
Utils.updateHeaderToSpecificActivityFromMetaDataOrRemove(this, target, header);
|
|
} else if (id == R.id.call_settings) {
|
|
if (!Utils.isVoiceCapable(this))
|
|
target.remove(header);
|
|
} else if (id == R.id.wifi_settings) {
|
|
// Remove WiFi Settings if WiFi service is not available.
|
|
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_WIFI)) {
|
|
target.remove(header);
|
|
}
|
|
} else if (id == R.id.bluetooth_settings) {
|
|
// Remove Bluetooth Settings if Bluetooth service is not available.
|
|
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH)) {
|
|
target.remove(header);
|
|
}
|
|
}
|
|
|
|
// Increment if the current one wasn't removed by the Utils code.
|
|
if (target.get(i) == header) {
|
|
// Hold on to the first header, when we need to reset to the top-level
|
|
if (i == 0) mFirstHeader = header;
|
|
mHeaderIndexMap.put(id, i);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
private boolean needsDockSettings() {
|
|
return getResources().getBoolean(R.bool.has_dock_settings);
|
|
}
|
|
|
|
private void getMetaData() {
|
|
try {
|
|
ActivityInfo ai = getPackageManager().getActivityInfo(getComponentName(),
|
|
PackageManager.GET_META_DATA);
|
|
if (ai == null || ai.metaData == null) return;
|
|
mTopLevelHeaderId = ai.metaData.getInt(META_DATA_KEY_HEADER_ID);
|
|
mFragmentClass = ai.metaData.getString(META_DATA_KEY_FRAGMENT_CLASS);
|
|
|
|
// Check if it has a parent specified and create a Header object
|
|
final int parentHeaderTitleRes = ai.metaData.getInt(META_DATA_KEY_PARENT_TITLE);
|
|
String parentFragmentClass = ai.metaData.getString(META_DATA_KEY_PARENT_FRAGMENT_CLASS);
|
|
if (parentFragmentClass != null) {
|
|
mParentHeader = new Header();
|
|
mParentHeader.fragment = parentFragmentClass;
|
|
if (parentHeaderTitleRes != 0) {
|
|
mParentHeader.title = getResources().getString(parentHeaderTitleRes);
|
|
}
|
|
}
|
|
} catch (NameNotFoundException nnfe) {
|
|
// No recovery
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean hasNextButton() {
|
|
return super.hasNextButton();
|
|
}
|
|
|
|
@Override
|
|
public Button getNextButton() {
|
|
return super.getNextButton();
|
|
}
|
|
|
|
private static class HeaderAdapter extends ArrayAdapter<Header> {
|
|
static final int HEADER_TYPE_CATEGORY = 0;
|
|
static final int HEADER_TYPE_NORMAL = 1;
|
|
static final int HEADER_TYPE_SWITCH = 2;
|
|
private static final int HEADER_TYPE_COUNT = HEADER_TYPE_SWITCH + 1;
|
|
|
|
private final WifiEnabler mWifiEnabler;
|
|
private final BluetoothEnabler mBluetoothEnabler;
|
|
|
|
private static class HeaderViewHolder {
|
|
ImageView icon;
|
|
TextView title;
|
|
TextView summary;
|
|
Switch switch_;
|
|
}
|
|
|
|
private LayoutInflater mInflater;
|
|
|
|
static int getHeaderType(Header header) {
|
|
if (header.fragment == null && header.intent == null) {
|
|
return HEADER_TYPE_CATEGORY;
|
|
} else if (header.id == R.id.wifi_settings || header.id == R.id.bluetooth_settings) {
|
|
return HEADER_TYPE_SWITCH;
|
|
} else {
|
|
return HEADER_TYPE_NORMAL;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public int getItemViewType(int position) {
|
|
Header header = getItem(position);
|
|
return getHeaderType(header);
|
|
}
|
|
|
|
@Override
|
|
public boolean areAllItemsEnabled() {
|
|
return false; // because of categories
|
|
}
|
|
|
|
@Override
|
|
public boolean isEnabled(int position) {
|
|
return getItemViewType(position) != HEADER_TYPE_CATEGORY;
|
|
}
|
|
|
|
@Override
|
|
public int getViewTypeCount() {
|
|
return HEADER_TYPE_COUNT;
|
|
}
|
|
|
|
@Override
|
|
public boolean hasStableIds() {
|
|
return true;
|
|
}
|
|
|
|
public HeaderAdapter(Context context, List<Header> objects) {
|
|
super(context, 0, objects);
|
|
mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
|
|
|
// These Switches are provided as placeholder until the adapter replaces these with actual
|
|
// Switches inflated from their layouts. Must be done before adapter is set in super
|
|
mWifiEnabler = new WifiEnabler(context, new Switch(context));
|
|
mBluetoothEnabler = new BluetoothEnabler(context, new Switch(context));
|
|
}
|
|
|
|
@Override
|
|
public View getView(int position, View convertView, ViewGroup parent) {
|
|
HeaderViewHolder holder;
|
|
Header header = getItem(position);
|
|
int headerType = getHeaderType(header);
|
|
View view = null;
|
|
|
|
if (convertView == null) {
|
|
holder = new HeaderViewHolder();
|
|
switch (headerType) {
|
|
case HEADER_TYPE_CATEGORY:
|
|
view = new TextView(getContext(), null, android.R.attr.listSeparatorTextViewStyle);
|
|
holder.title = (TextView) view;
|
|
break;
|
|
|
|
case HEADER_TYPE_SWITCH:
|
|
view = mInflater.inflate(R.layout.preference_header_switch_item, parent, false);
|
|
holder.icon = (ImageView) view.findViewById(R.id.icon);
|
|
holder.title = (TextView) view.findViewById(com.android.internal.R.id.title);
|
|
holder.summary = (TextView) view.findViewById(com.android.internal.R.id.summary);
|
|
holder.switch_ = (Switch) view.findViewById(R.id.switchWidget);
|
|
break;
|
|
|
|
case HEADER_TYPE_NORMAL:
|
|
view = mInflater.inflate(com.android.internal.R.layout.preference_header_item, parent, false);
|
|
holder.icon = (ImageView) view.findViewById(com.android.internal.R.id.icon);
|
|
holder.title = (TextView) view.findViewById(com.android.internal.R.id.title);
|
|
holder.summary = (TextView) view.findViewById(com.android.internal.R.id.summary);
|
|
break;
|
|
}
|
|
view.setTag(holder);
|
|
} else {
|
|
view = convertView;
|
|
holder = (HeaderViewHolder) view.getTag();
|
|
}
|
|
|
|
// All view fields must be updated every time, because the view may be recycled
|
|
switch (headerType) {
|
|
case HEADER_TYPE_CATEGORY:
|
|
holder.title.setText(header.getTitle(getContext().getResources()));
|
|
break;
|
|
|
|
case HEADER_TYPE_SWITCH:
|
|
// Would need a different treatment if the main menu had more switches
|
|
if (header.id == R.id.wifi_settings) {
|
|
mWifiEnabler.setSwitch(holder.switch_);
|
|
} else {
|
|
mBluetoothEnabler.setSwitch(holder.switch_);
|
|
}
|
|
// No break, fall through on purpose to update common fields
|
|
|
|
//$FALL-THROUGH$
|
|
case HEADER_TYPE_NORMAL:
|
|
holder.icon.setImageResource(header.iconRes);
|
|
holder.title.setText(header.getTitle(getContext().getResources()));
|
|
CharSequence summary = header.getSummary(getContext().getResources());
|
|
if (!TextUtils.isEmpty(summary)) {
|
|
holder.summary.setVisibility(View.VISIBLE);
|
|
holder.summary.setText(summary);
|
|
} else {
|
|
holder.summary.setVisibility(View.GONE);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return view;
|
|
}
|
|
|
|
public void resume() {
|
|
mWifiEnabler.resume();
|
|
mBluetoothEnabler.resume();
|
|
}
|
|
|
|
public void pause() {
|
|
mWifiEnabler.pause();
|
|
mBluetoothEnabler.pause();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void setListAdapter(ListAdapter adapter) {
|
|
if (mHeaders == null) {
|
|
mHeaders = new ArrayList<Header>();
|
|
// When the saved state provides the list of headers, onBuildHeaders is not called
|
|
// Copy the list of Headers from the adapter, preserving their order
|
|
for (int i = 0; i < adapter.getCount(); i++) {
|
|
mHeaders.add((Header) adapter.getItem(i));
|
|
}
|
|
}
|
|
|
|
// Ignore the adapter provided by PreferenceActivity and substitute ours instead
|
|
super.setListAdapter(new HeaderAdapter(this, mHeaders));
|
|
}
|
|
|
|
/*
|
|
* Settings subclasses for launching independently.
|
|
*/
|
|
|
|
public static class BluetoothSettingsActivity extends Settings { /* empty */ }
|
|
public static class WirelessSettingsActivity extends Settings { /* empty */ }
|
|
public static class TetherSettingsActivity extends Settings { /* empty */ }
|
|
public static class VpnSettingsActivity extends Settings { /* empty */ }
|
|
public static class DateTimeSettingsActivity extends Settings { /* empty */ }
|
|
public static class StorageSettingsActivity extends Settings { /* empty */ }
|
|
public static class WifiSettingsActivity extends Settings { /* empty */ }
|
|
public static class InputMethodAndLanguageSettingsActivity extends Settings { /* empty */ }
|
|
public static class InputMethodConfigActivity extends Settings { /* empty */ }
|
|
public static class InputMethodAndSubtypeEnablerActivity extends Settings { /* empty */ }
|
|
public static class LocalePickerActivity extends Settings { /* empty */ }
|
|
public static class UserDictionarySettingsActivity extends Settings { /* empty */ }
|
|
public static class SoundSettingsActivity extends Settings { /* empty */ }
|
|
public static class DisplaySettingsActivity extends Settings { /* empty */ }
|
|
public static class DeviceInfoSettingsActivity extends Settings { /* empty */ }
|
|
public static class ApplicationSettingsActivity extends Settings { /* empty */ }
|
|
public static class ManageApplicationsActivity extends Settings { /* empty */ }
|
|
public static class StorageUseActivity extends Settings { /* empty */ }
|
|
public static class DevelopmentSettingsActivity extends Settings { /* empty */ }
|
|
public static class AccessibilitySettingsActivity extends Settings { /* empty */ }
|
|
public static class SecuritySettingsActivity extends Settings { /* empty */ }
|
|
public static class PrivacySettingsActivity extends Settings { /* empty */ }
|
|
public static class DockSettingsActivity extends Settings { /* empty */ }
|
|
public static class RunningServicesActivity extends Settings { /* empty */ }
|
|
public static class ManageAccountsSettingsActivity extends Settings { /* empty */ }
|
|
public static class PowerUsageSummaryActivity extends Settings { /* empty */ }
|
|
public static class AccountSyncSettingsActivity extends Settings { /* empty */ }
|
|
public static class AccountSyncSettingsInAddAccountActivity extends Settings { /* empty */ }
|
|
public static class CryptKeeperSettingsActivity extends Settings { /* empty */ }
|
|
public static class DeviceAdminSettingsActivity extends Settings { /* empty */ }
|
|
public static class DataUsageSummaryActivity extends Settings { /* empty */ }
|
|
}
|