最近做了一个TabHost的界面,在做的过程中发现了一些问题,故和大家分享一下。

首先我的界面如下:

目前就我所知,创建TabHost有两种方式,第一种是继承TabActivity类,然后用getTabHost方法来得到一个TabHost的实例,然后就可以给这个TabHost添加Tab了。示例代码如下:

  1. public class PlotHost extends TabActivity  {
  2. @Override
  3. protected void onCreate(Bundle savedInstanceState) {
  4. // TODO Auto-generated method stub
  5. super.onCreate(savedInstanceState);
  6. //      setContentView(R.layout.activity_main);
  7. setContentView(R.layout.tabhost);
  8. Resources res=getResources();
  9. TabHost tabhost=getTabHost();
  10. TabSpec dynamicplotSpec;
  11. TabSpec staticplotSpec;
  12. Intent dynamicplotIntent;
  13. Intent staticplotIntent;
  14. View dynamicplotView;
  15. View staticplotView;
  16. dynamicplotIntent=new Intent(this,Tab1.class);
  17. dynamicplotView=new TabView(this, R.drawable.dynamicploto, R.drawable.dynamicplots, "动态曲线");
  18. dynamicplotSpec=tabhost.newTabSpec("DynamicplotTab");
  19. dynamicplotSpec.setIndicator(dynamicplotView).setContent(dynamicplotIntent);
  20. tabhost.addTab(dynamicplotSpec);
  21. staticplotSpec=tabhost.newTabSpec("StaticplotTab");
  22. staticplotView=new TabView(this, R.drawable.staticploto, R.drawable.staticplots, "历史曲线");
  23. staticplotIntent=new Intent(this,Tab2.class);
  24. staticplotSpec.setIndicator(staticplotView).setContent(staticplotIntent);
  25. tabhost.addTab(staticplotSpec);
  26. tabhost.setCurrentTab(0);
  27. }
  28. //自定义一个tab选项卡显示样式
  29. private  class TabView extends LinearLayout {
  30. ImageView imageView ;
  31. public TabView(Context c, int drawableSelected, int drawableUnselected,String label) {
  32. super(c);
  33. this.setOrientation(VERTICAL);
  34. imageView = new ImageView(c);
  35. StateListDrawable listDrawable = new StateListDrawable();
  36. listDrawable.addState(SELECTED_STATE_SET, this.getResources()
  37. .getDrawable(drawableSelected));
  38. listDrawable.addState(ENABLED_STATE_SET, this.getResources()
  39. .getDrawable(drawableUnselected));
  40. imageView.setImageDrawable(listDrawable);
  41. imageView.setBackgroundColor(Color.TRANSPARENT);
  42. setGravity(Gravity.CENTER);
  43. addView(imageView);
  44. TextView tv_label=new TextView(c);
  45. tv_label.setText(label);
  46. tv_label.setGravity(Gravity.CENTER);
  47. addView(tv_label);
  48. }
  49. }
  50. }

具体代码我就不用多分析,只需要提一点的就是,用这种方式根本就不用你自定义一个TabHost的组建,getTabHost方法会自动调用系统默认的布局来帮助你进行显示,所以代码里面使用setContentView是多余的。

