- getIntrinsicWidth(),getIntrinsicHeight()方法,然后设为背景
*/
@Override
protected void dispatchDraw(Canvas canvas) {
super.dispatchDraw(canvas);
/*
-
save:用来保存Canvas的状态。save之后,可以调用Canvas的平移、放缩、旋转、错切、裁剪等操作。
-
restore:用来恢复Canvas之前保存的状态。防止save后对Canvas执行的操作对后续的绘制有影响。
-
save和restore要配对使用(restore可以比save少,但不能多),如果restore调用次数比save多,会引发Error。
*/
canvas.save();
canvas.translate(mTranslationX, RECT_HEIGHT);
//绘制一个封闭的区域
canvas.drawPath(mPath, mPaint);
canvas.restore();
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
//矩形的宽 就是屏幕的宽度/可见个数
mRectWidth = w / mTabVisibleCount;
initRect();
}
/**
- 初始化矩形
*/
private void initRect() {
//矩形的位置在 子控件的底部
mPath = new Path();
mPath.moveTo(0, getHeight()); //将下一个轮廓的开始设置为点(x,y)。
//从最后一点添加一行到指定点(x,y)。如果没有对此轮廓进行moveTo()调用,则第一个点为自动设置为(0,0)。
mPath.lineTo(mRectWidth, getHeight());
mPath.lineTo(mRectWidth, getHeight() - RECT_HEIGHT - RECT_HEIGHT);
mPath.lineTo(0, getHeight() - RECT_HEIGHT - RECT_HEIGHT);
// 关闭当前轮廓,完成闭合
mPath.close();
}
/**
-
移动指示器
-
@param position 需要移动到的那个位置
-
@param offset 移动偏移量百分比
*/
public void scroll(int position, float offset) {
//每个tab的宽度
int tabWidth = getWidth() / mTabVisibleCount;
//指示器需要偏移的距离
mTranslationX = (int) (tabWidth * offset + tabWidth * position);
//当移动到倒数第二个时,就需要移动ViewPagerIndicator 当前这个ViewGroup了
if (position >= (mTabVisibleCount - 2) && offset > 0 && getChildCount() >
mTabVisibleCount) {
//如果mTabVisibleCount == 1时,mTabVisibleCount - 2是负数
if (mTabVisibleCount != 1) {
//如果是移动到所有页签的倒数第二个则不能再移动了
if (position != getChildCount() - 2) {
Log.d(TAG, “scroll: position----”+position);
this.scrollTo((int) ((position - (mTabVisibleCount - 2)) * tabWidth + tabWidth *
offset), 0);
//当移动到界面可见数的最后一个时,直接移动到0 不然会出现第一个显示不全的bug
if (position == mTabVisibleCount-2) {
Log.d(TAG, “scroll: 0”);
this.scrollTo(0,0);
}
}
} else {
this.scrollTo((int) (position * tabWidth + tabWidth * offset), 0);
}
}
//重绘
invalidate();
}
/**
-
当xml加载完成之后就会调用此方法,必须调用super()方法
-
我们需要在这做的事情是 将每个子控件的weight改为0,然后将默认的可以显示的个数的子控件显示出来
-
通过布局添加tab的时候,调用到此方法
*/
@Override
protected void onFinishInflate() {
super.onFinishInflate();
//获取子控件个数
int childCount = getChildCount();
if (childCount < 0) {
return;
}
//设置每个页签的宽度
for (int i = 0; i < childCount; i++) {
View view = getChildAt(i);
LinearLayout.LayoutParams layoutParams = (LayoutParams) view.getLayoutParams();
layoutParams.weight = 0;
layoutParams.width = getScreenWidth() / mTabVisibleCount;
view.setLayoutParams(layoutParams);
}
//设置每个tab的点击事件
setTabItemClickEvent();
}
/**
-
获取屏幕宽度
-
@return
*/
private int getScreenWidth() {
//拿到WindowManager对象
WindowManager windowManager = (WindowManager) getContext().getSystemService(Context
.WINDOW_SERVICE);
Display defaultDisplay = windowManager.getDefaultDisplay();
DisplayMetrics metrics = new DisplayMetrics();
defaultDisplay.getMetrics(metrics);
return metrics.widthPixels;
}
/**
-
动态地设置可见tab的个数,必须大于0
-
必须在setTabItemTitles()之前调用此方法才能生效
-
@param mTabVisibleCount
*/
public void setmTabVisibleCount(int mTabVisibleCount) {
if (mTabVisibleCount > 0) {
this.mTabVisibleCount = mTabVisibleCount;
} else {
this.mTabVisibleCount = COUNT_DEFAULT_TAB; //默认值
}
}
/**
-
动态设置tab
-
@param titles 所有tab的标题
*/
public void setTabItemTitles(List titles) {
if (titles != null && titles.size() > 0) {
//用户是想设置Tab,则无视布局文件中写的TextView,全部移除
removeAllViews();
//生成标题
for (String title : titles) {
addView(generateTextView(title));
}
}
//设置每个tab的点击事件
setTabItemClickEvent();
}
/**
-
根据title动态地生成一个tab 其实就是一个TextView,用来显示tab的名称用的
-
@param title
-
@return
*/
private View generateTextView(String title) {
TextView textView = new TextView(getContext());
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout
.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT);
if (mTabVisibleCount == 0) {
params.width = getScreenWidth() / COUNT_DEFAULT_TAB;
} else {
params.width = getScreenWidth() / mTabVisibleCount;
}
textView.setGravity(Gravity.CENTER);
textView.setTextSize(TypedValue.COMPLEX_UNIT_SP, 18);
textView.setText(title);
textView.setTextColor(COLOR_TEXT_NORMAL);
textView.setLayoutParams(params);
return textView;
}
/**
-
设置关联的ViewPager
-
@param viewPager
-
@param position 设置当前位置
*/
public void setViewPager(ViewPager viewPager, int position) {
this.mViewPager = viewPager;
//页面改变事件 滑动
viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int position, float positionOffset, int
positionOffsetPixels) {
//移动指示器
scroll(position, positionOffset);
//监听器 全部需要回调此方法 参考源码
if (mOnPageChangeListeners != null) {
for (int i = 0, z = mOnPageChangeListeners.size(); i < z; i++) {
OnPageChangeListener listener = mOnPageChangeListeners.get(i);
if (listener != null) {
listener.onPageScrolled(position, positionOffset, positionOffsetPixels);
}
}
}
}
@Override
public void onPageSelected(int position) {
if (mOnPageChangeListeners != null) {
for (int i = 0, z = mOnPageChangeListeners.size(); i < z; i++) {
OnPageChangeListener listener = mOnPageChangeListeners.get(i);
if (listener != null) {
listener.onPageSelected(position);
}
}
}
//高亮当前选中的位置
highLightTextView(position);
}
@Override
public void onPageScrollStateChanged(int state) {
if (mOnPageChangeListeners != null) {
for (int i = 0, z = mOnPageChangeListeners.size(); i < z; i++) {
OnPageChangeListener listener = mOnPageChangeListeners.get(i);
if (listener != null) {
listener.onPageScrollStateChanged(state);
}
}
}
}
});
//设置当前的tab是position位置
viewPager.setCurrentItem(position);
//高亮当前选中的位置
highLightTextView(position);
}
/**
-
需要暴露给开发者的ViewPager的滑动事件 接口 和ViewPager的监听器定义地一模一样
-
可能开发者也需要用到ViewPager的监听事件
*/
public interface OnPageChangeListener {
void onPageScrolled(int position, float positionOffset, int positionOffsetPixels);
void onPageSelected(int position);
void onPageScrollStateChanged(int state);
}
/**
-
给ViewPager添加监听器
-
官方已经弃用setOnPageChangeListener()了,取而代之的是
-
Use {@link #addOnPageChangeListener(OnPageChangeListener)}
-
and {@link #removeOnPageChangeListener(OnPageChangeListener)} instead.
-
@param listener
*/
public void addOnPageChangeListener(OnPageChangeListener listener) {
if (mOnPageChangeListeners == null) {
mOnPageChangeListeners = new ArrayList<>();
}
mOnPageChangeListeners.add(listener);
}
/**
-
Remove a listener that was previously added via
-
{@link #addOnPageChangeListener(OnPageChangeListener)}.
-
@param listener listener to remove
*/
public void removeOnPageChangeListener(OnPageChangeListener listener) {
if (mOnPageChangeListeners != null) {
mOnPageChangeListeners.remove(listener);
}
}
/**
-
Remove all listeners that are notified of any changes in scroll state or position.
-
还是用英文吧,虽然知道意思,但是硬翻译过来总觉得不舒服
*/
public void clearOnPageChangeListeners() {
if (mOnPageChangeListeners != null) {
mOnPageChangeListeners.clear();
}
}
/**
-
高亮显示tab标题
-
@param position 那个需要高亮的位置
*/
总结
最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的Android开发中高级必知必会核心笔记,共计2968页PDF、58w字,囊括Android开发648个知识点,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。
虽然面试失败了,但我也不会放弃入职字节跳动的决心的!建议大家面试之前都要有充分的准备,顺顺利利的拿到自己心仪的offer。
) {
if (mOnPageChangeListeners != null) {
mOnPageChangeListeners.clear();
}
}
/**
-
高亮显示tab标题
-
@param position 那个需要高亮的位置
*/
总结
最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的Android开发中高级必知必会核心笔记,共计2968页PDF、58w字,囊括Android开发648个知识点,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。
[外链图片转存中…(img-fkkNJfvz-1726074899019)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。
虽然面试失败了,但我也不会放弃入职字节跳动的决心的!建议大家面试之前都要有充分的准备,顺顺利利的拿到自己心仪的offer。