android 打造不同的Seekbar
最近项目需要用到双向的seekbar,网上找了好多野不能达到要求,偶然一次机会看到了大众点评的例子,然后我最他做了优化,并对常用的seekbar做了总结.
向上两张图:
比如双向seekbar
public class SimpleRangeSeekBar extends View { private int lineWidth = 5; private int textSize = 25; private int inRangeColor = 0xff26b5f5; private int outRangeColor = 0xffC2C2C2; private int textColor = 0xff0000ff; private int textMarginBottom = 10; private int lowerCenterX; private int upperCenterX; private int bmpWidth; private int bmpHeight; private Bitmap lowerBmp; private Bitmap upperBmp; private boolean isLowerMoving = false; private boolean isUpperMoving = false; private OnRangeChangedListener onRangeChangedListener; private int paddingLeft = 50; private int paddingRight = 50; private int paddingTop = 10; private int paddingBottom = 10; private int lineHeight=20; private int lineLength = Utils.getScreenWidth(getContext())-paddingLeft-paddingRight-50; private int lineStart = paddingLeft; private int lineEnd = lineLength + paddingLeft; private int smallRange = 0; private int maxRange = 100; private int textHeight; public SimpleRangeSeekBar(Context context, AttributeSet attrs) { this(context, attrs,-1); } public SimpleRangeSeekBar(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); init(context, attrs, defStyle); } private void init(Context context, AttributeSet attrs, int defStyle) { initAttr(context, attrs, defStyle); initView(); } private void initView() { lowerBmp = BitmapFactory.decodeResource(getResources(), R.drawable.seekbar_n); upperBmp = BitmapFactory.decodeResource(getResources(), R.drawable.seekbar_n); bmpWidth = upperBmp.getWidth(); bmpHeight = upperBmp.getHeight(); lowerCenterX = lineStart; upperCenterX = lineEnd; lineHeight = getHeight() - paddingBottom - lowerBmp.getHeight() / 2; textHeight = lineHeight + lowerBmp.getHeight() / 2 + 10; } private void initAttr(Context context, AttributeSet attrs, int defStyle) { TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.RangeSeekBar, defStyle, 0); // lineWidth = typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_layout_width, UIUtils.getScreenWidth(getContext())-paddingLeft-paddingRight-50); // lineHeight=typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_layout_height,15); paddingLeft = typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_paddingLeft, 50); paddingRight= typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_paddingRight, 50); paddingTop= typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_paddingTop, 30); paddingBottom= typedArray.getDimensionPixelSize(R.styleable.RangeSeekBar_paddingBottom, 20); smallRange = typedArray.getInt(R.styleable.RangeSeekBar_line_small_range,0); maxRange = typedArray.getInt(R.styleable.RangeSeekBar_line_max_range,100); typedArray.recycle(); } private int measureWidth(int measureSpec) { int result = 0; int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); if (specMode == MeasureSpec.EXACTLY) { result = specSize; } else { result = paddingLeft + paddingRight + bmpWidth * 2; if (specMode == MeasureSpec.AT_MOST) { result = Math.min(result, specSize); } } return result; } private int measureHeight(int measureHeight) { int result = 0; int specMode = MeasureSpec.getMode(measureHeight); int specSize = MeasureSpec.getSize(measureHeight); if (specMode == MeasureSpec.EXACTLY) { result = bmpHeight * 2; } else { result = bmpHeight + paddingTop; if (specMode == MeasureSpec.AT_MOST) { result = Math.min(result, specSize); } } return result; } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { widthMeasureSpec = measureWidth(widthMeasureSpec); heightMeasureSpec = measureHeight(heightMeasureSpec); setMeasuredDimension(widthMeasureSpec, heightMeasureSpec); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); bmpWidth = upperBmp.getWidth(); bmpHeight = upperBmp.getHeight(); lineHeight = getHeight() - paddingBottom - lowerBmp.getHeight() / 2; textHeight = lineHeight - bmpHeight / 2 - textMarginBottom; // 画线 Paint linePaint = new Paint(); linePaint.setAntiAlias(true); linePaint.setStrokeWidth(lineWidth); // 绘制处于图片滑块之间线段 linePaint.setColor(inRangeColor); canvas.drawLine(lowerCenterX, lineHeight, upperCenterX, lineHeight, linePaint); // 绘制处于图片滑块两端的线段 linePaint.setColor(outRangeColor); canvas.drawLine(lineStart, lineHeight, lowerCenterX, lineHeight, linePaint); canvas.drawLine(upperCenterX, lineHeight, lineEnd, lineHeight, linePaint); // 画图片滑块 Paint bmpPaint = new Paint(); canvas.drawBitmap(lowerBmp, lowerCenterX - bmpWidth / 2, lineHeight - bmpHeight / 2, bmpPaint); canvas.drawBitmap(lowerBmp, upperCenterX - bmpWidth / 2, lineHeight - bmpHeight / 2, bmpPaint); // 画范围文字 drawText(canvas); } private void drawText(Canvas canvas) { Paint textPaint = new Paint(); textPaint.setColor(textColor); textPaint.setTextSize(textSize); textPaint.setAntiAlias(true); textPaint.setStrokeWidth(lineWidth); canvas.drawText(String.valueOf(smallRange), lowerCenterX - bmpWidth / 2, textHeight, textPaint); canvas.drawText(String.valueOf(maxRange), upperCenterX - bmpWidth / 2, textHeight, textPaint); } @Override public boolean onTouchEvent(MotionEvent event) { super.onTouchEvent(event); float xPos = event.getX(); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: // 如果按下的位置在垂直方向没有与图片接触,则不会滑动滑块 float yPos = event.getY(); if (Math.abs(yPos - lineHeight) > bmpHeight / 2) { return false; } // 表示当前按下的滑块是左边的滑块 if (Math.abs(xPos - lowerCenterX) < bmpWidth / 2) { isLowerMoving = true; } // //表示当前按下的滑块是右边的滑块 if (Math.abs(xPos - upperCenterX) < bmpWidth / 2) { isUpperMoving = true; } // 单击左边滑块的左边线条时,左边滑块滑动到对应的位置 if (xPos >= lineStart && xPos <= lowerCenterX - bmpWidth / 2) { lowerCenterX = (int) xPos; updateRange(); postInvalidate(); } // 单击右边滑块的右边线条时, 右边滑块滑动到对应的位置 if (xPos <= lineEnd && xPos >= upperCenterX + bmpWidth / 2) { upperCenterX = (int) xPos; updateRange(); postInvalidate(); } break; case MotionEvent.ACTION_MOVE: if (isLowerMoving) { if (xPos >= lineStart && xPos < upperCenterX - bmpWidth) { lowerCenterX = (int) xPos; updateRange(); postInvalidate(); } } if (isUpperMoving) { if (xPos > lowerCenterX + bmpWidth && xPos < lineEnd) { upperCenterX = (int) xPos; updateRange(); postInvalidate(); } } break; case MotionEvent.ACTION_UP: isLowerMoving = false; isUpperMoving = false; break; default: break; } return true; } private int computeRange(int range) { return (range - lineStart) * (maxRange - smallRange) / lineLength + smallRange; } private void updateRange() { smallRange = computeRange(lowerCenterX); maxRange = computeRange(upperCenterX); if (null != onRangeChangedListener) { onRangeChangedListener.onRangeChanged(smallRange, maxRange); } } public void setOnRangeChangedListener( OnRangeChangedListener onRangeChangedListener) { this.onRangeChangedListener = onRangeChangedListener; } public interface OnRangeChangedListener { void onRangeChanged(int lowRange, int maxRange); } }
为了方便大家使用我使用了自定义属性方便大家配置。
大家有兴趣还是去我的链接下载吧
http://download.csdn.net/detail/xiangzhihong8/9578539
android 打造不同的Seekbar的更多相关文章
- Android 打造炫目的圆形菜单 秒秒钟高仿建行圆形菜单
原文:Android 打造炫目的圆形菜单 秒秒钟高仿建行圆形菜单 转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/43131133, ...
- Android 打造自己的ImageLoader
Android 打造自己的ImageLoader 学习和参考 Android开发艺术探索 https://blog.csdn.net/column/details/15318.html 郭霖大神的Gl ...
- Android 打造完美的侧滑菜单/侧滑View控件
概述 Android 打造完美的侧滑菜单/侧滑View控件,完全自定义实现,支持左右两个方向弹出,代码高度简洁流畅,兼容性高,控件实用方便. 详细 代码下载:http://www.demodashi. ...
- Android基础控件SeekBar拖动条的使用
1.简介 SeekBar继承ProgressBar,相关属性和三种不同状态下的触发方法: <!--<SeekBar--> <!--android:layout_width=&q ...
- Android打造属于自己的数据库操作类。
1.概述 开发Android的同学都知道sdk已经为我们提供了一个SQLiteOpenHelper类来创建和管理SQLite数据库,通过写一个子类去继承它,就可以方便的创建.管理数据库.但是当我们需要 ...
- Android课程---Android 如何用滑杆(SeekBar)组件设置图片颜色的透明度(转)
Android系统支持的颜色是由4个值组成的,前3个为RGB,也就是我们常说的三原色(红.绿.蓝),最后一个值是A,也就是Alpha.这4个值都在0~255之间.颜色值越小,表示该颜色越淡,颜色值越大 ...
- android学习笔记八——SeekBar
SeekBar——拖动条 拖动条(SeekBar)组件与ProgressBar水平形式的显示进度条类似,不过其最大的区别在于,拖动条可以由用户自己进行手工的调节,例如当用户需要调整播放器音量或者电影的 ...
- android打造万能的适配器(转)
荒废了两天,今天与大家分享一个ListView的适配器 前段时间在学习慕课网的视频,觉得这种实现方式较好,便记录了下来,最近的项目中也使用了多次,节省了大量的代码,特此拿来与大家分享一下. 还是先看图 ...
- Android 拖动条(SeekBar)实例 附完整demo项目代码
1.拖动条的事件 实现SeekBar.OnSeekBarChangeListener接口.需要监听三个事件:数值改变(onProgressChanged)开始拖动(onStartTrackingTou ...
随机推荐
- iOS控制反转(IoC)与依赖注入(DI)的实现
背景 最近接触了一段时间的SpringMVC,对其控制反转(IoC)和依赖注入(DI)印象深刻,此后便一直在思考如何使用OC语言较好的实现这两个功能.Java语言自带的注解特性为IoC和DI带来了极大 ...
- MPAndroidChart图形联动
MPAndroidChart图形联动 本篇基于博客MPAndroidChart的K线图上添加均线,两个MPAndroidChart是有联动效果的 原理 获取正在滑动的Chart的触摸事件,将事件同步给 ...
- Premake可生成vcxproj.filters
Premake可生成vcxproj.filters (金庆的专栏) 添加 vcxproj.filters 文件可以用目录结构组织源文件. 例如premake5添加所有文件: files { ...
- GIF动态图制作
GIF动态图制作 博客写了也有一阵了,一直好奇大牛的博客里demo的动态图是怎么做的,今天抽空研究了一下,找了一个软件,以后再发现有好的工具再继续推荐 GIF制作工具--LICEcap 效果要比下面的 ...
- JVM基础知识GC
在网上看到一篇很不错的讲解JVM GC的文章,看完之后觉得可以留着以后多看几遍便转载了下来.但是找了半天也没有找到原作者地址.抱歉不能标明原文地址了.以下是文章内容. 几年前写过一篇关于JVM调优的文 ...
- 1.Cocos2d-x-3.2编写3d打飞机,粒子管理器代码
Cocos2d-x中的一个单例效果: #ifndef __Moon3d__ParticleManager__ #define __Moon3d__ParticleManager__ #inclu ...
- ubuntu16.04主题美化和软件推荐
前几天把ubuntu从15.10更新到了16.10,在网上看到有很多直接更新出问题的,正好赶上换SSD,于是采用全新安装,之前用ubuntu的时候装软件最让人头疼了,这回又得头疼一次了!! 索性把他记 ...
- Linux系统调用的实现机制分析
API/POSIX/C库的关系 系统调用的实现 3.1 系统调用处理程序 添加新系统调用 给Linux添加一个新的系统调用是件相对容易的工作.怎样设计和实现一个系统调用是难题所在, ...
- JavaScript中的三种弹出对话框
学习过js的小伙伴会发现,我们在一些实例中用到了alert()方法.prompt()方法.prompt()方法,他们都是在屏幕上弹出一个对话框,并且在上面显示括号内的内容,使用这种方法使得页面的交互性 ...
- Struts2进阶(一)运行原理及搭建步骤
Struts2进阶(一)运行原理 Struts2框架 Struts2框架搭建步骤 致力于web服务,不可避免的涉及到编程实现部分功能.考虑使用到SSH框架中的Struts2.本篇文章只为深入理解Str ...