新浪微博布局学习——妙用TabHost

简介:

正文

  一、效果图

 

 

红色部分是本文要实现的目标。

 

  二、实现

    maintabs.xml

<? xml version="1.0" encoding="UTF-8" ?>
< TabHost  android:id ="@android:id/tabhost"  android:layout_width ="fill_parent"  android:layout_height ="fill_parent"
  xmlns:android
="http://schemas.android.com/apk/res/android" >
    
< LinearLayout  android:orientation ="vertical"  android:layout_width ="fill_parent"  android:layout_height ="fill_parent" >
        
< FrameLayout  android:id ="@android:id/tabcontent"  android:layout_width ="fill_parent"  android:layout_height ="0.0dip"  android:layout_weight ="1.0"   />
        
< TabWidget  android:id ="@android:id/tabs"  android:visibility ="gone"  android:layout_width ="fill_parent"  android:layout_height ="wrap_content"  android:layout_weight ="0.0"   />
        
< RadioGroup  android:gravity ="center_vertical"  android:layout_gravity ="bottom"  android:orientation ="horizontal"  android:id ="@id/main_radio"  android:background ="@drawable/maintab_toolbar_bg"  android:layout_width ="fill_parent"  android:layout_height ="wrap_content" >
            
< RadioButton    android:text ="@string/main_home"  android:checked ="true"  android:id ="@+id/radio_button0"  android:layout_marginTop ="2.0dip"  android:drawableTop ="@drawable/icon_1_n"  style ="@style/main_tab_bottom"   />
            
< RadioButton  android:id ="@+id/radio_button1"  android:layout_marginTop ="2.0dip"  android:text ="@string/main_news"  android:drawableTop ="@drawable/icon_2_n"  style ="@style/main_tab_bottom"   />
            
< RadioButton  android:id ="@+id/radio_button2"  android:layout_marginTop ="2.0dip"  android:text ="@string/main_my_info"  android:drawableTop ="@drawable/icon_3_n"  style ="@style/main_tab_bottom"   />
            
< RadioButton  android:id ="@+id/radio_button3"  android:layout_marginTop ="2.0dip"  android:text ="@string/menu_search"  android:drawableTop ="@drawable/icon_4_n"  style ="@style/main_tab_bottom"   />
            
< RadioButton  android:id ="@+id/radio_button4"  android:layout_marginTop ="2.0dip"  android:text ="@string/more"  android:drawableTop ="@drawable/icon_5_n"  style ="@style/main_tab_bottom"   />
        
</ RadioGroup >
    
</ LinearLayout >
</ TabHost >

    styles.xml

     < style  name ="main_tab_bottom" >
        
< item  name ="android:textSize" > @dimen/bottom_tab_font_size </ item >
        
< item  name ="android:textColor" > #ffffffff </ item >
        
< item  name ="android:ellipsize" > marquee </ item >
        
< item  name ="android:gravity" > center_horizontal </ item >
        
< item  name ="android:background" > @drawable/home_btn_bg </ item >
        
< item  name ="android:paddingTop" > @dimen/bottom_tab_padding_up </ item >
        
< item  name ="android:layout_width" > fill_parent </ item >
        
< item  name ="android:layout_height" > wrap_content </ item >
        
< item  name ="android:button" > @null </ item >
        
< item  name ="android:singleLine" > true </ item >
        
< item  name ="android:drawablePadding" > @dimen/bottom_tab_padding_drawable </ item >
        
< item  name ="android:layout_weight" > 1.0 </ item >
    
</ style >

     home_btn_bg.xml

         < selector
          
xmlns:android ="http://schemas.android.com/apk/res/android" >
            
< item  android:state_focused ="true"  android:state_enabled ="true"  android:state_pressed ="false"  android:drawable ="@drawable/home_btn_bg_s"   />
            
< item  android:state_enabled ="true"  android:state_pressed ="true"  android:drawable ="@drawable/home_btn_bg_s"   />
            
< item  android:state_enabled ="true"  android:state_checked ="true"  android:drawable ="@drawable/home_btn_bg_d"   />
            
< item  android:drawable ="@drawable/transparent"   />
        
</ selector >

    代码说明:

        1.  需要注意的是他这里把TabWidget的Visibility设置成了gone!也就是默认难看的风格不见了:,取而代之的是5个带风格的单选按钮.

        2.  注意为单选按钮设置的style,其中最重要的是为其background设置了home_btn_bg.xml,也就是自定义了选中效果。

    Java文件

public   class  MainTabActivity  extends  TabActivity  implements
        OnCheckedChangeListener {

    
private  TabHost mHost;
    
private  Intent mMBlogIntent;
    
private  Intent mMoreIntent;
    
private  Intent mInfoIntent;
    
private  Intent mSearchIntent;
    
private  Intent mUserInfoIntent;

    @Override
    
protected   void  onCreate(Bundle savedInstanceState) {
        
super .onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.maintabs);

        
//  ~~~~~~~~~~~~ 初始化
         this .mMBlogIntent  =   new  Intent( this , HomeListActivity. class );
        
this .mSearchIntent  =   new  Intent( this , SearchSquareActivity. class );
        
this .mInfoIntent  =   new  Intent( this , MessageGroup. class );
        
this .mUserInfoIntent  =   new  Intent( this , MyInfoActivity. class );
        
this .mMoreIntent  =   new  Intent( this , MoreItemsActivity. class );

        initRadios();
        
        setupIntent();
    }

    
