Android自定义View实现分段选择按钮的实现代码
作者:danledian 发布时间:2022-09-06 07:46:21
首先演示下效果,分段选择按钮,支持点击和滑动切换。
视图绘制过程中,要执行onMeasure
、onLayout
、onDraw
等方法,这也是自定义控件最常用到的几个方法。onMeasure
:测量视图的大小,可以根据MeasureSpec的Mode确定父视图和子视图的大小。onLayout
:确定视图的位置onDraw
:绘制视图
这里就不做过多的介绍,主要介绍本控件涉及的到的部分。
1.1 获取item大小、起始位置
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
if(isItemZero() || getMeasuredWidth() == 0)
return;
mHeight = getMeasuredHeight();
int width = getMeasuredWidth();
mItemWidth = (width - 2 * itemHorizontalMargin)/getCount();
mStart = itemHorizontalMargin + mItemWidth * selectedItem;
mEnd = width - itemHorizontalMargin - mItemWidth;
}
1.2 绘制
绘制背景,所有的Item,以及选中项
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if(isItemZero())
return;
drawBackgroundRect(canvas);
drawUnselectedItemsText(canvas);
drawSelectedItem(canvas);
drawSelectedItemsText(canvas);
}
* 绘制背景区域
背景区域就是个带圆角的长方形
/**
* 画背景区域
* @param canvas
*/
private void drawBackgroundRect(Canvas canvas) {
float r = cornersMode == Round?cornersRadius: mHeight >> 1;
mPaint.setXfermode(null);
mPaint.setColor(backgroundColor);
mRectF.set(0, 0, getWidth(), getHeight());
canvas.drawRoundRect(mRectF, r, r, mPaint);
}
* 绘制所有未选中Item的文字
轮流绘制所有Item的文字
/**
* 画所有未选中Item的文字
* @param canvas
*/
private void drawUnselectedItemsText(Canvas canvas) {
mTextPaint.setColor(textColor);
mTextPaint.setXfermode(null);
for (int i = 0; i< getCount(); i++){
int start = itemHorizontalMargin + i * mItemWidth;
float x = start + (mItemWidth >> 1) - mTextPaint.measureText(getName(i))/2;
float y = (getHeight() >> 1) - (mTextPaint.ascent() + mTextPaint.descent())/2;
canvas.drawText(getName(i), x, y, mTextPaint);
}
}
* 绘制选中项
/**
* 画选中项
* @param canvas
*/
private void drawSelectedItem(Canvas canvas) {
float r = cornersMode == Round?cornersRadius: (mHeight >> 1) - itemVerticalMargin;
mPaint.setColor(selectedItemBackgroundColor);
mRectF.set(mStart, itemVerticalMargin, mStart + mItemWidth, getHeight() - itemVerticalMargin);
canvas.drawRoundRect(mRectF, r, r, mPaint);
}
* 绘制选中Item的文字
当选中项移动时,刚移动到下一个Item时,颜色应该是选中的颜色。这里在原来文字之上再画选中Item的文字颜色,就有了被选中的效果。
/**
* 画选中Item的文字
* @param canvas
*/
private void drawSelectedItemsText(Canvas canvas) {
canvas.saveLayer(mStart, 0, mStart + mItemWidth, getHeight(), null, Canvas.ALL_SAVE_FLAG);
mTextPaint.setColor(selectedItemTextColor);
mTextPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_OUT));
int begin = mStart/mItemWidth;
int end = (begin + 2) < getCount()?begin+2:getCount();
for (int i = begin; i< end; i++){
int start = itemHorizontalMargin + i * mItemWidth;
float x = start + (mItemWidth >> 1) - mTextPaint.measureText(getName(i))/2;
float y = (getHeight() >> 1) - (mTextPaint.ascent() + mTextPaint.descent())/2;
canvas.drawText(getName(i), x, y, mTextPaint);
}
canvas.restore();
}
1.3 添加手势事件
手势分为三种,ACTION_DOWN、ACTION_MOVE、ACTION_UP,对应动作就是按下,滑动,按起。
当按下时确定按下位置,是在当前Item,则不做处理,当按下位置为其它Item位置,就滑动到其它Item位置。
当手势滑动时,计算相对滑动值,通过改变mStart
,改变选中项的位置。
当手势按起时,根据按下位置、速度和方向,判断是否可用移动到下一个Item。
@Override
public boolean onTouchEvent(MotionEvent event) {
if(!isEnabled() || !isInTouchMode() || getCount() == 0)
return false;
if (mVelocityTracker == null) {
mVelocityTracker = VelocityTracker.obtain();
}
mVelocityTracker.addMovement(event);
int action = event.getActionMasked();
if(action == MotionEvent.ACTION_DOWN){
x = event.getX();
onClickDownPosition = -1;
final float y = event.getY();
if(isItemInside(x, y)){
return scrollSelectEnabled;
}else if(isItemOutside(x, y)){
if(!mScroller.isFinished()){
mScroller.abortAnimation();
}
onClickDownPosition = (int) ((x - itemHorizontalMargin)/ mItemWidth);
startScroll(positionStart(x));
return true;
}
return false;
}else if(action == MotionEvent.ACTION_MOVE){
if(!mScroller.isFinished() || !scrollSelectEnabled){
return true;
}
float dx = event.getX() - x;
if(Math.abs(dx) > MIN_MOVE_X){
mStart = (int) (mStart + dx);
mStart = Math.min(Math.max(mStart, itemHorizontalMargin), mEnd);
postInvalidate();
x = event.getX();
}
return true;
}else if(action == MotionEvent.ACTION_UP){
int newSelectedItem;
float offset = (mStart - itemHorizontalMargin)%mItemWidth;
float itemStartPosition = (mStart - itemHorizontalMargin) * 1.0f/ mItemWidth;
if(!mScroller.isFinished() && onClickDownPosition != -1){
newSelectedItem = onClickDownPosition;
}else{
if(offset == 0f){
newSelectedItem = (int)itemStartPosition;
}else {
VelocityTracker velocityTracker = mVelocityTracker;
velocityTracker.computeCurrentVelocity(VELOCITY_UNITS, mMaximumFlingVelocity);
int initialVelocity = (int) velocityTracker.getXVelocity();
float itemRate = offset/mItemWidth;
if (isXVelocityCanMoveNextItem(initialVelocity, itemRate)){
newSelectedItem = initialVelocity > 0?(int)itemStartPosition+1:(int)itemStartPosition;
}else {
newSelectedItem = Math.round(itemStartPosition);
}
newSelectedItem = Math.max(Math.min(newSelectedItem, getCount() - 1), 0);
startScroll(getXByPosition(newSelectedItem));
}
}
onStateChange(newSelectedItem);
mVelocityTracker = null;
onClickDownPosition = -1;
return true;
}
return super.onTouchEvent(event);
}
1.4 保存状态
当手机屏幕方向转换或者内存不足等情况下, 视图会重新加载,这样就会导致状态丢失。使用onSaveInstanceState
和onRestoreInstanceState
方法保存并恢复状态。
@Override
public Parcelable onSaveInstanceState() {
Parcelable parcelable = super.onSaveInstanceState();
SelectedItemState pullToLoadState = new SelectedItemState(parcelable);
pullToLoadState.setSelectedItem(selectedItem);
return pullToLoadState;
}
@Override
public void onRestoreInstanceState(Parcelable state) {
if(!(state instanceof SelectedItemState))
return;
SelectedItemState pullToLoadState = ((SelectedItemState)state);
super.onRestoreInstanceState(pullToLoadState.getSuperState());
selectedItem = pullToLoadState.getSelectedItem();
invalidate();
}
想要学习的同学,建议还是直接看项目源码。项目源码地址:https://github.com/danledian/SegmentedControl
来源:https://blog.csdn.net/songnigo6/article/details/111934357


