Android自定义TabLayout效果
发布时间 - 2026-01-10 23:25:38 点击率:次周末就要到了,今天项目中遇到这样一个Tab,选中tab的背景是个圆角矩形,方向指向器没有了,这样普通的TabLayout不能满足我的要求,可能会想到动态的去设置选中Tab的背景不就可以了,但是那样的话太生硬了,没有动画效果,其实想想也还比较简单,今天就简单的说一说这个YzzTab。效果如下图:
这里是四个Tab,一版只显示3个,这里假设有num个Tab,当滑动到第3个时,这里就需要考虑如何让TabLayout和指示器一起移动呢?
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
if (positionOffset>1){
return;
}
int leftCop = (int) (positionOffset*(getMeasuredWidth()/mMaxLineNum)+position*getMeasuredWidth()/mMaxLineNum);
if (leftCop!=leftForTabLayout){
//这里要做判断是否滑动,当选择的位置大于TabLayout中显示的最大数-1时,会向左右滑动,指示器也会
//跟这滑动,相对静止,否则指示器滑动,Tab布局不移动
if (position>=mMaxLineNum-1) {
scrollContent += leftCop - leftForTabLayout;
scrollTo(scrollContent, 0);
//这里要重新layout
update();
}
leftForTabLayout = leftCop;
invalidate();
}
}
首先,在ViewPage的监听中,positionOffset有时候可能大于1,这点需要注意的,当两次left的坐标相等 时,我们就不进行绘制了,接下来就是
如何确定left的值了,对于这点我也想了很久,最后终于得出结论:
int leftCop = (int) (positionOffset*(getMeasuredWidth()/mMaxLineNum)+position*getMeasuredWidth()/mMaxLineNum);
因为当positionOffset的值在向右滑动80%左右的时候getCurrentItem()的值会发生变化,这点可以试验一下,所以getCurrentItem()方法不能用了,只能用参数position.Layout滑动的实际代码注释很详细了,我就不再阐述了,可以试试。在布局滑动了以后,必须要layout,不然View的属性不会变,点击没法应,但是也可以不更新,动态的告诉用户点击的真是Tab,这样也可以。
private void update() {
for (int i = 0; i <mChildCount ; i++) {
View v = getChildAt(i);
v.setLeft(v.getLeft()+scrollContent);
}
//必须调用,不然不会重新layout
requestLayout();
}
接下来就是绘制了,ViewGroup是默认不调用onDraw(Canvas canvas)方法的,原因很简单,ViewGroup是个容器,主要作用是起承载作用,绘画就交给子View了,但是还是有办法让其调用该方法的,如下:
setWillNotDraw(false);
这就告诉该容器,需要绘制;
接下来就是绘制指向器和选中背景了,一个圆角矩形和一条线,比较简单,我就不再详细说明了。
@Override
protected void onDraw(Canvas canvas) {
//left = getMeasuredWidth() / mChildCount * mSelectPosition;
super.onDraw(canvas);
mPaint.setColor(Color.GREEN);
int top = getMeasuredHeight() / 4;
int right = leftForTabLayout + getMeasuredWidth() / mMaxLineNum;
int bottom = getMeasuredHeight() - getMeasuredHeight() / 4;
RectF rectF = new RectF(leftForTabLayout, top, right, bottom);
mPaint.setAntiAlias(true);
mPaint.setStyle(Paint.Style.FILL);
canvas.drawRoundRect(rectF, 10, 10, mPaint);
mPaint.setColor(Color.RED);
mPaint.setStrokeWidth(5);
canvas.drawLine(leftForTabLayout,getMeasuredHeight()-5,right,getMeasuredHeight()-5,mPaint);
}
接下来介绍建与ViewPager建立关联的方法
/**
* 于ViewPager建立联系,这里必须先要给ViewPager设置Adapter
*
* @param viewPager
*/
public void setUpWithViewPager(ViewPager viewPager) {
mViewPager = viewPager;
mChildCount = viewPager.getAdapter().getCount();
mSelectPosition = viewPager.getCurrentItem();
viewPager.setOnPageChangeListener(this);
}
初始化的方法
/**
* 为Tab添加View
*/
private void init() {
setWillNotDraw(false);
mPaint = new Paint();
for (int i = 0; i < mChildCount; i++) {
final TextView tv = new TextView(getContext());
int w = getMeasuredWidth()/mMaxLineNum;
LinearLayout.LayoutParams lp = new LayoutParams(w, ViewGroup.LayoutParams.MATCH_PARENT);
tv.setText("tab" + i);
tv.setGravity(Gravity.CENTER);
tv.setLayoutParams(lp);
final int finalI = i;
tv.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (monTabSelecterListener != null){
monTabSelecterListener.selecter(finalI,tv);
}
}
});
addView(tv);
}
}
这里只是很简单的加了几个TextView进去,也可以弄个方法,通过用户动态添加自己想要的View,都可以实现的。至于监听的话就很简单了.上面已经写到了。
YzzTab的代码
package a6he.android.yzz.com.mytablayout;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.RectF;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.LinearLayout;
import android.widget.Switch;
import android.widget.TextView;
/**
* Created by yzz on 2017/2/24 0024.
* <p/>
* 实现背景随着ViewPager的滑动跟着移动
*/
public class YzzTab extends LinearLayout implements ViewPager.OnPageChangeListener {
private ViewPager mViewPager;
private Paint mPaint;
//tab的数量
private int mChildCount;
//tab选中的位置
private int mSelectPosition;
//绘制指向器的左顶点
private int leftForTabLayout = 0;
private int leftForInvidator = 0;
private int mMaxLineNum = 3;
private int scrollContent = 0;
private onTabSelecterListener monTabSelecterListener;
public YzzTab(Context context) {
super(context);
}
public YzzTab(Context context, AttributeSet attrs) {
super(context, attrs);
}
public YzzTab(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
protected void onFinishInflate() {
super.onFinishInflate();
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
init();
}
/**
* 于ViewPager建立联系,这里必须先要给ViewPager设置Adapter
*
* @param viewPager
*/
public void setUpWithViewPager(ViewPager viewPager) {
mViewPager = viewPager;
mChildCount = viewPager.getAdapter().getCount();
mSelectPosition = viewPager.getCurrentItem();
viewPager.setOnPageChangeListener(this);
}
/**
* 为Tab添加View
*/
private void init() {
setWillNotDraw(false);
mPaint = new Paint();
for (int i = 0; i < mChildCount; i++) {
final TextView tv = new TextView(getContext());
int w = getMeasuredWidth()/mMaxLineNum;
LinearLayout.LayoutParams lp = new LayoutParams(w, ViewGroup.LayoutParams.MATCH_PARENT);
tv.setText("tab" + i);
tv.setGravity(Gravity.CENTER);
tv.setLayoutParams(lp);
final int finalI = i;
tv.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (monTabSelecterListener != null){
monTabSelecterListener.selecter(finalI,tv);
}
}
});
addView(tv);
}
}
@Override
protected void onDraw(Canvas canvas) {
//left = getMeasuredWidth() / mChildCount * mSelectPosition;
super.onDraw(canvas);
mPaint.setColor(Color.GREEN);
int top = getMeasuredHeight() / 4;
int right = leftForTabLayout + getMeasuredWidth() / mMaxLineNum;
int bottom = getMeasuredHeight() - getMeasuredHeight() / 4;
RectF rectF = new RectF(leftForTabLayout, top, right, bottom);
mPaint.setAntiAlias(true);
mPaint.setStyle(Paint.Style.FILL);
canvas.drawRoundRect(rectF, 10, 10, mPaint);
mPaint.setColor(Color.RED);
mPaint.setStrokeWidth(5);
canvas.drawLine(leftForTabLayout,getMeasuredHeight()-5,right,getMeasuredHeight()-5,mPaint);
}
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
if (positionOffset>1){
return;
}
int leftCop = (int) (positionOffset*(getMeasuredWidth()/mMaxLineNum)+position*getMeasuredWidth()/mMaxLineNum);
if (leftCop!=leftForTabLayout){
//这里要做判断是否滑动,当选择的位置大于TabLayout中显示的最大数-1时,会向左右滑动,指示器也会
//跟这滑动,相对静止,否则指示器滑动,Tab布局不移动
if (position>=mMaxLineNum-1) {
scrollContent += leftCop - leftForTabLayout;
scrollTo(scrollContent, 0);
//这里要重新layout
update();
}
leftForTabLayout = leftCop;
invalidate();
}
}
private void update() {
for (int i = 0; i <mChildCount ; i++) {
View v = getChildAt(i);
v.setLeft(v.getLeft()+scrollContent);
}
requestLayout();
}
@Override
public void onPageSelected(int position) {
}
@Override
public void onPageScrollStateChanged(int state) {
switch (state){
}
}
public void setmMaxLineNum(int mMaxLineNum) {
this.mMaxLineNum = mMaxLineNum;
}
public void setonTabSelecterListener(onTabSelecterListener monTabSelecterListener) {
this.monTabSelecterListener = monTabSelecterListener;
}
interface onTabSelecterListener{
void selecter(int position,View view);
}
}
好啦,就介绍这么多,还有待完善,继续封装,完成更强大的功能。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# TabLayout
# Android TabLayout(选项卡布局)简单用法实例分析
# Android TabLayout设置指示器宽度的方法
# android TabLayout使用方法详解
# Android中修改TabLayout底部导航条Indicator长短的方法
# Android 自定义View结合自定义TabLayout实现顶部标签滑动效果
# Android中TabLayout结合ViewPager实现页面切换效果
# android TabLayout的指示器宽度问题
# Android原生TabLayout使用的超全解析(看这篇就够了)
# 很简单
# 我就
# 是个
# 也会
# 要做
# 要给
# 必须先
# 会向
# 判断是否
# 圆角
# 我也
# 几个
# 是有
# 的说
# 就不
# 这么多
# 两次
# 很久
# 这就
# 用了
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel广播系统如何实现实时通信_Laravel Reverb与WebSockets实战教程
简历没回改:利用AI润色让你的文字更专业
Laravel怎么返回JSON格式数据_Laravel API资源Response响应格式化【技巧】
Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤
在centOS 7安装mysql 5.7的详细教程
东莞市网站制作公司有哪些,东莞找工作用什么网站好?
如何正确选择百度移动适配建站域名?
JavaScript如何实现路由_前端路由原理是什么
标题:Vue + Vuex + JWT 身份认证的正确实践与常见误区解析
进行网站优化必须要坚持的四大原则
Laravel如何与Inertia.js和Vue/React构建现代单页应用
郑州企业网站制作公司,郑州招聘网站有哪些?
Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】
PHP正则匹配日期和时间(时间戳转换)的实例代码
香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧
Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面
宙斯浏览器怎么屏蔽图片浏览 节省手机流量使用设置方法
Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)
大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?
Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例
作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】
Laravel如何使用withoutEvents方法临时禁用模型事件
广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?
Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧
如何快速生成凡客建站的专业级图册?
Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程
Laravel怎么使用Intervention Image库处理图片上传和缩放
详解CentOS6.5 安装 MySQL5.1.71的方法
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
如何为不同团队 ID 动态生成多个“认领值班”按钮
网站制作软件有哪些,制图软件有哪些?
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
大型企业网站制作流程,做网站需要注册公司吗?
桂林网站制作公司有哪些,桂林马拉松怎么报名?
网站页面设计需要考虑到这些问题
Laravel如何实现用户角色和权限系统_Laravel角色权限管理机制
Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】
Laravel如何配置Horizon来管理队列?(安装和使用)
如何利用DOS批处理实现定时关机操作详解
Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布
ai格式如何转html_将AI设计稿转换为HTML页面流程【页面】
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
Laravel观察者模式如何使用_Laravel Model Observer配置
phpredis提高消息队列的实时性方法(推荐)
JavaScript中的标签模板是什么_它如何扩展字符串功能
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
历史网站制作软件,华为如何找回被删除的网站?
网站建设整体流程解析,建站其实很容易!
Laravel如何使用Telescope进行调试?(安装和使用教程)
香港服务器如何优化才能显著提升网站加载速度?

