你好披头士狂热!
老实说,我不知道您正在寻找的东西是否可行(编辑:以您实现它的方式,作为子菜单),但我会这样做:
创建一个看起来像您要显示的子菜单的 Activity。
它可能看起来有点复杂,但它是直截了当的,以这种方式,如果您选择/取消选择一个项目,它不会消失,并且您可以实现更多功能。
这是我个人的做法:
- 创建一个类来表示子菜单项。它应该包含一个字符串(描述)和一个布尔值(存储它是否被选中)。
public class SettingCheckBox implements Serializable {
private static final long serialVersionUID = 1L;
private static final String DEFAULT_DESCRIPTION = "N/A";
private final String description;
private boolean checked;
public String getDescription () {
return description == null ? DEFAULT_DESCRIPTION : description;
}
public void setChecked ( final boolean checked ) {
this.checked = checked;
}
public boolean getChecked () {
return checked;
}
public SettingCheckBox ( final String description ) {
this.description = description;
}
}
如您所见,该类实现了 Serializable 以便该类的对象可以使用意图/捆绑从一个活动传递到另一个活动。
- 将以下内容添加到您当前的活动中(我假设它称为 MainActivity,因此在您尝试时,请将 MainActivity 替换为您的活动名称)。
public static final String SETTING_CHECK_BOX = "SETTING_CHECK_BOX";
private ArrayList < SettingCheckBox > settingList;
@Override
public void onCreate(Bundle savedInstanceState) {
// ...
settingList = new ArrayList < SettingCheckBox > ();
settingList.add ( new SettingCheckBox ( "Option A" ) );
settingList.add ( new SettingCheckBox ( "Option B" ) );
// ... add more items
// restore any previously saved list
if ( savedInstanceState != null ) {
settingList = (ArrayList < SettingCheckBox >) savedInstanceState.getSerializable ( SETTING_CHECK_BOX );
}
// ...
}
protected void onSaveInstanceState ( Bundle outState ) {
super.onSaveInstanceState ( outState );
outState.putSerializable ( SETTING_CHECK_BOX , settingList );
}
列表 (ArrayList) 用于承载所有带有复选框的设置子菜单项。
如您所见,每个 SettingCheckBox 对象都有描述和状态(选中或未选中)。默认情况下,一旦创建,对象状态为unchecked。
您应该在 onCreate 方法中初始化列表。
静态和最终变量 SETTING_CHECK_BOX 用作键,用于在活动重新创建之前/之后(如屏幕旋转)保存/恢复该列表的内容,并从活动到另一个。 (稍后解释)
<?xml version="1.0" encoding="UTF-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@+id/select_options"
android:title="Add/Remove">
</item>
</menu>
不再需要子菜单,因为您将实现一个行为类似的活动。
现在,要将菜单项链接到将显示设置的活动,您应该在当前活动中使用 onOptionsItemSelected 方法,如下所示:
@Override
public boolean onOptionsItemSelected ( MenuItem menuItem ) {
if ( menuItem.getItemId () == R.id.select_options ) {
Intent intent = new Intent ( this , MyActivity_Settings.class );
intent.putExtra ( SETTING_CHECK_BOX , settingList );
startActivityForResult ( intent , 0 );
}
}
开始设置活动以获得结果。这意味着它就像一个子活动,并且可以将结果返回给它的父活动。
设置列表通过意图传递给设置活动。
如果子活动结束并向父活动返回数据,则调用以下方法:
protected void onActivityResult ( int requestCode , int resultCode , Intent data ) {
if ( resultCode != RESULT_OK || data == null )
return;
settingList = (ArrayList < SettingCheckBox >) data.getSerializableExtra ( SETTING_CHECK_BOX );
}
您应该让子/设置活动返回(新的/修改的)设置列表,并且如上所示,设置了新列表。
- 创建以下名为 sub_menu 的布局 xml 文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<ListView
android:id="@android:id/list"
android:layout_width="match_parent"
android:layout_height="wrap_content" >
</ListView>
</LinearLayout>
这是作为子菜单的 Activity 布局。它实际上是一个列表活动,可以包含任意数量的选项(只需将它们添加到上面活动中声明的数组列表中)。
- 创建以下名为 sub_menu_item 的布局 xml 文件:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical" >
<TextView
android:id="@+id/option_title"
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_weight="1"
android:textAppearance="@android:style/TextAppearance.Medium" />
<CheckBox
android:id="@+id/option_checkbox"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</LinearLayout>
这是列表中每一行的布局,有一个文本视图和复选框(就像您已经使用的子菜单一样)。
- 创建一个名为 MyActivity_Settings 的新类,其中应包含以下内容:
public class MyActivity_Settings extends ListActivity {
private ArrayList < SettingCheckBox > settingList;
@Override
public void onCreate ( Bundle savedInstanceState ) {
super.onCreate(savedInstanceState);
setContentView ( R.layout.sub_menu );
setTitle ( "Add/Remove" );
settingList = getIntent ().getSerializableExtra ( MainActivity.SETTING_CHECK_BOX );
if ( savedInstanceState != null ) {
settingList = (ArrayList < SettingCheckBox >) savedInstanceState.getSerializable ( MainActivity.SETTING_CHECK_BOX );
}
setListAdapter ( new MyActivity_Settings_Adapter ( this , R.layout.item_layout , settingList ) );
}
protected void onSaveInstanceState ( Bundle outState ) {
super.onSaveInstanceState ( outState );
outState.putSerializable ( MainActivity.SETTING_CHECK_BOX , settingList );
}
@Override
public void finish () {
setResult ( RESULT_OK , new Intent ().putExtra ( MainActivity.SETTING_CHECK_BOX , settingList ) );
super.finish ();
}
}
class MyActivity_Settings_Adapter extends ArrayAdapter < SettingCheckBox > {
private final LayoutInflater layoutInflater;
private final int itemResourceId;
// Holder pattern (used instead of findViewById for better performance)
static class ViewHolder {
public TextView title;
public CheckBox checkBox;
}
// Constructor
public MyActivity_Settings_Adapter ( ListActivity context, int itemResourceId , List < SettingCheckBox > options ) {
super ( context , itemResourceId , options );
layoutInflater = context.getLayoutInflater ();
this.itemResourceId = itemResourceId;
}
// Method called by the list view every time to display a row
@Override
public View getView ( int position , View convertView , ViewGroup parent ) {
// Declare and initialize the row view
View rowView = convertView;
// Declare the row view holder
ViewHolder viewHolder;
// Check if an inflated view is provided
if ( rowView == null ) {
// A new view must be inflated
rowView = layoutInflater.inflate ( itemResourceId , null );
// Declare and initialize a view holder
viewHolder = new ViewHolder ();
// Retrieve a reference to the row title
viewHolder.title = (TextView) rowView.findViewById ( R.id.option_title );
// Retrieve a reference to the row check box
viewHolder.checkBox = (CheckBox) rowView.findViewById ( R.id.option_checkbox );
// Store the view holder as tag
rowView.setTag ( viewHolder );
} // End if
else
// An inflated view is already provided, retrieve the stored view holder
viewHolder = (ViewHolder) rowView.getTag ();
// Set the option title
viewHolder.title.setText ( getItem ( position ).getDescription () );
// Set the option check box state
viewHolder.checkBox.setChecked ( getItem ( position ).getChecked () );
// Assign a click listener to the checkbox
viewHolder.checkBox.setOnClickListener( new OnClickListener() {
public void onClick ( View checkBox ) {
// Retrieve the stored view holder
ViewHolder viewHolder = (ViewHolder) ((View) checkBox.getParent()).getTag();
// Update the option state
getItem ( position ).setChecked ( ! getItem ( position ).getChecked () );
// Display the new option state
viewHolder.checkBox.setChecked ( getItem ( position ).getChecked () );
}
});
// Return the row view for display
return rowView;
} // End of getView
}
这个类表示将作为您的子菜单的活动。正如我之前所说,它是一个 List Activity(因此应该扩展 ListActivity)。为了在列表中显示各种选项,您需要一个适配器(在这种情况下数组适配器就足够了),这就是 MyActivity_Settings_Adapter 类(扩展 ArrayAdapter)的作用。
如果列表活动完成(用户单击后退按钮,或显示为对话框的活动之外的任何位置),它(活动)将带有新检查值的新选项列表返回给父活动.
适配器将为要显示的列表构建每一行。
此外,适配器会为每个复选框分配一个点击监听器,这样如果选中(或未选中),该选项将被相应地修改。
如果您单击子菜单外的任何位置(或只需按下后退按钮),子菜单将消失,并且用户选择将保留在您的主活动的布尔数组中。
如果您不熟悉 ListActivity 和 ArrayAdapter,这个tutorial 会很有帮助!
- 不要忘记在你的 android manifest xml 文件中添加这个(在应用程序标签中):
<activity android:name=".MyActivity_Settings"
android:theme="@android:style/Theme.Dialog" />
应用的主题 (@android:style/Theme.Dialog) 将使活动看起来像一个子菜单。
希望对您有所帮助!
我试过了,效果很好!试试看,让我知道会发生什么。