在android中实现菜单功能有多种方法。

Options Menu:用户按下menu Button时显示的菜单。

Context Menu:用户长时间按下屏幕,所显示出来的菜单也称为上下文菜单。

Submenu:子菜单。

但是有时候这些内置的菜单并不能满足我们功能,这就需要自己自定义一种菜单。接下来我说的这种就是通过TabHost与RadioGroup结合完成的菜单。这也是很常用的一种底部菜单做法。先上图:

首先看布局文件:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <TabHost android:id="@android:id/tabhost" android:layout_width="fill_parent" android:layout_height="fill_parent"
  3. xmlns:android="http://schemas.android.com/apk/res/android">
  4. <LinearLayout
  5. android:orientation="vertical"
  6. android:layout_width="fill_parent"
  7. android:layout_height="fill_parent">
  8. <FrameLayout
  9. android:id="@android:id/tabcontent"
  10. android:layout_width="fill_parent"
  11. android:layout_height="0.0dip"
  12. android:layout_weight="1.0" />
  13. <TabWidget
  14. android:id="@android:id/tabs"
  15. android:visibility="gone"
  16. android:layout_width="fill_parent"
  17. android:layout_height="wrap_content"
  18. android:layout_weight="0.0" />
  19. <RadioGroup
  20. android:gravity="center_vertical"
  21. android:layout_gravity="bottom"
  22. android:orientation="horizontal"
  23. android:id="@+id/main_radio"
  24. android:background="@drawable/maintab_toolbar_bg"
  25. android:layout_width="fill_parent"
  26. android:layout_height="wrap_content">
  27. <RadioButton
  28. android:id="@+id/radio_button0"
  29. android:tag="radio_button0"
  30. android:layout_marginTop="2.0dip"
  31. android:text="@string/alarm"
  32. android:drawableTop="@drawable/icon_1"
  33. style="@style/main_tab_bottom" />
  34. <RadioButton
  35. android:id="@+id/radio_button1"
  36. android:tag="radio_button1"
  37. android:layout_marginTop="2.0dip"
  38. android:text="@string/message"
  39. android:drawableTop="@drawable/icon_2"
  40. style="@style/main_tab_bottom" />
  41. <RadioButton
  42. android:id="@+id/radio_button2"
  43. android:tag="radio_button2"
  44. android:layout_marginTop="2.0dip"
  45. android:text="@string/photo"
  46. android:drawableTop="@drawable/icon_3"
  47. style="@style/main_tab_bottom" />
  48. <RadioButton
  49. android:id="@+id/radio_button3"
  50. android:tag="radio_button3"
  51. android:layout_marginTop="2.0dip"
  52. android:text="@string/music"
  53. android:drawableTop="@drawable/icon_4"
  54. style="@style/main_tab_bottom" />
  55. <RadioButton
  56. android:id="@+id/radio_button4"
  57. android:tag="radio_button4"
  58. android:layout_marginTop="2.0dip"
  59. android:text="@string/setting"
  60. android:drawableTop="@drawable/icon_5"
  61. style="@style/main_tab_bottom" />
  62. </RadioGroup>
  63. </LinearLayout>
  64. </TabHost>

需要注意的是,如果用TabHost这个控件,其中有几个ID是必须这么写的,android:id="@android:id/tabhost   ;android:id="@android:id/tabcontent" ;android:id="@android:id/tabs" ;之所以要这么写是因为在TabHost这个类中。需要实例化上述这个ID的控件。看源码:

在TabActivity类中有么个方法:

  1. @Override
  2. public void onContentChanged() {
  3. super.onContentChanged();
  4. mTabHost = (TabHost) findViewById(com.android.internal.R.id.tabhost);
  5. if (mTabHost == null) {
  6. throw new RuntimeException(
  7. "Your content must have a TabHost whose id attribute is " +
  8. "'android.R.id.tabhost'");
  9. }
  10. mTabHost.setup(getLocalActivityManager());
  11. }

当内容发生改变时它会调用这个方法,来更新列表或者其他视图,而这个方法中需要实例化TabHost,所以必须通过ID为tabhost实例化。

