Android自定义SeekBar(带数字显示跟着变化)

首先看一下最后实现的效果图

seekbar.png


可左右滑动,点击加减按钮动态改变数值和位置显示。

思路:

1.根据图上UI知道有三种状态
<li>未滑中区域的灰色底部状态</li>
<li>滑中的颜色改变状态</li>
<li>椭圆的边框颜色,和内部字体显示</li>

实现:

首先我们自定义一个类MySeekBar继承View。并实现最重要的onDraw,onMeasure方法。
1.首先定义一个样式,用来以后设置底部颜色,滑中颜色,字体大小,颜色等

资源样式
    <declare-styleable name="MySeekBar">
        <attr name="mTopBarColor" format="color"></attr> 滑动时候横条颜色
        <attr name="mBottomBarColor" format="color"></attr>默认的横条颜色
        <attr name="strokeColor" format="color"></attr>椭圆边框颜色
        <attr name="month" format="boolean"></attr>判断是按年份显示还是月份显示
        <attr name="mytext" format="string"></attr>获得textview显示的后缀
    </declare-styleable>

2.实现MySeekBar构造器。初始化资源,与需要的画笔。(注意获得资源后一定要使用recycle(),去进行回收)

  public MySeekBar(Context context, AttributeSet attrs) {
        super(context, attrs);
        TypedArray attr = getTypedArray(context, attrs, R.styleable.MySeekBar);
        mTopBarColor = attr.getColor(R.styleable.MySeekBar_mTopBarColor,     getResources().getColor(R.color.color_EB7046));//设置滑动时候横条颜色
        mBottomBarColor = attr.getColor(R.styleable.MySeekBar_mBottomBarColor, getResources().getColor(R.color.login_font));//默认的横条颜色
        month = attr.getBoolean(R.styleable.MySeekBar_month, false);//判断是按年份显示还是月份显示
        myText = attr.getString(R.styleable.MySeekBar_mytext);//获得textview显示的后缀
        mContext = context;
        Data(Calendar.getInstance());//得到当前的日期
        windows_width = DisplayMetricsTool.getWidth(mContext);//获得屏幕宽度用来适配各种屏幕
        dip_size = DisplayMetricsTool.dip2px(mContext, 75);
        //矩形的白色底部画笔
        mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        paint = new Paint(); //设置一个笔刷大小是3的黄色的画笔
        paint.setStyle(Paint.Style.FILL);//充满
        paint.setColor(Color.WHITE);
        paint.setAntiAlias(true);// 设置画笔的锯齿效果
        //矩形边框的的画笔
        whit = new Paint(); //设置一个笔刷大小是3的黄色的画笔
        whit.setStyle(Paint.Style.STROKE);
        whit.setColor(mTopBarColor);
        whit.setStrokeWidth(1);
        whit.setAntiAlias(true);// 设置画笔的锯齿效果
        //设置字体
        textPaint = new Paint();
        textPaint.setColor(mTopBarColor);
        textPaint.setTextSize(DisplayMetricsTool.canvasTextSize(mContext));
        textPaint.setStyle(Paint.Style.FILL);
        //该方法即为设置基线上那个点究竟是left,center,还是right  这里我设置为center
        textPaint.setTextAlign(Paint.Align.CENTER);
        textPaint.setAntiAlias(true);
        attr.recycle();
}

3.自定义一个方法设置进度值,默认满进度100.

   /**
     * 顶部进度条初始百分比进度
     */
    private float mOriginalPercent = 0f;
   /**
     * 设置初始进度值,默认进度最大值为100
     *
     * @param progress
     */

    public void setOriginalProgress(int progress) {
        if (month) {//按年显示
            progress = 100 - (year - progress);//总进度减去今年和传入年份的差值
        } else {
            //按月显示
            progress = (int) (progress * 8.4);//12月占100的百分比8.4
        }
        mOriginalPercent = progress / 100f;
        if (mOriginalPercent < 0) {
            mOriginalPercent = 0.0f;
        } else if (mOriginalPercent > 1.0f) {
            mOriginalPercent = 1.0f;
        }
    }

4.需要在onMeasure方法中计算,当前传入的进度换算之后占在屏幕的哪个位置。

    //获得密度转像素的大小
    private int dip_size;
    // view的宽
    private int mViewWidth;
   // view的高
   private int mViewHeight;
   // 底部进度条的宽
  private int mBottomBarWidth;

   //底部进度条左边位置
 private int mBottomBarLeft;
    // 底部进度条上边位置
 private int mBottomBarTop;
  // 底部进度条右边位置
 private int mBottomBarRight;
  //  底部进度条底边位置
