【问题标题】:How to select multiple checkboxes in submenu on Android?如何在 Android 的子菜单中选择多个复选框?
【发布时间】:2013-04-09 19:05:11
【问题描述】:

我有一个带有“添加/删除”选项的选项菜单,单击该选项时会显示一个可检查的列表。我目前拥有的代码的问题是您一次只能选择一项,并且菜单消失了。我希望能够一次检查列表中的多个项目,并且在用户触摸屏幕上其他位置之前它不会消失。我怎样才能做到这一点?这是我所拥有的总体思路:

<?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">
        <menu>
            <group android:checkableBehavior="all">
                <item android:id="@+id/A" 
                      android:checked="true" 
                      android:title="Option One" />
                <item android:id="@+id/B" 
                      android:checked="true" 
                      android:title="Option Two" />
            </group>
        </menu>
    </item>
</menu>

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.selection_menu, menu);
    return true;
}

@Override
public boolean onOptionsItemSelected(MenuItem item){
    switch (item.getItemId()){
    case R.id.A:
        item.setChecked(!item.isChecked());
        return true;
    case R.id.B:
        item.setChecked(!item.isChecked());
        return true;
   default:
        return super.onOptionsItemSelected(item);
   }
}

【问题讨论】:

  • 你有没有发现它?我正在寻找相同的

标签: android menu checkbox


【解决方案1】:

你好披头士狂热!

老实说,我不知道您正在寻找的东西是否可行(编辑:以您实现它的方式,作为子菜单),但我会这样做:

创建一个看起来像您要显示的子菜单的 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 文件如下所示:
<?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) 将使活动看起来像一个子菜单。

希望对您有所帮助! 我试过了,效果很好!试试看,让我知道会发生什么。

【讨论】:

  • 太棒了!我正在使用上面的代码,但是当我按下后退按钮或返回父活动时,如何获取已检查的项目列表?
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-01-21
  • 2015-11-30
  • 1970-01-01
  • 2011-11-06
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多