再看看TabHost这个类中,

  1. public void setup() {
  2. mTabWidget = (TabWidget) findViewById(com.android.internal.R.id.tabs);
  3. if (mTabWidget == null) {
  4. throw new RuntimeException(
  5. "Your TabHost must have a TabWidget whose id attribute is 'android.R.id.tabs'");
  6. }
  7. // KeyListener to attach to all tabs. Detects non-navigation keys
  8. // and relays them to the tab content.
  9. mTabKeyListener = new OnKeyListener() {
  10. public boolean onKey(View v, int keyCode, KeyEvent event) {
  11. switch (keyCode) {
  12. case KeyEvent.KEYCODE_DPAD_CENTER:
  13. case KeyEvent.KEYCODE_DPAD_LEFT:
  14. case KeyEvent.KEYCODE_DPAD_RIGHT:
  15. case KeyEvent.KEYCODE_DPAD_UP:
  16. case KeyEvent.KEYCODE_DPAD_DOWN:
  17. case KeyEvent.KEYCODE_ENTER:
  18. return false;
  19. }
  20. mTabContent.requestFocus(View.FOCUS_FORWARD);
  21. return mTabContent.dispatchKeyEvent(event);
  22. }
  23. };
  24. mTabWidget.setTabSelectionListener(new TabWidget.OnTabSelectionChanged() {
  25. public void onTabSelectionChanged(int tabIndex, boolean clicked) {
  26. setCurrentTab(tabIndex);
  27. if (clicked) {
  28. mTabContent.requestFocus(View.FOCUS_FORWARD);
  29. }
  30. }
  31. });
  32. mTabContent = (FrameLayout) findViewById(com.android.internal.R.id.tabcontent);
  33. if (mTabContent == null) {
  34. throw new RuntimeException(
  35. "Your TabHost must have a FrameLayout whose id attribute is "
  36. + "'android.R.id.tabcontent'");
  37. }
  38. }

这个方法,是在增加选项卡之前由系统调用。在这个方法中需要通过tabs 这个ID实例化一个TabWidget,通过tabcontent这个ID实例化一个FrameLayout,用来放置选项卡内容。所以这两个ID也是固定的。

在上述布局文件中隐藏了系统默认的Widget,取而代之的是带有图片的Button。

看一下主要代码:

  1. package com.iteye.androidtoast;
  2. import android.app.TabActivity;
  3. import android.content.Intent;
  4. import android.os.Bundle;
  5. import android.widget.RadioGroup;
  6. import android.widget.RadioGroup.OnCheckedChangeListener;
  7. import android.widget.TabHost;
  8. public class MainActivity extends TabActivity implements OnCheckedChangeListener{
  9. /** Called when the activity is first created. */
  10. private TabHost mHost;
  11. private RadioGroup radioderGroup;
  12. @Override
  13. public void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.maintabs);
  16. //实例化TabHost
  17. mHost=this.getTabHost();
  18. //添加选项卡
  19. mHost.addTab(mHost.newTabSpec("ONE").setIndicator("ONE")
  20. .setContent(new Intent(this,OneActivity.class)));
  21. mHost.addTab(mHost.newTabSpec("TWO").setIndicator("TWO")
  22. .setContent(new Intent(this,TwoActivity.class)));
  23. mHost.addTab(mHost.newTabSpec("THREE").setIndicator("THREE")
  24. .setContent(new Intent(this,ThreeActivity.class)));
  25. mHost.addTab(mHost.newTabSpec("FOUR").setIndicator("FOUR")
  26. .setContent(new Intent(this,FourActivity.class)));
  27. mHost.addTab(mHost.newTabSpec("FIVE").setIndicator("FIVE")
  28. .setContent(new Intent(this,FiveActivity.class)));
  29. radioderGroup = (RadioGroup) findViewById(R.id.main_radio);
  30. radioderGroup.setOnCheckedChangeListener(this);
  31. }
  32. @Override
  33. public void onCheckedChanged(RadioGroup group, int checkedId) {
  34. switch(checkedId){
  35. case R.id.radio_button0:
  36. mHost.setCurrentTabByTag("ONE");
  37. break;
  38. case R.id.radio_button1:
  39. mHost.setCurrentTabByTag("TWO");
  40. break;
  41. case R.id.radio_button2:
  42. mHost.setCurrentTabByTag("THREE");
  43. break;
  44. case R.id.radio_button3:
  45. mHost.setCurrentTabByTag("FOUR");
  46. break;
  47. case R.id.radio_button4:
  48. mHost.setCurrentTabByTag("FIVE");
  49. break;
  50. }
  51. }
  52. }

转载:http://androidtoast.iteye.com/blog/1187274