/**
     * 初始化底部按钮
     
*/
    
private   void  initRadios() {
         ((RadioButton) findViewById(R.id.radio_button0)).setOnCheckedChangeListener(
this );
         ((RadioButton) findViewById(R.id.radio_button1)).setOnCheckedChangeListener(
this );
         ((RadioButton) findViewById(R.id.radio_button2)).setOnCheckedChangeListener(
this );
         ((RadioButton) findViewById(R.id.radio_button3)).setOnCheckedChangeListener(
this );
         ((RadioButton) findViewById(R.id.radio_button4)).setOnCheckedChangeListener(
this );
    }

    
/**
     * 切换模块
     
*/
    @Override
    
public   void  onCheckedChanged(CompoundButton buttonView,  boolean  isChecked) {
        
if  (isChecked) {
            
switch  (buttonView.getId()) {
            
case  R.id.radio_button0:
                
this .mHost.setCurrentTabByTag( " mblog_tab " );
                
break ;
            
case  R.id.radio_button1:
                
this .mHost.setCurrentTabByTag( " message_tab " );
                
break ;
            
case  R.id.radio_button2:
                
this .mHost.setCurrentTabByTag( " userinfo_tab " );
                
break ;
            
case  R.id.radio_button3:
                
this .mHost.setCurrentTabByTag( " search_tab " );
                
break ;
            
case  R.id.radio_button4:
                
this .mHost.setCurrentTabByTag( " more_tab " );
                
break ;
            }
        }
    }

    
private   void  setupIntent() {
        
this .mHost  =  getTabHost();
        TabHost localTabHost 
=   this .mHost;

        localTabHost.addTab(buildTabSpec(
" mblog_tab " , R.string.main_home,
                R.drawable.icon_1_n, 
this .mMBlogIntent));

        localTabHost.addTab(buildTabSpec(
" message_tab " , R.string.main_news,
                R.drawable.icon_2_n, 
this .mInfoIntent));

        localTabHost.addTab(buildTabSpec(
" userinfo_tab " , R.string.main_my_info,
                R.drawable.icon_3_n, 
this .mUserInfoIntent));

        localTabHost.addTab(buildTabSpec(
" search_tab " , R.string.menu_search,
                R.drawable.icon_4_n, 
this .mSearchIntent));

        localTabHost.addTab(buildTabSpec(
" more_tab " , R.string.more,
                R.drawable.icon_5_n, 
this .mMoreIntent));

    }

    
private  TabHost.TabSpec buildTabSpec(String tag,  int  resLabel,  int  resIcon,
            
final  Intent content) {
        
return   this .mHost
                .newTabSpec(tag)
                .setIndicator(getString(resLabel),
                        getResources().getDrawable(resIcon))
                .setContent(content);
    }

    代码说明

      1.  由于TabWidget被隐藏,所以相关的事件也会无效,这里取巧用RadioGroup与RadioButton的特性来处理切换,然后监听事件调用setCurrentTabByTag来切换Activity。

      2.  注意即使TabWidget被隐藏,也要为其设置indicator,否则会保持。

 

  三、总结
 

    在这之前如果要做这种效果我恐怕第一时间就会想到用ActivityGroup来做,主要是因为TabHost的TabWidget非常难看,用起来也不方便。其实从源码可以看出,TabActivity也是继承自ActivityGroup,这里结合了单选按钮和TabHost,各取其长,有时间可以专门写一个这样的自定义控件:)



本文转自over140 51CTO博客,原文链接:http://blog.51cto.com/over140/582149,如需转载请自行联系原作者

相关文章
|
6月前
|
Android开发 容器
Android开发,学习LinearLayout布局
Android开发,学习LinearLayout布局
74 0
|
JSON iOS开发 数据格式
iOS开发 - 关于启动页动画的杂谈
iOS开发 - 关于启动页动画的杂谈
259 0
iOS开发 - 关于启动页动画的杂谈
|
Android开发
【Android教程】01 布局
【Android教程】01 布局
138 0
【Android教程】01 布局
|
XML Java Android开发
Android开发布局 案例二
Android开发布局 案例二
209 0
Android开发布局 案例二
|
Android开发
Android开发布局 案例一
Android开发布局 案例一
123 0
Android开发布局 案例一
|
移动开发 Dart
【新年快乐第二弹】在 Flutter 中使用交错网格视图创建瀑布流布局
马上过新年了,想好如何过年了吗?,今天我带大家在瀑布流布局中写新年快乐。 在 Web 和移动开发世界中,当我们想要显示大小不相同的项目网格时,瀑布流布局很有用。一个轴使用严格的网格布局,通常是列。在另一个轴上,项目具有不同的高度,但可以灵活排列以填满可用空间。使用瀑布流布局的一个著名例子是 Pinterest。他们为他们的网站和移动应用程序实现了这种布局,以显示不同大小的图像。
313 0
|
XML Android开发 数据格式
【Android开发学习笔记之一】5大布局方式详解
Android中常用的5大布局方式有以下几种: 线性布局(LinearLayout):按照垂直或者水平方向布局的组件。 帧布局(FrameLayout):组件从屏幕左上方布局组件。 表格布局(TableLayout):按照行列方式布局组件。
1305 0