Android 兩個可拖動的SeekBar 兩點拖動
DoubleSeekBar.java類
package cn.deerlands.deerland.mvp.ui.wiget;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BlurMaskFilter;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.RectF;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import androidx.annotation.Nullable;
import androidx.core.content.ContextCompat;
import cn.deerlands.deerland.R;
import cn.deerlands.deerland.mvp.ui.util.DisplayUtil;
/**
* 雙滑塊的bar
*/
public class DoubleSeekBar extends View {
private static final String TAG = "DoubleSeekBar";
private static final int CLICK_TYPE_NULL = 0;
private static final int CLICK_TYPE_LEFT = 1;
private static final int CLICK_TYPE_RIGHT = 2;
private Bitmap leftIcon;
private Bitmap rightIcon;
private Bitmap clickIcon;
private int maxValue;//最大值
private int minValue;//最小值
private int leftValue, rightValue;
private int defaultWidth = 0;
private int defaultHeight = 0;
private float bgHeight;
private float paddingLeft, paddingRight;
private float indexLeftX, indexLeftY;//左按鈕的座標值
private float indexRightX, indexRightY;//右按鈕的座標值
private int viewWidth, viewHeight;
private float startLeftX, startRightX;//左右按鈕的開始x軸座標值
private int clickedType = CLICK_TYPE_NULL;//0 沒有點擊,1點擊中左按鈕,2點擊中右按鈕
private Paint paintBg = new Paint(Paint.ANTI_ALIAS_FLAG);
private Paint paintBg2 = new Paint(Paint.ANTI_ALIAS_FLAG);
private Paint iconPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
private OnChanged mOnChanged;
private Paint text_Paint = new Paint();//文字paint
private int displayHeight; //文字偏移高度
public DoubleSeekBar(Context context) {
this(context, null);
}
public DoubleSeekBar(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
public DoubleSeekBar(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
bgHeight = DisplayUtil.dip2px(context, 2);//設置背景線條的高度
leftIcon = getBitmap(R.drawable.filter_seek_thumb);
rightIcon = getBitmap(R.drawable.filter_seek_thumb);
clickIcon = getBitmap(R.drawable.filter_seek_thumb);
paddingLeft = DisplayUtil.dip2px(context, 16);
paddingRight = DisplayUtil.dip2px(context, 16);
paintBg.setColor(getResources().getColor(R.color.colorButtonRadio));
paintBg2.setColor(getResources().getColor(R.color.colorSeekProgress));
maxValue = 100;//default maxvalue
minValue = 0;//default minvalue
leftValue = minValue;
rightValue = maxValue;
defaultWidth = DisplayUtil.dip2px(context, 100);
defaultHeight = DisplayUtil.dip2px(context, 50);
displayHeight = DisplayUtil.dip2px(context, 10);
text_Paint.setTextAlign(Paint.Align.CENTER);
text_Paint.setColor(ContextCompat.getColor(context, R.color.colorButtonRadio));
text_Paint.setTextSize(DisplayUtil.dip2px(context, 10));
setLayerType(LAYER_TYPE_SOFTWARE, null);
}
private Bitmap getBitmap(int drawableRes) {
Drawable drawable = getResources().getDrawable(drawableRes);
Canvas canvas = new Canvas();
Bitmap bitmap = Bitmap.createBitmap(drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight(), Bitmap.Config.ARGB_8888);
canvas.setBitmap(bitmap);
drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
drawable.draw(canvas);
return bitmap;
}
public void setMaxValue(int maxValue) {
if (maxValue < 0) {
maxValue = 0;
}
this.maxValue = maxValue;
}
public void setMinValue(int minValue) {
if (minValue < 0) {
minValue = 0;
}
this.minValue = minValue;
}
public void setOnChanged(OnChanged mOnChanged) {
this.mOnChanged = mOnChanged;
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
if (isClickedIcon(event)) {
postInvalidate();
return true;
}
break;
case MotionEvent.ACTION_UP:
clickedType = CLICK_TYPE_NULL;
postInvalidate();
break;
case MotionEvent.ACTION_MOVE:
if (handleMoveEvent(event)) {
return true;
}
break;
default:
break;
}
return super.onTouchEvent(event);
}
/**
* 判斷是否點擊到按鈕了
*
* @param event
* @return
*/
private boolean isClickedIcon(MotionEvent event) {
float x = event.getX();
float y = event.getY();
if (x < indexLeftX + leftIcon.getWidth() && x > indexLeftX && y > indexLeftY && y < indexLeftY + leftIcon.getHeight()) {
clickedType = CLICK_TYPE_LEFT;
return true;
}
if (x < indexRightX + rightIcon.getWidth() && x > indexRightX && y > indexRightY && y < indexRightY + rightIcon.getHeight()) {
clickedType = CLICK_TYPE_RIGHT;
return true;
}
clickedType = CLICK_TYPE_NULL;
return false;
}
/**
* 滑動事件處理
*
* @param motionEvent
*/
private boolean handleMoveEvent(MotionEvent motionEvent) {
Log.d(TAG, "handleMoveEvent: start");
float x = motionEvent.getX();
if (clickedType == CLICK_TYPE_LEFT) {
if (x < indexRightX - leftIcon.getWidth() && x > startLeftX) {//左按鈕的範圍小於右按鈕的位置,大於初始值位置
indexLeftX = x;
}
Log.d(TAG, "handleMoveEvent: start indexLeftX = " + indexLeftX);
} else if (clickedType == CLICK_TYPE_RIGHT) {
if (x > indexLeftX + rightIcon.getWidth() && x < startRightX) {//右按鈕的範圍大於左按鈕的位置,小於初始值位置
indexRightX = x;
}
Log.d(TAG, "handleMoveEvent: start indexRightX = " + indexRightX);
}
leftValue = (int) ((maxValue - minValue) * (indexLeftX - startLeftX) / (startRightX - startLeftX));
rightValue = (int) ((maxValue - minValue) * (indexRightX - startLeftX) / (startRightX - startLeftX));
if (mOnChanged != null) {
mOnChanged.onChange(leftValue, rightValue);
}
Log.d(TAG, "handleMoveEvent: start leftValue = " + leftValue + " rightValue = " + rightValue);
postInvalidate();
return true;
}
@Override
protected void onFinishInflate() {
super.onFinishInflate();
}
/**
* 注意,在LinearLayout佈局中,設置weight = 1時,onlayout 會多次回調,導致indexL(/R)X值一直爲默認值,導致無法滑動
* 可以將佈局改成releativeLayout。
*
* @param changed
* @param left
* @param top
* @param right
* @param bottom
*/
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
Log.d(TAG, "onLayout: ");
viewWidth = getWidth();
viewHeight = getHeight();
//左右按鈕的初始值位置計算
startLeftX = 0 + paddingLeft - leftIcon.getWidth() / 2;
indexLeftX = startLeftX;
indexLeftY = viewHeight / 2 - bgHeight / 2 - leftIcon.getHeight() / 2;
startRightX = viewWidth - paddingRight - rightIcon.getWidth() / 2;
indexRightX = startRightX;
indexRightY = viewHeight / 2 - bgHeight / 2 - rightIcon.getHeight() / 2;
leftValue = (int) ((maxValue - minValue) * (indexLeftX - startLeftX) / (startRightX - startLeftX));
rightValue = (int) ((maxValue - minValue) * (indexRightX - startLeftX) / (startRightX - startLeftX));
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Log.d(TAG, "onDraw: getWidth = " + viewWidth);
drawBg(canvas);
drawLeftIcon(canvas);
drawRightIcon(canvas);
}
/**
* 繪製背景
*
* @param canvas
*/
private void drawBg(Canvas canvas) {
paintBg.setAntiAlias(true);
//畫兩端的半圓
float circleLeftCenterX = 0 + paddingLeft;
float circleRightCenterX = viewWidth - paddingRight;
canvas.drawCircle(circleLeftCenterX, viewHeight / 2, bgHeight / 2, paintBg);
canvas.drawCircle(circleRightCenterX, viewHeight / 2, bgHeight / 2, paintBg);
canvas.drawRect(
new RectF(circleLeftCenterX, viewHeight / 2 - bgHeight / 2, circleRightCenterX, viewHeight / 2 + bgHeight / 2)
, paintBg);
//左右按鈕滑動區域之外繪製其他顏色
int centerLeftIconX = (int) (indexLeftX + leftIcon.getWidth() / 2);//左按鈕圖標的中心點
int centerRightIconX = (int) (indexRightX + rightIcon.getWidth() / 2);//右按鈕圖標的中心點
if (centerLeftIconX > circleLeftCenterX) {
canvas.drawCircle(circleLeftCenterX, viewHeight / 2, bgHeight / 2, paintBg2);
canvas.drawRect(
new RectF(circleLeftCenterX, viewHeight / 2 - bgHeight / 2, centerLeftIconX, viewHeight / 2 + bgHeight / 2)
, paintBg2);
}
if (centerRightIconX < circleRightCenterX) {
canvas.drawCircle(circleRightCenterX, viewHeight / 2, bgHeight / 2, paintBg2);
canvas.drawRect(
new RectF(centerRightIconX, viewHeight / 2 - bgHeight / 2, circleRightCenterX, viewHeight / 2 + bgHeight / 2)
, paintBg2);
}
}
/**
* 繪製左按鈕
*
* @param canvas
*/
private void drawLeftIcon(Canvas canvas) {
if (clickedType == CLICK_TYPE_LEFT) {
iconPaint.setMaskFilter(new BlurMaskFilter(50, BlurMaskFilter.Blur.INNER));
canvas.drawBitmap(clickIcon, indexLeftX, indexLeftY, iconPaint);
} else {
iconPaint.setMaskFilter(null);
canvas.drawBitmap(leftIcon, indexLeftX, indexLeftY, iconPaint);
}
//畫文字
int num = leftValue + minValue;
canvas.drawText(num + "", (int) indexLeftX+15, displayHeight, text_Paint);
}
/**
* 繪製右按鈕
*
* @param canvas
*/
private void drawRightIcon(Canvas canvas) {
Log.d(TAG, "drawLeftIcon: getWidht = " + viewWidth + " indexRightX= " + indexRightX + " indexRightY = " + indexRightY);
if (clickedType == CLICK_TYPE_RIGHT) {
iconPaint.setMaskFilter(new BlurMaskFilter(50, BlurMaskFilter.Blur.INNER));
canvas.drawBitmap(clickIcon, indexRightX, indexRightY, iconPaint);
} else {
iconPaint.setMaskFilter(null);
canvas.drawBitmap(rightIcon, indexRightX, indexRightY, iconPaint);
}
int num = rightValue + minValue;
if (num < maxValue) {
num +=1;
}
canvas.drawText(num + "", (int) indexRightX + 15, displayHeight, text_Paint);
}
/**
* 回調接口,回調左右值
*/
public interface OnChanged {
void onChange(int leftValue, int rightValue);
}
/**
* 重寫onMeasure方法,設置wrap_content 時需要默認大小
*
* @param widthMeasureSpec
* @param heightMeasureSpec
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
int widthSpecSize = MeasureSpec.getSize(widthMeasureSpec);
int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
int heightSpecSize = MeasureSpec.getSize(heightMeasureSpec);
if (widthSpecMode == MeasureSpec.AT_MOST && heightSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(defaultWidth, defaultHeight);
} else if (widthSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(defaultWidth, heightSpecSize);
} else if (heightSpecMode == MeasureSpec.AT_MOST) {
setMeasuredDimension(widthSpecSize, defaultHeight);
}
}
}