Android(安卓)UI学习 - Menu
16lz
2021-01-26
标签: Android menu 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处、作者信息和本声明。否则将追究法律责任。 http://android.blog.51cto.com/268543/306424 <!--正文 begin--> Android系统里面有3种类型的菜单:options menu,context menu,sub menu。 options menu 按Menu键就会显示,用于当前的Activity。 它包括两种菜单项: 因为options menu在屏幕底部最多只能显示6个菜单项,这些菜单项称为 icon menu,icon menu只支持文字(title) 以及icon,可以设置快捷键,不支持checkbox以及radio控件,所以不能设置checkable选项。 而多于6的菜单项会以“more” icon menu来调出,称为 expanded menu。它不支持icon,其他的特性都和icon menu一样! 在Activity里面,一般通过以下函数来使用options menu: Activity::onCreateOptionsMenu (Menu menu) 创建options menu,这个函数只会在menu第一次显示时调用。 Activity::onPrepareOptionsMenu (Menu menu)更新改变options menu的内容,这个函数会在menu每次显示时调用。 Activity::onOptionsItemSelected (MenuItem item)处理选中的菜单项。 context menu 要在相应的view上按几秒后才显示的,用于view,跟某个具体的view绑定在一起。 这类型的菜单不支持icon和快捷键! 在Activity里面,一般通过以下函数来使用context menu: Activity::registerForContextMenu(View view)为某个view注册context menu,一般在Activity::onCreate里面调用。 Activity::onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo)创建context menu,和options menu不同,context meun每次显示时都会调用这个函数。 Activity::onContextItemSelected(MenuItem item)处理选中的菜单项。 sub menu 以上两种menu都可以加入子菜单,但子菜单不能嵌套子菜单,这意味着在Android系统,菜单只有两层,设计时需要注意的!同时子菜单不支持icon。
xml形式的menu定义及应用
上述的三种类型的menu都能够定义为xml资源,但需要手动地使用 MenuInflater来得到Menu对象的引用。 一个菜单,对应一个xml文件,因为要求只能有一个根节点<menu>。官方说<?xml>声明可以不写,但我觉得还是写上好些,很多时候那个<?xml>声明主要是为了声明编码格式utf-8之类的。xml文件保存为res/menu/some_file.xml。Java代码引用资源: R.menu.some_file 接下来介绍相关的节点和属性(所有的属性都定义为android空间内,例如android:icon="@drawable/icon"):<menu>根节点,没有属性。
<group>表示在它里面的<item>在同一group。相关属性包括:
id:group id
menuCategory:对应常量Menu CATEGORY_* — 定义了一组的优先权,有 效值:container,system,secondary,和alternative orderInCategory:定义这组菜单在菜单中的默认次序,int值 checkableBehavior:这组菜单项是否checkable。有效值:none,all(单选/单选按钮radio button),single(非单选/复选类型checkboxes) visible:这组菜单是否可见 true or false enabled:这组菜单是否可用,true or false <item> 菜单项,可以嵌入<menu>作为子菜单。相关属性包括: id: item id menuCategory: 用来定义menu类别 orderInCategory: 用来定义次序,与一个组在一起(Used to define the order of the item, within a group ) title: 标题 titleCondensed:标题摘要, 当原标题太长的时候,需要用简短的字符串来代替title icon: icon 图标 alphabeticShortcut: 字母快捷键 numericShortcut:数学快捷键 checkable:是否为checkbox, true or false checked:是否设置为checked状态,true or false visible: 是否可见, true or false enabled:是否可用,true or false xml示例:
Java代码
- <?xmlversion="1.0"encoding="utf-8"?>
- <menuxmlns:android="http://schemas.android.com/apk/res/android">
- <itemandroid:id="@+id/item1"
- android:title="Item1"
- android:icon="@drawable/icon"
- android:checkable="true"
- android:checked="false"
- />
- <groupandroid:id="@+id/group_1"
- android:checkableBehavior="single">
- <itemandroid:id="@+id/group_item1"
- android:title="Item1ingroup"
- />
- <itemandroid:id="@+id/group_item2"
- android:title="Item2ingroup"
- android:checked="true"
- />
- </group>
- <itemandroid:id="@+id/submenu"
- android:title="SubMenu">
- <menu>
- <itemandroid:id="@+id/submenu_item"
- android:title="SubMenuItem"
- />
- </menu>
- </item>
- <itemandroid:id="@+id/item3"
- android:title="item3"
- android:checkable="true"
- android:checked="true"
- />
- </menu>
效果图 由于这是contextMenu,所以可以看到即使xml定义里面的item1.seticon了,但还是没有显示出来的,即那语句是无效的! 另外,要明确的是,要显示radio,需要用group,而group里面的item设置了checked = true即选中。而checkable和checked的区别,一开始我是很困惑的,但写了代码并运行后,明白它们的区别了: checkable=true表示这个item是checkbox, checked则表示是否选中。所以对于checkbox item,最好先写 checkable="true",然后再写checked。
- publicvoidonCreate(BundlesavedInstanceState){
- ...
- registerForContextMenu(editText);
- }
- @Override
- publicvoidonCreateContextMenu(ContextMenumenu,Viewv,
- ContextMenuInfomenuInfo){
- super.onCreateContextMenu(menu,v,menuInfo);
- getMenuInflater().inflate(R.menu.menu1,menu);
- }
Java实现
用Java来实现以上的效果图,就比较麻烦些:在编写过程中,发现groupId的影响很大,不推荐使用Menu.add(int titleRes)和add(CharSequence title)方法来添加MenuItem,因为没有指定groupID,默认为0,这样子和后面的menu group 一组了,导致执行完 menu.setGroupCheckable(groupId,true,true)后同一group的Item都变成radio。 OptionsMenu的Java实现
- privatestaticfinalintMENU_GROUPITEM1=Menu.FIRST+8;
- privatestaticfinalintMENU_GROUPITEM2=Menu.FIRST+9;
- privatestaticfinalintMENU_ITEM1=Menu.FIRST+10;
- publicvoidonCreate(BundlesavedInstanceState){
- ...
- registerForContextMenu(findViewById(R.id.edittext));
- }
- @Override
- publicvoidonCreateContextMenu(ContextMenumenu,Viewv,
- ContextMenuInfomenuInfo){
- super.onCreateContextMenu(menu,v,menuInfo);
- menu.add(1,MENU_ITEM1,Menu.NONE,"Item1").setCheckable(true).setChecked(false);
- //GroupID
- intgroupId=0;
- //Theorderpositionoftheitem
- intmenuItemOrder=Menu.NONE;
- menu.add(groupId,MENU_GROUPITEM1,menuItemOrder,"Item1ingroup");
- menu.add(groupId,MENU_GROUPITEM2,menuItemOrder,"Item2ingroup")
- .setChecked(true);
- menu.setGroupCheckable(groupId,true,true);//这句要写在groupitem的最后
- SubMenusubMenu=menu.addSubMenu("SubMenu1");
- subMenu.add("SubMenuItem")
- .setOnMenuItemClickListener(newMenuItem.OnMenuItemClickListener(){
- @Override
- publicbooleanonMenuItemClick(MenuItemitem){
- Toast.makeText(HelloDemo.this,
- "SubMenuItemselected",
- Toast.LENGTH_SHORT).show();
- returntrue;//true表示完成当前item的click处理,不再传递到父类处理
- }
- });
- menu.add("Item3").setCheckable(true).setChecked(true);
- }
处理菜单点击事件 方法一: 利用菜单自带的监听器功能,直接监听,就象处理控件事件一样,像上面的ContextMenu的 subMenu.add("SubMenuItem")设置MenuItem.OnMenuItemClickListener。 方法二: 在Activity和View都直接提供了一个菜单点击统一处理函数, Activity::onOptionsItemSelected (MenuItem item); Activity::onContextItemSelected(MenuItem item) ;
- @Override
- publicbooleanonCreateOptionsMenu(Menumenu){
- //GroupID
- intgroupId=0;
- //Theorderpositionoftheitem
- intmenuItemOrder=Menu.NONE;
- menu.add(groupId,MENU_COPY,menuItemOrder,"Copy")
- .setIcon(R.drawable.icon);
- menu.add(groupId,MENU_EDIT,menuItemOrder,"Edit");
- menu.add(groupId,MENU_PASTE,menuItemOrder,"Paste");
- menu.add(groupId,MENU_DELETE,menuItemOrder,"Delete");
- menu.add(groupId,MENU_OK,menuItemOrder,"Ok");
- menu.add(groupId,MENU_CANCEL,menuItemOrder,"Cancel");
- menu.add(groupId,MENU_TEST,menuItemOrder,"Test");
- menu.add(groupId,MENU_DEMO,menuItemOrder,"Demo");
- //.setIcon(R.drawable.icon);moreexpandmenu不支持icon,setIcon不会报错,但运行时还是看不到icon的
- //returnsuper.onCreateOptionsMenu(menu);
- returntrue;//true表示要显示menu;false表示不显示menu
- }
效果图
- @Override
- publicbooleanonOptionsItemSelected(MenuItemitem){
- switch(item.getItemId()){
- caseMENU_COPY:
- Toast.makeText(this,"CopyItemselected",Toast.LENGTH_SHORT).show();
- break;
- default:break;
- }
- returnfalse;//false表示继续传递到父类处理
- }
动态菜单
对于OptionsMenu,一般可以使用onPrepareOptionsMenu来改变。 另外,使用函数 android.view.Menu.addIntentOptions(int groupId,int itemId,int order,ComponentName caller, Intent[] specifics, Intent intent,int flags,MenuItem[] outSpecificItems) Specifics 以action+uri的具体方式来增加激活相应activity的菜单项 Intent 以categroy+uri这种一般形式来增加激活相应activity的菜单项 参数Intent和Specifics的区别是,一个用categroy+uri来匹配activity,一个用action+uri来匹配activity。 //按Action查找 Intent[] specifics = new Intent[1]; specifics[0] = new Intent(Intent.ACTION_EDIT, uri); //按Category查找,Action设为null Intent intent = new Intent(null, uri); intent.addCategory(Intent.CATEGORY_ALTERNATIVE); MenuItem[] items = new MenuItem[1]; menu.addIntentOptions(Menu.CATEGORY_ALTERNATIVE, 0, 0, null, specifics, intent, 0, items); 有关Menu的创建可以参考官方的 http://androidappdocs.appspot.com/guide/topics/ui/menus.html。另外官方提供了Menu Design Guidelines http://androidappdocs.appspot.com/guide/practices/ui_guidelines/menu_design.html本文出自 “学习Android” 博客,请务必保留此出处http://android.blog.51cto.com/268543/306424
更多相关文章
- 【android】类似微信底部按钮标签实现
- (三)Kotlin 高阶函数
- android 对pdf文件的下载、缓存、显示,包含android-pdfview框架使
- Android(安卓)开源库——侧滑菜单栏(SlidingMenu)的导入和使用
- android:三种菜单(Menu)的设置
- Android之Menu选项菜单
- 【Android(安卓)Demo】图片之画廊效果(Gallery Switcher)
- git的使用(上传项目到github)
- Android的短信接收类