TabLayout学习笔记(简介及进阶)(1)

<include

layout=“@layout/include_toolbar”/>

<android.support.design.widget.TabLayout

android:id=“@+id/tl_tab”

android:layout_width=“match_parent”

android:layout_height=“@dimen/dp_48”

android:background=“@color/blue”>

</android.support.design.widget.TabLayout>

<android.support.v4.view.ViewPager

android:id=“@+id/vp_content”

android:layout_width=“match_parent”

android:layout_height=“match_parent”>

</android.support.v4.view.ViewPager>

然后我们看一下 TabLayoutActivity 中的代码:

package com.yifeng.mdstudysamples;

import android.os.Bundle;

import android.support.annotation.Nullable;

import android.support.design.widget.TabLayout;

import android.support.v4.app.Fragment;

import android.support.v4.app.FragmentManager;

import android.support.v4.app.FragmentPagerAdapter;

import android.support.v4.content.ContextCompat;

import android.support.v4.view.ViewCompat;

import android.support.v4.view.ViewPager;

import android.view.Menu;

import android.view.MenuItem;

import java.util.ArrayList;

import java.util.List;

public class TabLayoutActivity extends BaseActivity {

private TabLayout mTabTl;

private ViewPager mContentVp;

private List tabIndicators;

private List tabFragments;

private ContentPagerAdapter contentAdapter;

@Override

protected void onCreate(@Nullable Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_tab_layout);

mTabTl = (TabLayout) findViewById(R.id.tl_tab);

mContentVp = (ViewPager) findViewById(R.id.vp_content);

initContent();

initTab();

}

private void initTab(){

mTabTl.setTabMode(TabLayout.MODE_SCROLLABLE);

mTabTl.setTabTextColors(ContextCompat.getColor(this, R.color.gray), ContextCompat.getColor(this, R.color.white));

mTabTl.setSelectedTabIndicatorColor(ContextCompat.getColor(this, R.color.white));

ViewCompat.setElevation(mTabTl, 10);

mTabTl.setupWithViewPager(mContentVp);

}

private void initContent(){

tabIndicators = new ArrayList<>();

for (int i = 0; i < 3; i++) {

tabIndicators.add("Tab " + i);

}

tabFragments = new ArrayList<>();

for (String s : tabIndicators) {

tabFragments.add(TabContentFragment.newInstance(s));

}

contentAdapter = new ContentPagerAdapter(getSupportFragmentManager());

mContentVp.setAdapter(contentAdapter);

}

@Override

public boolean onCreateOptionsMenu(Menu menu) {

getMenuInflater().inflate(R.menu.menu_tab_layout, menu);

return true;

}

@Override

public boolean onOptionsItemSelected(MenuItem item) {

switch (item.getItemId()) {

case R.id.tab_add:

tabIndicators.add("Tab " + tabIndicators.size());

tabFragments.add(TabContentFragment.newInstance(tabIndicators.get(tabIndicators.size()-1)));

contentAdapter.notifyDataSetChanged();

return true;

case R.id.tab_mode_fixed:

mTabTl.setTabMode(TabLayout.MODE_FIXED);

return true;

case R.id.tab_mode_scrollable:

mTabTl.setTabMode(TabLayout.MODE_SCROLLABLE);

break;

}

return super.onOptionsItemSelected(item);

}

class ContentPagerAdapter extends FragmentPagerAdapter{

public ContentPagerAdapter(FragmentManager fm) {

super(fm);

}

@Override

public Fragment getItem(int position) {

return tabFragments.get(position);

}

@Override

public int getCount() {

return tabIndicators.size();

}

@Override

public CharSequence getPageTitle(int position) {

return tabIndicators.get(position);

}

}

}

核心代码有两个地方,第一个是 setupWithViewPager 方法将TabLayout和ViewPager绑定在一起,使双方各自的改变都能直接影响另一方,解放了开发人员对双方变动事件的监听;第二个是在ViewPager的Adapter适配器中重写 getPageTitle 方法,在这个方法中设置标签指示器的标题。

底部菜单栏

上面我们使用了系统定义好的View做了一个纯文字加下划线组合的标签指示器。其实,我们也能自定义一个布局,然后赋值给TabLayout的Tab视图,比如做一个微信首页界面。