猜你喜欢
- 实例如下:/// <summary> /// 上传ftp服务 /// </summary>
- 在学习java中的collection时注意到,collection层次的根接口Collection实现了Iterable<T>
- Java ThreadPoolExecutor的参数深入理解一、使用Executors创建线程池 &nb
- 前言最学习动态规划思想的路上,遇见了‘分割回文串问题',如临大敌啊,题目听起来蛮简单,思考起来却也没那么容易,比解决问题更头疼的是如
- JAVA 中Spring的@Async用法总结引言: 在Java应用中,绝大多数情况下都是通过同步的方式来实现交互处理的;但是在处理与第三方
- 前言当初年少懵懂,那年夏天填志愿选专业,父母听其他长辈说选择计算机专业好。从那以后,我的身上就有了计院深深的烙印。从寝室到机房,从机房到图书
- 在前面的博客中,https://www.jb51.net/article/134866.htm 我们使用了spring boot的异步操作,
- 我计划在后续的一段时间内,写一系列关于java 9的文章,虽然java 9 不像Java 8或者Java 11那样的核心java版本,但是还
- 一、Media FrameWork背景Media Framework (媒体函数库):此函数库让Android 可以播放与录制许多常见的音频
- synchronized是Java里的一个关键字,起到的一个效果是“监视器锁”~~,它的功能就是保证操作的原子性,同时禁止指令重排序和保证内
- 对智能手机有所了解的朋友都知道其中一个应用广泛的手机操作系统Android 开源手机操作系统。那么在这一系统中想要实现通话的监听功能的话,我
- 把C#编译成DLL或者Axtive控件,再由C调用!比如使用C++调用C#的DLL。SwfDotNet是.net下输出flash的类库。Sw
- Synchronized关键字Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码
- 函数四个方面:函数的定义函数的特点函数的应用函数的重载一、函数的定义及特点1) 什么是函数?函数就是定义在类中的具有特定功能的一段独立小程序
- 这个例子用于演示在Spring Boot应用中如何验证Web 应用的输入,我们将会建立一个简单的Spring MVC应用,来读取用户输入并使
- Console.WriteLine("This is a Client, host name is {0}", Dns.
- 本文实例讲述了Java编程使用卡片布局管理器。分享给大家供大家参考,具体如下:运行效果:完整示例代码:package com.han;imp
- 介绍微服务横行的互联网世界, 跨服务调用显得很平凡, 我们除了采用传统的http方式接口调用, 有没有更为优雅方便的方法呢?答案是肯定的,f
- 1. 源码阅读环境搭建ide:IntelliJ IDEA 2020.1包管理:gradleeureka版本:1.10.11Spring Cl
- 场景既然要搞懂Redis分布式锁,那肯定要有一个需要它的场景。高并发售票问题就是一个经典案例。搭建环境准备redis服务,设置redis的键