android中TabHost和RadioGroup的更多相关文章

  1. Android中TabHost中实现标签的滚动以及一些TabHost开发的奇怪问题

    最近在使用TabHost的时候遇到了一些奇怪的问题,在这里总结分享备忘一下. 首先说一点TabActivity将会被FragmentActivity所替代,但是本文中却是使用的TabActivity. ...

  2. Android中TabHost嵌套TabHost

    在嵌套TabHost时,先后遇到了以下情况: 问题1:内部TabHos无显示,只显示了其中的一个Activity: 解决:按下文比对主子TabHos的布局文件和java文件并修改: 问题2:如上所做后 ...

  3. 【转】Android中dip(dp)与px之间单位转换

    Android中dip(dp)与px之间单位转换 dp这个单位可能对web开发的人比较陌生,因为一般都是使用px(像素)但是,现在在开始android应用和游戏后,基本上都转换成用dp作用为单位了,因 ...

  4. Android中RadioGroup的初始化和简单的使用

    一简介: RadioGroup作为一个单选按钮组,可以设置为性别选择男或则女,地址选择等等,作为一个android入门级选手,就简单的说一下RadioGroup组中RadioButton的布局和初始化 ...

  5. Android中全屏 取消标题栏,TabHost中设置NoTitleBar的三种方法(转)

    Android中全屏 取消标题栏,TabHost中设置NoTitleBar的三种方法http://www.cnblogs.com/zdz8207/archive/2013/02/27/android- ...

  6. Android中的TabHost

    TabHost是整个Tab的容器,包含TabWidget和FrameLayout两个部分,TabWidget是每个Tab的表情,FrameLayout是Tab内容. 实现方法:继承TabActivit ...

  7. android的tabhost+RadioGroup+PopupWindow

    根据网上的代码稍作修改了下,放着记录学习. 效果图如下: 主代码如下: package com.andyidea.tabdemo; import android.app.TabActivity; im ...

  8. Android 中常见控件的介绍和使用

    1 TextView文本框 1.1 TextView类的结构 TextView 是用于显示字符串的组件,对于用户来说就是屏幕中一块用于显示文本的区域.TextView类的层次关系如下: java.la ...

  9. android学习--TabHost选项卡组件

    TabHost是一种非常有用的组件,TabHost能够非常方便地在窗体上放置多个标签页,每一个标签页获得了一个与外部容器同样大小的组件摆放区域.在手机系统的应用类似"未接电话".& ...

随机推荐

  1. Spark(十) -- Spark Streaming API编程

    本文测试的Spark版本是1.3.1 Spark Streaming编程模型: 第一步: 需要一个StreamingContext对象,该对象是Spark Streaming操作的入口 ,而构建一个S ...

  2. html DOM 的继承关系

    零散的知识聚合在一起,就会形成力量,就有了生命力. 如各种语言的开发框架, 都是右各个碎片化的功能聚合在一起,构成有机地整体,便有了强大的力量.will be powerful! 如: jquery ...

  3. leetcode 二分查找 Search in Rotated Sorted Array

    Search in Rotated Sorted Array Total Accepted: 28132 Total Submissions: 98526My Submissions Suppose ...

  4. webpack入门学习总结

    <h5概述</h5> webpack是当下最热门的前端资源模块化管理和打包工具.它可以将许多松散的模块按照依赖和规则打包成符合生产环境部署的前端资源.还可以将按需加载的模块进行代码分 ...

  5. Django——Django,uwsgi Nginx组合建站

    近来抽时间看了点关于python的资料,试着架设了下运行环境,于是写了本文记录之. 用到的软件: Django-1.6.1 uwsgi-2.0.7 python-3.3.3 nginx-1.7.7 安 ...

  6. 豆瓣电台笔记3:cell添加从中间向两侧放大的动画

    步骤: 1.设置动画属性的初始值 cell.layer.transform = CATransform3DMakeScale(0.1,0.1,1) 2.在指定时间内执行动画 UIView.animat ...

  7. 基于iOS 10、realm封装的下载器

    代码地址如下:http://www.demodashi.com/demo/11653.html 概要 在决定自己封装一个下载器前,我本以为没有那么复杂,可在实际开发过程中困难重重,再加上iOS10和X ...

  8. MongoDB笔记(二):MongoDB下Shell的基本操作

    一.mongoDB与关系型数据库对比 对比项              mongoDB             关系型数据库(oracle.mysql) 表 集合List 二维表table 表的一行数 ...

  9. android-异步消息处理机制初步

    Android的异步消息处理主要由4个部分组成,Message.Handler.MessageQueue和Looper Message:在线程之间传递的消息,它可以在内部携带少量的信息,用于在不同线程 ...

  10. Quartz的cron表达式 (spring定时器 crontab)

    http://tangshuo.iteye.com/blog/184824 表达式位数最少六位,如每天凌晨一点启动:"0 0 1 * * ?" 顺序按   秒 分 时 日期 月份 ...