这里就说下tablayout+viewpager的实现方式;tablayout是android5.0推出来的一个MaterialDesign风格的控件,是专门用来实现tab栏效果的;功能强大,使用方便灵活;

一、引入依赖库

使用非常方便,Android Studio只需要在gradle中引入即可使用 .

apply plugin: 'com.android.application'

android {
compileSdkVersion 28
defaultConfig {
applicationId "com.example.panzq.tablayout"
minSdkVersion 22
targetSdkVersion 28
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
} dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'com.android.support:appcompat-v7:28.0.0'
implementation 'com.android.support.constraint:constraint-layout:1.1.3'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
implementation 'com.android.support:design:28.0.0'
implementation 'com.android.support:recyclerview-v7:28.0.0'
implementation 'com.android.support:support-v4:28.0.0'
}

二、layout中定义TabLayout控件及ViewPager控件

<!-- 
app:tabGravity="center" 对齐方式,可选fill和center
app:tabIndicatorColor="@color/colorAccent" 设置tab下划线的颜色
app:tabMode="scrollable" scrollable是可以横行滚动,fixed是指固定个数
app:tabSelectedTextColor="@color/colorPrimaryDark" 选择tab的文本颜色
app:tabTextColor="@color/colorPrimary" 普通tab字体颜色

app:tabIndicatorHeight 指示器高度
 app:tabBackground tab背景颜色
 app:tabMaxWidth tab栏最大宽度
 app:tabTextAppearance tab栏字体样式
 app:tabMinWidth tab栏最小宽度

-->
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/linear"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity"> <android.support.v4.view.ViewPager
android:id="@+id/viewPager"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_behavior="@string/appbar_scrolling_view_behavior"> <android.support.design.widget.TabLayout
android:id="@+id/tabLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:tabBackground="@color/colorPrimary"
app:tabGravity="fill"
app:tabMode="scrollable"
app:tabSelectedTextColor="@color/colorAccent"
app:tabIndicatorColor="@color/colorPrimaryLight"
app:tabTextColor="#ffffff"> <android.support.design.widget.TabItem
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:icon="@drawable/ic_call_black_24dp"
android:text="@string/recents" /> <android.support.design.widget.TabItem
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:icon="@drawable/ic_star_border_black_24dp"
android:text="@string/favourite" /> <android.support.design.widget.TabItem
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:icon="@drawable/ic_person_black_24dp"
android:text="@string/contacts" /> <android.support.design.widget.TabItem
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:icon="@drawable/ic_dialpad_black_24dp"
android:text="@string/keypad" /> <android.support.design.widget.TabItem
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:icon="@drawable/ic_voicemail_black_24dp"
android:text="@string/voicemail" />
</android.support.design.widget.TabLayout> </android.support.v4.view.ViewPager> </LinearLayout>

三 设置TabLayout和ViewPager关联

MainActivity

public class MainActivity extends AppCompatActivity {

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
TabLayout tabLayout = findViewById(R.id.tabLayout);
ViewPager viewPager = findViewById(R.id.viewPager); CallPagerAdapter callPagerAdapter = new CallPagerAdapter(getSupportFragmentManager());
callPagerAdapter.addFragment(new RecentCallFragment(), "RECENT");
callPagerAdapter.addFragment(new FavouriteCallFragment(), "FAVOURITE");
callPagerAdapter.addFragment(new ContactCallFragment(), "CONTACTS");
callPagerAdapter.addFragment(new KeypadCallFragment(), "KEYPAD");
callPagerAdapter.addFragment(new VoicemailCallFragment(), "VOICEMAIL");
viewPager.setAdapter(callPagerAdapter);
tabLayout.setupWithViewPager(viewPager);
tabLayout.getTabAt(0).setIcon(R.drawable.ic_call_black_24dp);
tabLayout.getTabAt(1).setIcon(R.drawable.ic_star_border_black_24dp);
tabLayout.getTabAt(2).setIcon(R.drawable.ic_person_black_24dp);
tabLayout.getTabAt(3).setIcon(R.drawable.ic_dialpad_black_24dp);
tabLayout.getTabAt(4).setIcon(R.drawable.ic_voicemail_black_24dp);
}
}

  需要注意的是setupWithViewPager();方法的调用必须在viewpager设置完适配器后调用,如果在设置适配器之前调用会抛异常,至于为什么会抛异常,后面tablayout的源码会说到;这样tab栏切换效果就实现了。