private int mBottomBarBottom;
//  顶部进度条的右边位置
  private int mTopBarRight;
   @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);

        if (widthMode == MeasureSpec.AT_MOST) {
            mViewWidth = mTvWidth * 4;
        } else if (widthMode == MeasureSpec.EXACTLY) {
            if (mTvWidth * 4 >= widthSize) {
                mViewWidth = mTvWidth * 4;
            } else {
                mViewWidth = widthSize;
            }
        }

        widthMeasureSpec = MeasureSpec.makeMeasureSpec(mViewWidth, MeasureSpec.EXACTLY);

        if (heightMode == MeasureSpec.AT_MOST) {
            mViewHeight = mTvHeight + 4 * mPadding;
        } else if (heightMode == MeasureSpec.EXACTLY) {
            if (heightSize <= mTvHeight + 4 * mPadding) {
                mViewHeight = mTvHeight + 4 * mPadding;
            } else {
                mViewHeight = heightSize;
            }
        }

        heightMeasureSpec = MeasureSpec.makeMeasureSpec(mViewHeight, MeasureSpec.EXACTLY);

        mBottomBarWidth = mViewWidth - 2 * mTvWidth - 6 * mPadding;

        mBottomBarLeft = mTvWidth + 3 * mPadding;
        mBottomBarRight = mViewWidth - mTvWidth - 3 * mPadding;
        mBottomBarBottom = (mViewHeight - mTvHeight) / 2 + mTvHeight;
        mBottomBarTop = mBottomBarBottom - mBarHeight;

        mCircleY = mBottomBarBottom - mBarHeight / 2;//圆心的坐标

        mPosition = (int) Math.round((mBottomBarWidth - DisplayMetricsTool.getWidth(mContext) / 6) * mOriginalPercent + 0.5) + mBottomBarLeft;
        if (mPosition <= mBottomBarLeft) {
            mPosition = mBottomBarLeft;
        } else if (mPosition >= mBottomBarRight) {
            mPosition = mBottomBarRight;
        }

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }

5.因为需要左右滑动,所以我们采用GestureDetector的手势滑动管理,判断当用户点击在椭圆中才能让滑动改变

  GestureDetector   mGestureDetector = new GestureDetector(mContext, new GestureDetector.OnGestureListener() {

            @Override
            public boolean onSingleTapUp(MotionEvent e) {
                if (mWhereClickedState == WHERR_CLICKED_CIRCLE) {
                } else if (mWhereClickedState == WHERR_CLICKED_BAR) {
                    mPosition = (int) Math.round(e.getX() + 0.5);
                    if (mPosition >= mBottomBarRight) {
                        mPosition = mBottomBarRight;
                    } else if (mPosition <= mBottomBarLeft) {
                        mPosition = mBottomBarLeft;
                    }
                    MySeekBar.this.invalidate();
                }
                return false;
            }

            @Override
            public void onShowPress(MotionEvent e) {
                // TODO Auto-generated method stub

            }

            @Override
            public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX,
                                    float distanceY) {
              /*  if (mWhereClickedState == WHERR_CLICKED_CIRCLE) {*/
                mPosition = (int) Math.round(e2.getX() + 0.5);
                if (mPosition >= mBottomBarRight) {
                    mPosition = mBottomBarRight;
                } else if (mPosition <= mBottomBarLeft) {
                    mPosition = mBottomBarLeft;
                }
                MySeekBar.this.invalidate();
              /*  } else if (mWhereClickedState == WHERR_CLICKED_BAR) {
                }*/
                return false;
            }

            @Override
            public void onLongPress(MotionEvent e) {
                // TODO Auto-generated method stub

            }

            @Override
            public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
                                   float velocityY) {
                // TODO Auto-generated method stub
                return false;
            }

            @Override
            public boolean onDown(MotionEvent e) {
                float event_x = e.getX();
                float event_y = e.getY();
                mWhereClickedState = judgeWhereClicked(event_x, event_y);
                if (mWhereClickedState == WHERR_CLICKED_VIEW) {
                    return false;
                } else {
                    return true;
                }
            }
        });
    /**
     * 判断用户点击位置状态
     *
     * @param x 用户点击的x坐标
     * @param y 用户点击的y坐标
     * @return 返回用户点击未知状态:
     * WHERR_CLICKED_CIRCLE
     * WHERR_CLICKED_BAR
     * WHERR_CLICKED_VIEW
     */
    public int judgeWhereClicked(float x, float y) {
        if (rectF.contains(x, y)) {
            return WHERR_CLICKED_CIRCLE;  //s_x和s_y界定的区域,即圆圈区域
        } else {
            return WHERR_CLICKED_VIEW;  //view除了上述两部分的部分
        }
    }

