Android中的Menu点滴记录

 

      菜单是许多应用程序不可或缺的一部分,Android中更是如此,所有搭载Android系统的手机甚至都要有一个"Menu"键,由此可见菜单在Android程序中的特殊性。Android SDK提供的菜单有如下几种:

选项菜单:最常规的菜单,android中把它叫做option menu

子菜单:android中点击子菜单将弹出悬浮窗口显示子菜单项。子菜单不支持嵌套,即子菜单中不能再包括其他子菜单。

上下文菜单:android长按视图控件后出现的菜单,windows点击右键弹出的菜单即上下文菜单

图标菜单:这个比较简单,就是带icon的菜单项,需要注意的是子菜单项、上下文菜单项、扩展菜单项均无法显示图标

 

创建options menu

之前提到,Androidactivity已经为我们提前创建好了android.view.Menu对象,并提供了回调方法onCreateOptionsMenu(Menu menu)供我们初始化菜单的内容。该方法只会在选项菜单第一次显示的时候被执行,如果你需要动态改变选项菜单的内容,请使用 onPrepareOptionsMenu(Menu menu)

@Override
public boolean onCreateOptionsMenu(Menu menu) {
	// 虽然目前android还没有系统菜单,但是为了兼容到以后的版本,最好加上
	super.onCreateOptionsMenu(menu);
	// 第一个参数代表组号,android中你可以给菜单分组,以便快速地操作同一组的菜单。
	// 第二个参数代表Menu的唯一的ID号,可以自己指定,也可以让系统来自动分配,在响应菜单时你需要通过ID号来判断点击了哪个菜单。
	// 第三个参数代表Menu显示顺序的编号,编号小的显示在前面。
	// 第四个参数代表标题
	menu.add(0, MENUITEM1, MENUITEM1, "关于");
	menu.add(0, MENUITEM2, MENUITEM2, "帮助");
	menu.add(0, MENUITEM3, MENUITEM3, "更多");

	return true;
}

 可以对Menu进行分组:

@Override
public boolean onCreateOptionsMenu(Menu menu) {
	super.onCreateOptionsMenu(menu);
	// 添加4个菜单项,分成2组
	int group1 = 1;
	int gourp2 = 2;
	menu.add(group1, 1, 1, "item 1");
	menu.add(group1, 2, 2, "item 2");
	menu.add(gourp2, 3, 3, "item 3");
	menu.add(gourp2, 4, 4, "item 4");
	return true;
}

 分组之后就能使用menu中提供的方法对组进行操作了,如下:menu.removeGroup(group1); //删除一组菜单

menu.removeGroup(group1);    //删除一组菜单
menu.setGroupVisible(gourp2, visible);    //设置一组菜单是否可见
menu.setGroupEnabled(gourp2, enabled);    //设置一组菜单是否可点
menu.setGroupCheckable(gourp2, checkable, exclusive);    //设置一组菜单的勾选情况

 响应菜单项

 

android提供了多种响应菜单项的方式,下面一一介绍

1、通过onOptionsItemSelected方法

使用的最多方法是重写activity类的 onOptionsItemSelected(Menu Item)回调方法,每当有菜单项被点击时,android就会调用该方法,并传入被点击菜单项。

 @Override

public boolean onOptionsItemSelected(MenuItem item) {
	switch (item.getItemId()) {
	// 响应每个菜单项(通过菜单项的ID)
	case MENUITEM1:
		break;
	case MENUITEM2:
		break;
	case MENUITEM3:
		break;
	default:
		// 对没有处理的事件,交给父类来处理
		return super.onOptionsItemSelected(item);
	}
	// 返回true表示处理完菜单项的事件,不需要将该事件继续传播下去了
	return true;
}

 2.使用监听器

虽然第一种方法是推荐使用的方法,android还是提供了类似java swing的监听器方式来响应菜单。使用监听器的方式分为两步:

第一步:创建监听器类

class MenuItemListener implements OnMenuItemClickListener {
	// 该方法先于onOptionsItemSelected执行。
	@Override
	public boolean onMenuItemClick(MenuItem item) {
		return false;
	}
}

 

 第二步:为菜单项注册监听器

menu.getItem(0).setOnMenuItemClickListener(new MenuItemListener()); // 为MenuItem添加监听事件

 3.使用Intent响应菜单

menu.getItem(1).setIntent(new Intent()); // 可以为MenuItem添加Intent

 

子菜单sub Menu

Android支持子菜单,你可以通过addSubMenu(int groupId, int itemId, int order, int titleRes)方法创建和响应子菜单。

/**
 * 使用的最多方法是重写activity类的 onOptionsItemSelected(Menu
 * Item)回调方法,每当有菜单项被点击时,android就会调用该方法,并传入被点击菜单项。
 */