如果发现程序出现如下错误

- ::47.677 -/com.example.panzq.tablayout E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.example.panzq.tablayout, PID:
android.view.InflateException: Binary XML file line #: Binary XML file line #: Error inflating class android.support.design.button.MaterialButton
Caused by: android.view.InflateException: Binary XML file line #: Error inflating class android.support.design.button.MaterialButton
Caused by: java.lang.reflect.InvocationTargetException
at java.lang.reflect.Constructor.newInstance0(Native Method)
at java.lang.reflect.Constructor.newInstance(Constructor.java:)
at android.view.LayoutInflater.createView(LayoutInflater.java:)
at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:)
at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:)
at android.view.LayoutInflater.rInflate(LayoutInflater.java:)
at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:)
at android.view.LayoutInflater.inflate(LayoutInflater.java:)
at android.view.LayoutInflater.inflate(LayoutInflater.java:)
at com.example.panzq.tablayout.fragments.KeypadCallFragment.onCreateView(KeypadCallFragment.java:)
at android.support.v4.app.Fragment.performCreateView(Fragment.java:)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:)
at android.support.v4.app.FragmentManagerImpl.moveFragmentToExpectedState(FragmentManager.java:)
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:)
at android.support.v4.app.BackStackRecord.executeOps(BackStackRecord.java:)
at android.support.v4.app.FragmentManagerImpl.executeOps(FragmentManager.java:)
at android.support.v4.app.FragmentManagerImpl.executeOpsTogether(FragmentManager.java:)
at android.support.v4.app.FragmentManagerImpl.removeRedundantOperationsAndExecute(FragmentManager.java:)
at android.support.v4.app.FragmentManagerImpl.execSingleAction(FragmentManager.java:)
at android.support.v4.app.BackStackRecord.commitNowAllowingStateLoss(BackStackRecord.java:)
at android.support.v4.app.FragmentStatePagerAdapter.finishUpdate(FragmentStatePagerAdapter.java:)
at android.support.v4.view.ViewPager.populate(ViewPager.java:)
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:)
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:)
at android.support.v4.view.ViewPager.setCurrentItem(ViewPager.java:)
at android.support.design.widget.TabLayout$ViewPagerOnTabSelectedListener.onTabSelected(TabLayout.java:)
at android.support.design.widget.TabLayout.dispatchTabSelected(TabLayout.java:)
at android.support.design.widget.TabLayout.selectTab(TabLayout.java:)
at android.support.design.widget.TabLayout.selectTab(TabLayout.java:)
at android.support.design.widget.TabLayout$Tab.select(TabLayout.java:)
at android.support.design.widget.TabLayout$TabView.performClick(TabLayout.java:)
at android.view.View$PerformClick.run(View.java:)
at android.os.Handler.handleCallback(Handler.java:)
at android.os.Handler.dispatchMessage(Handler.java:)
at android.os.Looper.loop(Looper.java:)
at android.app.ActivityThread.main(ActivityThread.java:)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:)
Caused by: java.lang.IllegalArgumentException: This component requires that you specify a valid TextAppearance attribute. Update your app theme to inherit from Theme.MaterialComponents (or a descendant).
at android.support.design.internal.ThemeEnforcement.checkTextAppearance(ThemeEnforcement.java:)
at android.support.design.internal.ThemeEnforcement.obtainStyledAttributes(ThemeEnforcement.java:)
at android.support.design.button.MaterialButton.<init>(MaterialButton.java:)
at android.support.design.button.MaterialButton.<init>(MaterialButton.java:)
at java.lang.reflect.Constructor.newInstance0(Native Method) 
at java.lang.reflect.Constructor.newInstance(Constructor.java:) 
at android.view.LayoutInflater.createView(LayoutInflater.java:) 
at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:) 
at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:) 
at android.view.LayoutInflater.rInflate(LayoutInflater.java:) 
at android.view.LayoutInflater.rInflateChildren(LayoutInflater.java:) 
at android.view.LayoutInflater.inflate(LayoutInflater.java:) 
at android.view.LayoutInflater.inflate(LayoutInflater.java:) 
at com.example.panzq.tablayout.fragments.KeypadCallFragment.onCreateView(KeypadCallFragment.java:) 
at android.support.v4.app.Fragment.performCreateView(Fragment.java:) 
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:) 
at android.support.v4.app.FragmentManagerImpl.moveFragmentToExpectedState(FragmentManager.java:) 
at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:) 
at android.support.v4.app.BackStackRecord.executeOps(BackStackRecord.java:) 
at android.support.v4.app.FragmentManagerImpl.executeOps(FragmentManager.java:) 
at android.support.v4.app.FragmentManagerImpl.executeOpsTogether(FragmentManager.java:) 
at android.support.v4.app.FragmentManagerImpl.removeRedundantOperationsAndExecute(FragmentManager.java:) 
at android.support.v4.app.FragmentManagerImpl.execSingleAction(FragmentManager.java:) 
at android.support.v4.app.BackStackRecord.commitNowAllowingStateLoss(BackStackRecord.java:) 
at android.support.v4.app.FragmentStatePagerAdapter.finishUpdate(FragmentStatePagerAdapter.java:) 
at android.support.v4.view.ViewPager.populate(ViewPager.java:) 
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:) 
at android.support.v4.view.ViewPager.setCurrentItemInternal(ViewPager.java:) 
at android.support.v4.view.ViewPager.setCurrentItem(ViewPager.java:) 
at android.support.design.widget.TabLayout$ViewPagerOnTabSelectedListener.onTabSelected(TabLayout.java:) 
at android.support.design.widget.TabLayout.dispatchTabSelected(TabLayout.java:) 
at android.support.design.widget.TabLayout.selectTab(TabLayout.java:) 
at android.support.design.widget.TabLayout.selectTab(TabLayout.java:) 
at android.support.design.widget.TabLayout$Tab.select(TabLayout.java:) 
at android.support.design.widget.TabLayout$TabView.performClick(TabLayout.java:) 
at android.view.View$PerformClick.run(View.java:) 
at android.os.Handler.handleCallback(Handler.java:) 
at android.os.Handler.dispatchMessage(Handler.java:) 
at android.os.Looper.loop(Looper.java:) 
at android.app.ActivityThread.main(ActivityThread.java:) 
at java.lang.reflect.Method.invoke(Native Method) 
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:) 
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:) 