第二种方式问题就比较多了,其中网上大部分都说直接集成Activity,然后使用findViewbyId找到自定义的TabHost的组件,最
后对这个TabHost调用setup方法即可。但是不知道是何原因,也许是我的API版本比较高的原因,使用这种方法始终没有成功,logcat里面报
java.lang.RuntimeException异常。最后google后发现,需要改一下方式,即继承ActivityGroup,然后最后
setup的时候需要使用setup(this.getLocalActivityManager)即可,代码如下:

  1. public class PlotHost extends ActivityGroup  {
  2. @Override
  3. protected void onCreate(Bundle savedInstanceState) {
  4. // TODO Auto-generated method stub
  5. super.onCreate(savedInstanceState);
  6. //      setContentView(R.layout.activity_main);
  7. setContentView(R.layout.tabhost);
  8. Resources res=getResources();
  9. TabHost tabhost=(TabHost)findViewById(R.id.tabhost);
  10. tabhost.setup(this.getLocalActivityManager());
  11. TabSpec dynamicplotSpec;
  12. TabSpec staticplotSpec;
  13. Intent dynamicplotIntent;
  14. Intent staticplotIntent;
  15. View dynamicplotView;
  16. View staticplotView;
  17. dynamicplotIntent=new Intent(this,Tab1.class);
  18. dynamicplotView=new TabView(this, R.drawable.dynamicploto, R.drawable.dynamicplots, "动态曲线");
  19. dynamicplotSpec=tabhost.newTabSpec("DynamicplotTab");
  20. dynamicplotSpec.setIndicator(dynamicplotView).setContent(dynamicplotIntent);
  21. tabhost.addTab(dynamicplotSpec);
  22. staticplotSpec=tabhost.newTabSpec("StaticplotTab");
  23. staticplotView=new TabView(this, R.drawable.staticploto, R.drawable.staticplots, "历史曲线");
  24. staticplotIntent=new Intent(this,Tab2.class);
  25. staticplotSpec.setIndicator(staticplotView).setContent(staticplotIntent);
  26. tabhost.addTab(staticplotSpec);
  27. tabhost.setCurrentTab(0);
  28. }
  29. //自定义一个tab选项卡显示样式
  30. private  class TabView extends LinearLayout {
  31. ImageView imageView ;
  32. public TabView(Context c, int drawableSelected, int drawableUnselected,String label) {
  33. super(c);
  34. this.setOrientation(VERTICAL);
  35. imageView = new ImageView(c);
  36. StateListDrawable listDrawable = new StateListDrawable();
  37. listDrawable.addState(SELECTED_STATE_SET, this.getResources()
  38. .getDrawable(drawableSelected));
  39. listDrawable.addState(ENABLED_STATE_SET, this.getResources()
  40. .getDrawable(drawableUnselected));
  41. imageView.setImageDrawable(listDrawable);
  42. imageView.setBackgroundColor(Color.TRANSPARENT);
  43. setGravity(Gravity.CENTER);
  44. addView(imageView);
  45. TextView tv_label=new TextView(c);
  46. tv_label.setText(label);
  47. tv_label.setGravity(Gravity.CENTER);
  48. addView(tv_label);
  49. }
  50. }
  51. }

其实内容大同小异,至于为什么要这么做,貌似是如果直接调用TabHost的setup方法,不能实例化它的TabWidget和TabContent对象,需要借助于LocalActivityManager自动对二者进行实例化。因为看了一个老兄的博客,setup主要完成的功能便是实例化它的TabWidget和TabContent。如下:

mTabWidget = (TabWidget) findViewById(com.android.internal.R.id.tabs);
        if (mTabWidget == null) {
            throw new RuntimeException(
                    "Your TabHost must have a TabWidget whose id attribute is 'android.R.id.tabs'");
        }

mTabContent = (FrameLayout) findViewById(com.android.internal.R.id.tabcontent);
        if (mTabContent == null) {
            throw new RuntimeException(
                    "Your TabHost must have a FrameLayout whose id attribute is 'android.R.id.tabcontent'");
        }

同时,还要补充,使用第二种方法的时候注意修改TabHost在布局当中的ID格式为“@+id/xxx”,其中xxx自定义。其他的步骤在网上一搜一大堆我就不唠叨了。