相比顶部标签指示器,底部菜单栏只是将TabLayout布局在了下面:

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

<include

layout=“@layout/include_toolbar”/>

<android.support.v4.view.ViewPager

android:id=“@+id/vp_content”

android:layout_width=“match_parent”

android:layout_height=“0dp”

android:layout_weight=“1”>

</android.support.v4.view.ViewPager>

<android.support.design.widget.TabLayout

android:id=“@+id/tl_tab”

android:layout_width=“match_parent”

android:layout_height=“@dimen/dp_56”

android:background=“@color/white”>

</android.support.design.widget.TabLayout>

在Activity代码中,设置TabLayout的指示器高度为0,即达到了隐藏Indicator的目的,然后通过getTabAt(position)的方法获取TabLayout的每一个Tab,并赋值为自定义布局视图,代码也很简单:

import android.os.Bundle;

import android.support.annotation.Nullable;

import android.support.design.widget.TabLayout;

import android.support.v4.app.Fragment;

import android.support.v4.app.FragmentManager;

import android.support.v4.app.FragmentPagerAdapter;

import android.support.v4.view.ViewCompat;

import android.support.v4.view.ViewPager;

import android.widget.TextView;

import java.util.ArrayList;

import java.util.List;

/**

  • Created by yifeng on 16/8/3.

*/

public class TabLayoutBottomActivity extends BaseActivity {

private TabLayout mTabTl;

private ViewPager mContentVp;

private List tabIndicators;

private List tabFragments;

private ContentPagerAdapter contentAdapter;

@Override

protected void onCreate(@Nullable Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_tab_layout_bottom);

mTabTl = (TabLayout) findViewById(R.id.tl_tab);

mContentVp = (ViewPager) findViewById(R.id.vp_content);

initContent();

initTab();

}

private void initTab(){

mTabTl.setTabMode(TabLayout.MODE_FIXED);

mTabTl.setSelectedTabIndicatorHeight(0);

ViewCompat.setElevation(mTabTl, 10);

mTabTl.setupWithViewPager(mContentVp);

for (int i = 0; i < tabIndicators.size(); i++) {

TabLayout.Tab itemTab = mTabTl.getTabAt(i);

if (itemTab!=null){

itemTab.setCustomView(R.layout.item_tab_layout_custom);

TextView itemTv = (TextView) itemTab.getCustomView().findViewById(R.id.tv_menu_item);

itemTv.setText(tabIndicators.get(i));

}

}

mTabTl.getTabAt(0).getCustomView().setSelected(true);

}

private void initContent(){

tabIndicators = new ArrayList<>();

for (int i = 0; i < 4; i++) {

tabIndicators.add("Tab " + i);

}

tabFragments = new ArrayList<>();

for (String s : tabIndicators) {

tabFragments.add(TabContentFragment.newInstance(s));

}

contentAdapter = new ContentPagerAdapter(getSupportFragmentManager());

mContentVp.setAdapter(contentAdapter);

}

class ContentPagerAdapter extends FragmentPagerAdapter{

public ContentPagerAdapter(FragmentManager fm) {

super(fm);

}

@Override

public Fragment getItem(int position) {

return tabFragments.get(position);

}

@Override

public int getCount() {

return tabIndicators.size();

}

@Override

public CharSequence getPageTitle(int position) {

return tabIndicators.get(position);

}

}

}

小结:

从这两种使用场景可以看出,利用TabLayout做一个滑动标签页或者底部菜单栏,实现起来非常方便,代码量也不多,极大地减少了我们的开发量。但是,TabLayout也不是万能的,如果想做出更多地特效还是需要我们自己去开发

TabLayout进阶:

与ActionBar类似的是TabLayout添加Tab也很方便,只需要调用其addTab方法即可:

public class TabLayoutActivity extends Activity {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.ac_tab_layout);

TabLayout tabLayout = (TabLayout) findViewById(R.id.ac_tab_layout);

TabLayout.Tab tab1 = tabLayout.newTab().setText(“Tab1”);

tabLayout.addTab(tab1);