则需要修改style.xml文件

<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">

改为

<style name="AppTheme" parent="Theme.MaterialComponents.Light.DarkActionBar">

代码地址:https://github.com/MichealPan9999/TabLayout-ViewPager

TabLayout和ViewPager的更多相关文章

  1. TabLayout和ViewPager简单实现页卡的滑动

    首先需要在当前的module中的build Gradle的 dependencies中加入以下句子 compile 'com.android.support:design:23.0.1' 因为我们用到 ...

  2. TabLayout和ViewPager联动时的问题及解决方案

    问题概述 TabLayout搭配ViewPager关联使用时,在未调用TabLayout的setupWithViewPager(mViewPager)方法之前,ViewPager的内容和TabLayo ...

  3. 关于TabLayout与ViewPager在Fragment中嵌套Fragment使用或配合使用的思考

    注意: 因为继承的是Fragment,所以getSupportFragmentManager()与getFragmentManager()方法无法使用,这里需要用到getChildFragmentMa ...

  4. TabLayout与ViewPager同步后Tab的标题不显示

    一.概述 1.1 问题描述 TabLayout+ViewPager后,TabLayout的TabItem不显示的问题: 1.2 截图 二.结论 mTabs.setupWithViewPager(mVi ...

  5. TabLayout+Fragment+ViewPager+FragmentStatePagerAdapter实现Tab标签

    首先来看下实现的效果吧: 最近在项目中实现这个效果的时候.尽管自己磕磕绊绊的实现了,可是知识确实模模糊糊的,今天天气异常的冷,在加上这个知识不太熟练,实在是没有心情进行接下来的计划,干脆借着这个时间, ...

  6. Android 使用TabLayout、ViewPager和Fragment实现顶部菜单可滑动切换

    效果图如下 首先,要使用控件需要添加design library,在Android Studio中添加 compile 'com.android.support:design:23.4.0' 然后是布 ...

  7. 巧力避免ViewPager的预加载数据,Tablayout+Fragment+viewPager

    问题描述 最近在进行一个项目的开发,其中使用到了Tablayout+Fragment+viewPager来搭建一个基本的框架,从而出现了设置数据适配器的时候,item的位置错乱问题.我打印log日志的 ...

  8. TabLayout + ViewPager

    一.实现思路 1.在build.gradle中添加依赖,例如: compile 'com.android.support:support-v4:23.4.0'compile 'com.android. ...

  9. 使用FragmentTabHost+TabLayout+ViewPager实现双层嵌套Tab

    大多数应用程序都会在底部使用3~5个Tab对应用程序的主要功能进行划分,对于一些信息量非常大的应用程序,还需要在每个Tab下继续划分子Tab对信息进行分类显示. 本文实现采用FragmentTabHo ...

随机推荐

  1. Utterance-Wise Recurrent Dropout And Iterative Speaker Adaptation For Robust Monaural Speech Recognition

    单声道语音识别的逐句循环Dropout迭代说话人自适应     WRBN(wide residual BLSTM network,宽残差双向长短时记忆网络) [2] J. Heymann, L. Dr ...

  2. LOJ #2196「SDOI2014」LIS

    直接退流复杂度好优越啊 LOJ #2196 题意 一段数列,每个点有点权$ A_i$,删除代价$ B_i$,附加属性$ C_i$ 求最小代价使得$ LIS$长度发生变化,且输出一种$ C_i$字典序最 ...

  3. 人人中的 shiro权限管理 简单说明

    maven  shiro包的引用路径 :C:\Users\yanfazhongxin\.m2\repository\org\apache\shiro\shiro-core\1.3.2\shiro-co ...

  4. python基础-----函数/装饰器

    函数 在Python中,定义一个函数要使用def语句,依次写出函数名.括号.括号中的参数和冒号:,然后,在缩进块中编写函数体,函数的返回值用return语句返回. 函数的优点之一是,可以将代码块与主程 ...

  5. 【VMware vSphere】VMware vSphere简单了解

    *什么是VMware vSphere           说VMware vSphere可能刚开始都是一脸懵逼,但是看到VMware时,首先想到的是不是VMware Workstation?也就是虚拟 ...

  6. python目标定位(借鉴csdn上大神)

    写博客是为了记录下来,毕竟好多东西记不住,看过就忘了,收藏又太多,还不如搬运到自己博客下面,随时可翻~~~ 近期再学目标识别与定位,看着原理都很简单,但是真自己做,又觉得困难重重. csdn上一个大神 ...

  7. Go语言中的struct tag

    有时在Go的结构体定义时会看到这样的形式: type User struct { UserId int `json:"user_id" bson:"b_user_id&q ...

  8. 负载均衡获得真实源IP的6种方法 【转】

    除了X-FORWARD-FOR,负载均衡中获得真实源IP的方法还有很多种, 本文抛砖引玉,主要介绍获得真实源IP的多种方法,而不是具体配置, 负载均衡获得真实IP的方法有很多种,将形成专题文章, 本文 ...

  9. Vue.js 子组件的异步加载及其生命周期控制

    前端开发社区的繁荣,造就了很多优秀的基于 MVVM 设计模式的框架,而组件化开发思想也越来越深入人心.这其中不得不提到 Vue.js 这个专注于 VM 层的框架. 本文主要对 Vue.js 组件化开发 ...

  10. 常用的16个c/c++面试题

    1. C中static有什么作用 (1)隐藏. 当我们同时编译多个文件时,所有未加static前缀的全局变量和函数都具有全局可见性,故使用static在不同的文件中定义同名函数和同名变量,而不必担心命 ...