创建TabHost的两种方式的简单分析的更多相关文章

  1. 【java并发】传统线程技术中创建线程的两种方式

    传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法:二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread.这两种方式 ...

  2. Java并发基础01. 传统线程技术中创建线程的两种方式

    传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法:二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread.这两种方式 ...

  3. 创建线程的两种方式比较Thread VS Runnable

    1.首先来说说创建线程的两种方式 一种方式是继承Thread类,并重写run()方法 public class MyThread extends Thread{ @Override public vo ...

  4. ubuntu创建用户的两种方式

    ubuntu创建用户有两种方式: useradd和adduser 这两者,就像零件与产品的关系.useradd是DIY,需要自己调配,adduser是品牌机,拿来就能用. 对于创建一般用户来讲,use ...

  5. Spring创建JobDetail的两种方式

    一.Spring创建JobDetail的两种方式 二.整合方式一示例步骤 1.将spring核心jar包.quartz.jar和Spring-context-support.jar导入类路径. 2.编 ...

  6. 用idea简单创建web项目——两种方式

    最近同学让我教她们用idea创建web项目,于是我用两种方式创建web项目,并整理截图给她们看,一种是用maven创建,一种是不用maven创建,适合菜鸟哈哈~ 方法一:不用maven 1.解压tom ...

  7. Easyui 创建dialog的两种方式,以及他们带来的问题

    $('#yy').dialog('open');//打开dialog 这地方要注意,加入你关闭窗口的地方使用$('#yy').dialog('destroy');那么你这个dialog就只能使用一次, ...

  8. Java值创建线程的两种方式对比

    在Java中创建线程的方式有两种,第一种是直接继承Thead类,另一种是实现Runable接口.那么这两种方式孰优孰劣呢? 采用继承Thead类实现多线程: 优势:编写简单,如果需要访问当前线程,只需 ...

  9. k8s 创建资源的两种方式 - 每天5分钟玩转 Docker 容器技术(124)

    命令 vs 配置文件 Kubernetes 支持两种方式创建资源: 1. 用 kubectl 命令直接创建,比如: kubectl run nginx-deployment --image=nginx ...

随机推荐

  1. FineUI Grid控件右键菜单的实现

    FineUI官方Demo上一直没有Grid右键菜单的实现,其实从4.1.x的版本开始,允许添加自定义的事件监听(Listeners),所以要实现这个功能已经相当容易了. ExtJs右键菜单有很多种,对 ...

  2. html+css学习笔记 5[表格、表单]

    表格 -- 默认样式重置 表格标签:     table 表格     thead 表格头     tbody 表格主体     tfoot 表格尾     tr 表格行     th 元素定义表头 ...

  3. DevExpress控件使用系列--ASPxGridView+Popup+Tab

      1.控件功能     列表控件展示数据.弹框控件执行编辑操作.Tab控件实现多标签编辑操官方说明 2.官方示例       2.1 ASPxGridView                http ...

  4. Matlab中mat2cell的使用

    怎样用mat2cell将一个100*100的矩阵分成10个10*100的矩阵? 根据帮助中 c = mat2cell(x,m,n)应该这样写 mat2cell(x,[10 10 10 10 10 10 ...

  5. 16进制串hex与ASCII字符串相互转换

    提供两个函数,方便十六进制串与ASCII 字符串之间的相互转换,使用函数需要注意的是返回的串是在堆上通过 calloc 分配的,所以,记得使用完返回值释放该块,并且将指向该块的指针 =NULL . c ...

  6. phonegap file操作

    phonegap中,有时候需要操作到手机中的文件,这个时候就需要用到phonegap官方提供的插件 file ,英文好的可以直接参考官方文档 首先是安装插件:(需要phonegap 3.0 以上,不止 ...

  7. ios开发小技巧之摇一摇截屏

    1. 监控摇一摇动作 1> 让当前视图控制器成为第一响应者 // 必须先让当前视图控制器成为第一响应者才能响应动作时间 [self becomeFirstResponder]; 2> 实现 ...

  8. RedHat Linux下注册Apache为系统服务并设为开机启动

    1.系统环境: 操作系统:Red Hat Enterprise Linux Server release 5.4 Apache版本:httpd-2.2.19 2.注册服务 #将apachectl复制到 ...

  9. 【WCF--初入江湖】目录

    [WCF--初入江湖]目录 [WCF--初入江湖]01 WCF编程概述 [WCF--初入江湖]02 WCF契约 [WCF--初入江湖]03 配置服务 [WCF--初入江湖]04 WCF通信模式 [WC ...

  10. 【面试题015】链表中倒数第k个结点

    [面试题015]链表中倒数第k个结点    可以用两个指针,当第一个指针指向了第k个时候,第二个指针让他指向链表的第一个元素,然后这两个指针同时向后面移动, 当第一个指针移动到末尾的时候,第二个指针指 ...