TabLayout.Tab tab2 = tabLayout.newTab().setText(“Tab2”);

tabLayout.addTab(tab2);

TabLayout.Tab tab3 = tabLayout.newTab().setText(“Tab3”);

tabLayout.addTab(tab3);

}

}

运行效果如下:

非常简单,不过需要注意的是,这里的Tab类是TabLayout的一个静态内部类,并且其构造方法是包内可访问的,也就是说你无法在外部构造其实例,只能通过TabLayout的newTab方法:

public class TabLayout extends HorizontalScrollView {

// …省去很多代码…

public static final class Tab {

// …省去很多代码…

Tab(TabLayout parent) {

mPosition = -1;

mParent = parent;

}

}

public Tab newTab() {

return new Tab(this);

}

// …省去很多代码…

}

addTab方法有四个不同的实现,拿参数最多的addTab(Tab tab, int position, boolean setSelected)方法来说,第一个参数tab就不说了,第二个参数position表示当前Tab需要插入的位置,其值必须小于当前TabLayout所拥有的Tab总数减一,如果大于则会报出数组下标溢出异常,这点很好理解,最后的boolean参数setSelected与其名意义一样,表示当前添加的Tab是否为选中状态,拿上面代码中的第三个tab3来说,我们修改其代码如下:

TabLayout.Tab tab3 = tabLayout.newTab().setText(“Tab3”);

tabLayout.addTab(tab3, 2, true);

运行后tab3默认就会是被选中状态:

TabLayout中的公共方法不算多,大多都很好理解,不过有一个方法爱哥表示对其极度不爽,setScrollPosition从名字上来看它是用来设置滚动位置的,大家注意我们的TabLayout中被选中的Tab下方不是有根线么,我们可以通过这个方法来设置它的位置。首先我们在布局中添加一个按钮:

<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:layout_height=“match_parent”>

<android.support.design.widget.TabLayout

android:id=“@+id/ac_tab_layout”

android:layout_width=“match_parent”

android:layout_height=“wrap_content” />

<Button

android:id=“@+id/ac_tab_btn”

android:layout_width=“wrap_content”

android:layout_centerInParent=“true”

android:layout_height=“wrap_content”

android:text=“Move” />

然后呢为这个按钮绑定一个点击事件监听并在其中调用setScrollPosition方法:

Button btnAdd = (Button) findViewById(R.id.ac_tab_btn);

btnAdd.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

tabLayout.setScrollPosition(0, 0.5F, false);

}

});

第一个参数表示当前Tab的位置,第二个参数是偏移值,从文档中看到该值的取值范围是0到1的一个半开区间,最后一个参数很好理解表示是否置移动后位置所对应的Tab为选中状态,打个比方,如果我从0移动到1的位置,如果updateSelectedText为true,那么1这个位置上的文本就会是一个选中状态。上面的代码中我们只是简单地让下方的横条从0的Tab位置移动到一个0.5F的位置,效果如下:

调用了updateIndicatorPosition来做具体的位置更新操作:

private void updateIndicatorPosition() {

View selectedTitle = getChildAt(mSelectedPosition);

int left;

int right;

if (selectedTitle != null && selectedTitle.getWidth() > 0) {

left = selectedTitle.getLeft();

right = selectedTitle.getRight();

if (mSelectionOffset > 0.0F && mSelectedPosition < getChildCount() - 1) {
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后是今天给大家分享的一些独家干货:

【Android开发核心知识点笔记】

【Android思维脑图(技能树)】

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

【Android高级架构视频学习资源】

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

图片转存中…(img-LYYSqUd4-1712299588462)]

[外链图片转存中…(img-OfUbAlhi-1712299588462)]

[外链图片转存中…(img-jIZ6wPLK-1712299588463)]

[外链图片转存中…(img-9RbpzcAA-1712299588463)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后是今天给大家分享的一些独家干货:

【Android开发核心知识点笔记】

[外链图片转存中…(img-GlseXJvO-1712299588463)]

【Android思维脑图(技能树)】

[外链图片转存中…(img-x2qDcN1g-1712299588463)]

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

[外链图片转存中…(img-eoJ5LOs6-1712299588464)]

【Android高级架构视频学习资源】

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值