@Override
public boolean onPrepareOptionsMenu(Menu menu) {
	Resources res = getBaseContext().getResources();
	// 第一个参数代表组号,android中你可以给菜单分组,以便快速地操作同一组的菜单。
	// 第二个参数代表Menu的唯一的ID号,可以自己指定,也可以让系统来自动分配,在响应菜单时你需要通过ID号来判断点击了哪个菜单。
	// 第三个参数代表Menu显示顺序的编号,编号小的显示在前面。
	// 第四个参数代表标题
	SubMenu about = menu.addSubMenu(0, MENUITEM1, MENUITEM1, "关于"); // 添加菜单项(多种方式)
	SubMenu help = menu.addSubMenu(0, MENUITEM2, MENUITEM2, "帮助");
	SubMenu more = menu.addSubMenu(0, MENUITEM3, MENUITEM3, "更多");
	// 为MenuItem设置图标
	about.setIcon(res.getDrawable(R.drawable.ic_launcher));
	help.setIcon(res.getDrawable(R.drawable.ic_launcher));
	more.setIcon(res.getDrawable(R.drawable.ic_launcher));

	// 为more菜单添加子订单
	MenuItem display = more.add(0, MENUITEM3, MENUITEM3, "设置");
	display.setIcon(R.drawable.ic_launcher);// 子菜单项不支持显示图标,这样做是没意义的,尽管不会报错!
	more.add(0, MENUITEM3, MENUITEM3, "退出");
	more.setHeaderIcon(R.drawable.ic_launcher); // 以添加子菜单项栏目的标题图标

	return true;
}

 Menu可以包含多个SubMenuSubMenu可以包含多个MenuItem,但是SubMenu不能包含SubMenu,及子菜单不能嵌套

 

图标菜单icon Menu

Android支持在菜单上显示各种各样的图标,这一点我们在上面创建子菜单时已经用到了。图标菜单严格上说并不算是一种菜单的新类型,它的使用也很简单,之所以单独设一节是为了说明使用Icon的一些限制。Android中并不是所谓的菜单项都能加上图标,以下菜单项都是不可以的(这并不意味着程序会报错,而是运行时图标得不到显示):

· 扩展的菜单项

· 子菜单的菜单项

· 上下文菜单的菜单项

除此以外,带Icon的菜单项不能加上复选框(check mark)

 

 

上下文菜单ContextMenu

android是通过长按某个视图元素来弹出上下文菜单的,图标和子菜单都无法用在Android的上下文菜单项中。

上下文菜单继承了android.view.Menu,因此我们可以像操作Options Menu那样给上下文菜单增加菜单项。上下文菜单与Options Menu最大的不同在于,Options Menu的拥有者是Activity,而上下文菜单的拥有者是Activity中的View。每个Activity有且只有一个Options Menu,它为整个Activity服务。而一个Activity往往有多个View,并不是每个View都有上下文菜单,这就需要我们显示地通过registerForContextMenu(Viewview)来指定

尽管上下文菜单的拥有者是View,生成上下文菜单却是通过Activity中的onCreateContextMenu(ContextMenu menu, View v,ContextMenu.ContextMenuInfo menuInfo)方法,该方法很像生成Options MenuonCreateOptionsMenu(Menu menu)方法。两者的不同在于,onCreateOptionsMenu只在用户第一次Menu键时被调用,而onCreateContextMenu会在用户每一次长按View时被调用,而且View必须已经注册了上下文菜单。

另一个值得注意的就是ContextMenuInfo,该类的对象被传入onCreateContextMenu(ContextMenu menu, View v,ContextMenu.ContextMenuInfo menuInfo)方法,那么它有什么用呢?有时候,视图元素需要向上下文菜单传递一些信息,比如该View对应DB记录的ID等,这就要使用ContextMenuInfo。需要传递额外信息的View需要重写getContextMenuInfo()方法,返回一个带有数据的ContextMenuInfo实现类对象。

1.activityonCreate(...)方法中为一个view注册上下文菜单

public class MyListActivity extends ListActivity {

	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.two);
		simpleShowList();

		this.registerForContextMenu(getListView());
	}

	private void simpleShowList() {
		String[] item = new String[] { "ListItem1", "ListItem2", "ListItem3", ListItem4" };
		ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,
				android.R.layout.simple_list_item_1, item);
		this.setListAdapter(adapter);
	}
}
 

2.onCreateContextMenuInfo(...)中生成上下文菜单。

@Override
public void onCreateContextMenu(ContextMenu menu, View v,
		ContextMenuInfo menuInfo) {
	menu.setHeaderTitle("文件操作");
	menu.add(0, 1, Menu.NONE, "发送");
	menu.add(0, 2, Menu.NONE, "复制");
	menu.add(0, 3, Menu.NONE, "重命名");
	menu.add(0, 4, Menu.NONE, "删除");
}
  

3.onContextItemSelected(...)中响应上下文菜单项。

 

@Override
public boolean onContextItemSelected(MenuItem item) {
	// 得到当前被选中的item信息
	AdapterContextMenuInfo menuInfo = (AdapterContextMenuInfo) item.getMenuInfo();
	switch (item.getItemId()) {
	case 1: 
		break;
	case 2: 
		break;
	case 3: 
		break;
	case 4:
		break;
	default:
		return super.onContextItemSelected(item);
	}
	return true;
}
 

 

Menu关闭

如果需要动态的显示菜单可以在菜单关闭时进行相应的清除处理:

@Override
public void onOptionsMenuClosed(Menu menu) {
	menu.clear();
	super.onOptionsMenuClosed(menu);
}

你可能感兴趣的:(android,菜单,menu)