读人就是读自己。 — 《等一个人读书》
写在前面
最近项目又改了UI,真是一件开心的事情(微笑脸),效果图见图一,右上角有一个水平的白色线条,还有一个灰色的背景线条,就是这个东西,它是一个指示器。起初在没看到代码之前,我以为添加应用的界面类似ScrollView这种东西做的,如图二的淘宝首页轮播图下面部分,指示器和内容联动,手指滑动内容,指示器就会实时随之变化,具体效果详见淘宝首页。但是这里面有一个问题,我们项目这个界面用ViewPager写的,所以解决方案是根据页数去更新指示器位置。
具体实现
上面讲明了需求,现在就让我们用代码实现该指示器,创建TrackView继承自View。
public class TrackView extends View {
// 背景色
private int mBackColor;
// 前景色
private int mForeColor;
// 背景宽度
private int mBackWidth;
// 前景宽度
private int mForeWidth;
// 高度
private int mHeight;
// 前景色距View开始距离
private float mForeDistance;
// 画笔
private Paint mPaint;
public TrackView(Context context) {
this(context, null);
}
public TrackView(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
public TrackView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(attrs);
}
private void init(AttributeSet attrs) {
if (null != attrs) {
// 获取自定义属性,获取不到则使用默认值
TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.TrackView);
mBackColor = typedArray.getColor(R.styleable.TrackView_back_color, Color.GRAY);
mForeColor = typedArray.getColor(R.styleable.TrackView_fore_color, Color.WHITE);
mBackWidth = typedArray.getDimensionPixelOffset(R.styleable.TrackView_back_width, 100);
mForeWidth = typedArray.getDimensionPixelOffset(R.styleable.TrackView_fore_width, 50);
mHeight = typedArray.getDimensionPixelOffset(R.styleable.TrackView_height, 10);
// 一定要回收
typedArray.recycle();
} else {
mBackColor = Color.GRAY;
mForeColor = Color.WHITE;
mBackWidth = 100;
mForeWidth = 50;
mHeight = 10;
}
// 创建画笔,设置抗锯齿
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
// 设置画笔开始和结束为圆角
mPaint.setStrokeCap(Paint.Cap.ROUND);
// 设置画笔宽度
mPaint.setStrokeWidth(mHeight);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
drawBackground(canvas);
drawForeground(canvas);
}
/**
* 绘制背景线条,固定的那条线
* 需要考虑内边距
* @param canvas
*/
private void drawBackground(Canvas canvas) {
mPaint.setColor(mBackColor);
canvas.drawLine(mHeight + getPaddingLeft(),
mHeight / 2 + getPaddingTop(),
mHeight + getPaddingLeft() + mBackWidth,
mHeight / 2 + getPaddingTop(),
mPaint);
}
/**
* 绘制前景线条,会动的那条线,根据mForeDistance改变位置
* 需要考虑内边距
* @param canvas
*/
private void drawForeground(Canvas canvas) {
mPaint.setColor(mForeColor);
canvas.drawLine(mHeight + getPaddingLeft() + mForeDistance,
mHeight / 2 + getPaddingTop(),
mHeight + getPaddingLeft() + mForeDistance + mForeWidth,
mHeight / 2 + getPaddingTop(),
mPaint);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// 重新计算宽高
int width = getSize(mHeight * 2 + mBackWidth + getPaddingLeft() + getPaddingRight(), widthMeasureSpec);
int height = getSize(mHeight + getPaddingTop() + getPaddingBottom(), heightMeasureSpec);
setMeasuredDimension(width, height);
}
private int getSize(int size, int measureSpec) {
int result = size;
int mode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
switch (mode) {
// 如果测量模式为未知或wrap_content,则返回默认值。
case MeasureSpec.UNSPECIFIED:
case MeasureSpec.AT_MOST:
result = size;
break;
// 如果测量模式为具体数值或match_parent,则返回具体数值。
case MeasureSpec.EXACTLY:
result = specSize;
break;
default:
break;
}
return result;
}
/**
* 根据页数更新指示器位置
* @param position 当前页数
* @param position 总页数
*/
public void updateByPage(int position, int count) {
float offset = mBackWidth - mForeWidth;
mForeDistance = offset / (count - 1) * position;
postInvalidate();
}
}
下面是自定义属性,在/src/main/res/values/attrs.xml中。
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="TrackView">
<attr name="back_color" format="color"/>
<attr name="fore_color" format="color"/>
<attr name="back_width" format="dimension"/>
<attr name="fore_width" format="dimension"/>
<attr name="height" format="dimension"/>
</declare-styleable>
</resources>
如何使用
下面通过一个Demo演示如何使用该自定义View。
1.创建布局
使用ConstraintLayout包裹ViewPager和TrackView,指定TrackView的自定义属性。
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v4.view.ViewPager
android:id="@+id/view_pager"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<com.chad.learning.track.view.TrackView
android:id="@+id/view_track"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
app:back_color="@android:color/darker_gray"
app:back_width="100dp"
app:fore_color="@android:color/background_dark"
app:fore_width="50dp"
app:height="10dp"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent" />
</android.support.constraint.ConstraintLayout>
2.创建适配器
ViewPager需要适配器才能加载内容,所以这里创建一个适配器,每一页的内容都是一个TextView。
public class ViewPagerAdapter extends PagerAdapter {
private Context mContext;
private List<String> mData;
public ViewPagerAdapter(Context context, List<String> data) {
mContext = context;
mData = data;
}
@Override
public int getCount() {
return mData == null ? 0 : mData.size();
}
@Override
public boolean isViewFromObject(@NonNull View view, @NonNull Object object) {
return view == object;
}
@NonNull
@Override
public Object instantiateItem(@NonNull ViewGroup container, int position) {
ViewGroup.LayoutParams layoutParams = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
TextView textView = new TextView(mContext);
textView.setLayoutParams(layoutParams);
textView.setTextColor(Color.BLACK);
textView.setTextSize(50);
textView.setText(mData.get(position));
textView.setGravity(Gravity.CENTER);
container.addView(textView);
return textView;
}
@Override
public void destroyItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
container.removeView((View) object);
}
}
3.创建Activity
新建Activity,重写onCreate函数,调用setContentView指定布局,初始化View并调用ViewPager的addOnPageChangeListener设置页数改变监听器。
public class TrackActivity extends AppCompatActivity implements ViewPager.OnPageChangeListener {
private ViewPager mViewPager;
private TrackView mTrackView;
private ViewPagerAdapter mViewPagerAdapter;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_track);
initView();
}
private void initView() {
mViewPager = findViewById(R.id.view_pager);
mTrackView = findViewById(R.id.view_track);
List<String> data = new ArrayList<>();
for (int i = 0; i < 5; i ++) {
data.add(String.format("当前页数:%s", i + 1));
}
mViewPagerAdapter = new ViewPagerAdapter(this, data);
mViewPager.setAdapter(mViewPagerAdapter);
mViewPager.addOnPageChangeListener(this);
}
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override
public void onPageSelected(int position) {
// 该函数为页数改变回调,通过该回调更新指示器
mTrackView.updateByPage(position, mViewPagerAdapter.getCount());
}
@Override
public void onPageScrollStateChanged(int state) {
}
}
运行效果如下:
最后
如果这个功能让我做,强烈要求使用类似ScrollView那样的效果实现,这样指示器相当于ScrollBar,类似淘宝那样的效果,用户体验很好,这篇文章就不演示这种实现方式了,实现起来也不是很难,留给有兴趣的同学们搞。