布局学习——妙用TabHost
前言
为了更好的开发Android应用程序,除了熟练掌握基本的UI组件和API外,还需要掌握一些技巧,而这些技巧可以通过阅读一些代码来提高,本系列将与大家分享一些新浪微博布局方面的收获,欢迎交流!
声明
欢迎转载,但请保留文章原始出处:)
博客园:http://www.cnblogs.com
农民伯伯: http://www.cnblogs.com/over140
版本
新浪微博 weibo_10235010.apk
正文
一、效果图
红色部分是本文要实现的目标。
二、实现
maintabs.xml
<TabHostandroid:id="@android:id/tabhost"android:layout_width="fill_parent"android:layout_height="fill_parent"
xmlns:android="http://schemas.android.com/apk/res/android">
<LinearLayoutandroid:orientation="vertical"android:layout_width="fill_parent"android:layout_height="fill_parent">
<FrameLayoutandroid:id="@android:id/tabcontent"android:layout_width="fill_parent"android:layout_height="0.0dip"android:layout_weight="1.0"/>
<TabWidgetandroid:id="@android:id/tabs"android:visibility="gone"android:layout_width="fill_parent"android:layout_height="wrap_content"android:layout_weight="0.0"/>
<RadioGroupandroid: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">
<RadioButtonandroid: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"/>
<RadioButtonandroid: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"/>
<RadioButtonandroid: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"/>
<RadioButtonandroid: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"/>
<RadioButtonandroid: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
<itemname="android:textSize">@dimen/bottom_tab_font_size</item>
<itemname="android:textColor">#ffffffff</item>
<itemname="android:ellipsize">marquee</item>
<itemname="android:gravity">center_horizontal</item>
<itemname="android:background">@drawable/home_btn_bg</item>
<itemname="android:paddingTop">@dimen/bottom_tab_padding_up</item>
<itemname="android:layout_width">fill_parent</item>
<itemname="android:layout_height">wrap_content</item>
<itemname="android:button">@null</item>
<itemname="android:singleLine">true</item>
<itemname="android:drawablePadding">@dimen/bottom_tab_padding_drawable</item>
<itemname="android:layout_weight">1.0</item>
</style>home_btn_bg.xml
xmlns:android="http://schemas.android.com/apk/res/android">
<itemandroid:state_focused="true"android:state_enabled="true"android:state_pressed="false"android:drawable="@drawable/home_btn_bg_s"/>
<itemandroid:state_enabled="true"android:state_pressed="true"android:drawable="@drawable/home_btn_bg_s"/>
<itemandroid:state_enabled="true"android:state_checked="true"android:drawable="@drawable/home_btn_bg_d"/>
<itemandroid:drawable="@drawable/transparent"/>
</selector>代码说明:
1. 需要注意的是他这里把TabWidget的Visibility设置成了gone!也就是默认难看的风格不见了:,取而代之的是5个带风格的单选按钮.
2. 注意为单选按钮设置的style,其中最重要的是为其background设置了home_btn_bg.xml,也就是自定义了选中效果。
Java文件
OnCheckedChangeListener{
privateTabHostmHost;
privateIntentmMBlogIntent;
privateIntentmMoreIntent;
privateIntentmInfoIntent;
privateIntentmSearchIntent;
privateIntentmUserInfoIntent;
@Override
protectedvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.maintabs);
//~~~~~~~~~~~~初始化
this.mMBlogIntent=newIntent(this,HomeListActivity.class);
this.mSearchIntent=newIntent(this,SearchSquareActivity.class);
this.mInfoIntent=newIntent(this,MessageGroup.class);
this.mUserInfoIntent=newIntent(this,MyInfoActivity.class);
this.mMoreIntent=newIntent(this,MoreItemsActivity.class);
initRadios();
setupIntent();
}
/**
*初始化底部按钮
*/
privatevoidinitRadios(){
((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
publicvoidonCheckedChanged(CompoundButtonbuttonView,booleanisChecked){
if(isChecked){
switch(buttonView.getId()){
caseR.id.radio_button0:
this.mHost.setCurrentTabByTag("mblog_tab");
break;
caseR.id.radio_button1:
this.mHost.setCurrentTabByTag("message_tab");
break;
caseR.id.radio_button2:
this.mHost.setCurrentTabByTag("userinfo_tab");
break;
caseR.id.radio_button3:
this.mHost.setCurrentTabByTag("search_tab");
break;
caseR.id.radio_button4:
this.mHost.setCurrentTabByTag("more_tab");
break;
}
}
}
privatevoidsetupIntent(){
this.mHost=getTabHost();
TabHostlocalTabHost=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));
}
privateTabHost.TabSpecbuildTabSpec(Stringtag,intresLabel,intresIcon,
finalIntentcontent){
returnthis.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,各取其长,有时间可以专门写一个这样的自定义控件:)
四、相关文章
[Android]使用ActivityGroup来切换Activity和Layout
结束
本文中使用的资源均反编译自apk文件,这里主要是讲思路,欢迎大家交流。