6.现在画东西的笔有了,架子有了,装饰材料有了,只剩往一块白布里面画东西。就成了。 接下来最重要的就是在onDraw方法中,把UI画上去。

    @Override
    protected void onDraw(Canvas canvas) {


        mTopBarRight = mPosition;
        mCircleX = mTopBarRight;

        mPaint.setColor(mBottomBarColor);
        canvas.drawRect(mBottomBarLeft, mBottomBarTop, mBottomBarRight, mBottomBarBottom, mPaint);

        mPaint.setColor(mTopBarColor);
        canvas.drawRect(mBottomBarLeft, mBottomBarTop, mTopBarRight, mBottomBarBottom, mPaint);
        if (mBottomBarRight - mTopBarRight > windows_width / 6 - 10) {//用来防止百分比是100的时候椭圆形画出边界的问题。

            aaa = mTopBarRight;
            rectF = new RectF(mTopBarRight, dip_size / 2 - dip_size / 5, mTopBarRight + windows_width / 6, dip_size / 2 + dip_size / 5);

            canvas.drawRoundRect(rectF,

                    (windows_width / 6) / 3, //x轴的半径

                    (windows_width / 6) / 3 - 5, //y轴的半径

                    paint);//用来画矩形白色背景
            canvas.drawRoundRect(rectF,
                    (windows_width / 6) / 3, //x轴的半径

                    (windows_width / 6) / 3 - 5, //y轴的半径

                    whit);//用来画矩形边框
        } else {
            rectF = new RectF(aaa, dip_size / 2 - dip_size / 5, aaa + windows_width / 6, dip_size / 2 + dip_size / 5);
            canvas.drawRoundRect(rectF,

                    (windows_width / 6) / 3, //x轴的半径

                    (windows_width / 6) / 3 - 5, //y轴的半径

                    paint);//用来画矩形白色背景
            canvas.drawRoundRect(rectF,
                    (windows_width / 6) / 3, //x轴的半径

                    (windows_width / 6) / 3 - 5, //y轴的半径


                    whit);//用来画矩形边框
        }
        Paint.FontMetrics fontMetrics = textPaint.getFontMetrics();
        float top = fontMetrics.top;//为基线到字体上边框的距离,即上图中的top
        float bottom = fontMetrics.bottom;//为基线到字体下边框的距离,即上图中的bottom

        int baseLineY = (int) (rectF.centerY() - top / 2 - bottom / 2);//基线中间点的y轴计算公式
        canvas.drawText(getProgress() + myText, rectF.centerX(), baseLineY, textPaint);
    }

7.获得当前进度由于这个项目是按年或者月显示的,所以获取进度时和设置进度时都需要按照满进度值100来进行换算,设置获得进度的方法getProgress()。

/**
     * 获得当前进度值
     *
     * @return int
     */
    public int getProgress() {

        float percent = (mPosition - mBottomBarLeft) * 1.0f / (mBottomBarWidth -windows_width/ 6);
        if (percent < 0.0) {
            percent = 0f;
        } else if (percent > 1) {
            percent = 1f;
        }

        int progress = ((int) Math.round(percent * 100 + 0.5) - 1);
        if (progress <= 0) {
            progress = 1;
        } else if (progress > 100) {
            progress = 100;
        }
        if (month) {
            return year - (100 - progress);
        } else {
            return progress / 8 == 0 ? 1 : progress / 8;
        }

    }

8.项目需要按左右按钮的时候必要你动态设置进度值,并改变椭圆的位置,那么我们只需要改变进度值,然后不断去onDraw重新绘制界面就行了。代码如下

   /**
     * 通过传入进度值来更新进度条
     */
    public void updatePositionFromProgress(int progress) {
        if (month) {//按年显示
            progress = 100 - (year - progress);//总进度减去今年和传入年份的差值
        } else {
            //按月显示
            progress = (int) (progress * 8.4);
        }

        float percent = progress / 100f;
        if (percent < 0) {
            percent = 0.0f;
        } else if (percent > 1.0f) {
            percent = 1.0f;
        }
        mPosition = (int) Math.round((mBottomBarWidth -windows_width / 6)* percent + 0.5) + mBottomBarLeft;
        if (mPosition <= mBottomBarLeft) {
            mPosition = mBottomBarLeft;
        } else if (mPosition >= mBottomBarRight) {
            mPosition = mBottomBarRight;
        }
        this.invalidate();
    }

#######此时基本的实现已经完成。使用方式基本就是在XML设置颜色,字体之类的事,然后动态去设置进度--->代码如下

            <ui.view.MySeekBar
                android:id="@+id/seekbar"
                android:layout_width="match_parent"
                android:layout_height="75dp"
                android:layout_centerVertical="true"
                android:layout_toLeftOf="@+id/right_ll"
                android:layout_toRightOf="@+id/left_ll"
                app:mTopBarColor="@color/color_EB7046"
                app:month="true"
                app:mytext="年" />
             year = calendar.get(Calendar.YEAR);//当前的年
                seekbar = (MySeekBar) view.findViewById(R.id.seekbar);
                seekbar.setOriginalProgress(year);
至此整个功能差不多已经完成。刚开始的时候也是研究了网上有关的项目去进行研究,然后自己重新更改封装,弄成符合自己项目需要东西,希望能给有需要的同学一点思路,不要想轮子能完全给你解决项目的难点,而是提供给你一个解决问题的思路,google,github,爆栈等等,有难点的时候大家可以好好利用。

<li>下一篇准备写自定义recyclerview的上拉加载功能,因为看了网上大多数的开源项目,基本上都是华丽但不实用,功能太多真正能在项目用得上的很少,增加了方法树,包的大小,所以将会写一个只有上拉加载的recyclerview功能<li>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容