自定义view之仿滴滴选择人数的SlideTab拖动选择器

这里只实现了逻辑,如果你需要可以自己把图片换了,设置一下图片的宽高,我这里只是用了两个点表示意思

一款用于选择的slidebar仿滴滴选择人数的效果,自己加了两个样式,可拖动和点击(还有一点文字居中的小问题,读者可以自行修改)

还是老规矩先看效果

1、


gif1.gif

2、

gif2.gif

3、

gif3.gif

4、特别说明要达到4这种效果,就要把代码中的那个两个todo的位置代码按todo处理一下

gif.gif

其实看过效果,第一感觉就是不难,无非就是监听手势滑动绘制view

相信看过之前我两篇自定义view的人都知道我自定义view的步骤了

1、分析需要的自定义属性
2、创建类这里继承view
3、创建attrs定义属性
4、完成自定义view所拥有的功能
这里我就不再一步一步创建了

先贴上自定义属性

<declare-styleable name="LSlideView">
        <attr name="round_radius" format="integer"/>
        <!--一下两个属性文章中没有用到,但是个人觉得必须要的,分别对应不同状态的颜色-->
        <attr name="select_color" format="color"/>
        <attr name="normal_color" format="color"/>

        <attr name="line_width" format="integer"/>
        <!--<attr name="text_width" format="integer"/>-->
        <attr name="margin_top" format="integer"/>
        <!--这两个属性是用来指定slidebar下面绘制的bitmap,文章中我是用的小圆点来指代的,但是像滴滴就是用车座位图标来显示,记得这四个属性是搭配的-->
        <attr name="normal_bitmap" format="reference"/>
        <attr name="select_bitmap" format="reference"/>
       <attr name="bitmap_width" format="integer"/>
        <attr name="bitmap_height" format="integer"/>

        <attr name="stytle_drag_view" format="enum">
            <enum name="FILL" value="1"/>
            <enum name="EMPTY" value="2"/>
        </attr>
        <attr name="is_draw_down_bitmap" format="boolean"/>
    </declare-styleable>

布局文件

<com.example.laer.myapplication.LSlideView
        android:id="@+id/dv"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="100px"
        android:clickable="true"//这里记得加上,要不然不走onTeach,暂未明白为啥,知道的可以给我说一下
        app:normal_bitmap="@drawable/normal"
        app:select_bitmap="@drawable/select"
        app:round_radius="60"
        app:line_width="20"
        app:margin_top="50"
        app:stytle_drag_view="EMPTY"
        app:is_draw_down_bitmap="false"/>

activity中获取选中为位置的方法

LSlideView dv = (LSlideView) findViewById(R.id.dv);
        dv.setCallBack(new LSlideView.CallBack() {
            @Override
            public void onCallBack(int postion) {
                Toast.makeText(MainActivity.this,postion+"",Toast.LENGTH_SHORT).show();
            }
        });

下面我就不再具体讲view中的代码了,直接贴代码,如果有需要我说的,可以留言


/**
 * 注:使用该控件时记得在布局文件中添加android:clickable="true"
 * 否则onTeach无效
 * Created by laer on 2016/10/31.
 */

public class LSlideView extends View {

    private Paint roundPaint;
    private Paint textPaint;
    private Paint line_paint;

    private Bitmap normal_bt;
    private Bitmap select_bt;

    private int widthView;//当前view的宽度
    private int r;//圆的半径
    private int line_width;//线的宽度
    private int margin_round;//每个圆之间的间距
    private int stopX;//当前结束绘制线的结束点的x坐标
    private int postion;//当前选中的位置
    private int normal_color;//默认的颜色,暂未使用,后期如果需要使用到可以自己添加
    private int select_color;//选中的颜色
    private int width_bitmap;// bitmap的宽
    private int height_bitmap;
    private int margin_top;//下面图标距上面圆的间距
    private int style;//样式
    private boolean is_draw_down_bitmap;//是否绘制下面的图片
    private CallBack callBack;

    public void setCallBack(CallBack callBack) {
        this.callBack = callBack;
    }

    public LSlideView(Context context) {
        super(context);
    }

    public LSlideView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context,attrs);
    }

    private void init(Context context, AttributeSet attrs) {
        TypedArray array = context.obtainStyledAttributes(attrs, R.styleable.LSlideView);
        line_width = array.getInt(R.styleable.LSlideView_line_width, 20);
        r = array.getInt(R.styleable.LSlideView_round_radius, 50);
        normal_color=array.getColor(R.styleable.LSlideView_normal_color,Color.GRAY);
        select_color=array.getColor(R.styleable.LSlideView_select_color,Color.BLUE);
        height_bitmap=array.getInt(R.styleable.LSlideView_bitmap_height,40);
        width_bitmap=array.getInt(R.styleable.LSlideView_bitmap_width,60);
        margin_top=array.getInt(R.styleable.LSlideView_margin_top,80);
        style=array.getInt(R.styleable.LSlideView_stytle_drag_view,2);
        is_draw_down_bitmap = array.getBoolean(R.styleable.LSlideView_is_draw_down_bitmap, true);
        BitmapDrawable normal = (BitmapDrawable) array.getDrawable(R.styleable.LSlideView_normal_bitmap);
        BitmapDrawable select_draw = (BitmapDrawable) array.getDrawable(R.styleable.LSlideView_select_bitmap);

        roundPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        roundPaint.setStyle(Paint.Style.FILL);

        line_paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        line_paint.setStrokeWidth(line_width);

        textPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        textPaint.setFakeBoldText(true);
        textPaint.setTextSize(r);

        normal_bt = Bitmap.createScaledBitmap(normal.getBitmap(), width_bitmap, height_bitmap, true);
        Bitmap select = select_draw.getBitmap();
        select_bt = Bitmap.createScaledBitmap(select,width_bitmap, height_bitmap, true);
        if (height_bitmap==0||width_bitmap==0){
            width_bitmap=2*r;
            height_bitmap=width_bitmap*select.getHeight()/select.getWidth();
        }
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        widthView = MeasureSpec.getSize(widthMeasureSpec);
        margin_round=(widthView-2*r)/4;
        setMeasuredDimension(widthView,r*2+margin_top+height_bitmap);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:

                break;
            case MotionEvent.ACTION_MOVE:
                stopX= (int) event.getX();
                postion = (int) ((event.getX() -r)/ margin_round);
                postInvalidate();
                return true;
            case MotionEvent.ACTION_UP:
                float i = (event.getX() -r)/ margin_round;
                //分离整数和小数部分
                int i2= (int) i;//整数
                if (i2>=3){//防止滑动的太远,导致i2大于3,导致显示不正常,限定了最多到4
                    i2=3;
                }
                float i3=i-i2;//小数
                if (i3 >0.5){//大于一半,则加1
                    postion=i2+1;
                    stopX=r+margin_round*(i2+1);
                }else{
                    stopX=r+margin_round*i2;
                    postion=i2;
                }
                callBack.onCallBack(postion);
                postInvalidate();
                return true;

        }
        return super.onTouchEvent(event);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
         //绘制默认的线
        line_paint.setColor(Color.GRAY);
        canvas.drawLine(r,r,widthView-r,r, line_paint);
       /* 
        *todo 1、将这里注释起来
       //绘制选中后的蓝色线
        if (postion!=0&&style==1){
            line_paint.setColor(Color.BLUE);
            canvas.drawLine(r,r,stopX,r, line_paint);
        }*/
        roundPaint.setStyle(Paint.Style.FILL_AND_STROKE);
        for (int i = 0; i < 5; i++) {
            //绘制节点
            float v = textPaint.measureText(i+"");
            int cx = r + margin_round * i;

            if (i==postion&&style==2){//绘制选中后的
                roundPaint.setColor(Color.BLUE);
                canvas.drawCircle(cx,r,r, roundPaint);

                textPaint.setColor(Color.WHITE);
                canvas.drawText(i+"",cx-v/2,r+r/2, textPaint);
                if (is_draw_down_bitmap)
                canvas.drawBitmap(select_bt,cx-width_bitmap/2,2*r+margin_top,line_paint);
            }else if (i==postion&&style==1){//todo 2、将这里的">="改成"=="
                roundPaint.setColor(Color.BLUE);
                canvas.drawCircle(cx,r,r, roundPaint);

                textPaint.setColor(Color.WHITE);
                canvas.drawText(i+"",cx-v/2,r+r/2, textPaint);
                if (is_draw_down_bitmap)
                canvas.drawBitmap(select_bt,cx-width_bitmap/2,2*r+margin_top,line_paint);
            }else {//默认的线和圆
                roundPaint.setColor(Color.GRAY);
                canvas.drawCircle(cx,r,r, roundPaint);

                textPaint.setColor(Color.BLACK);
                canvas.drawText(i+"",cx-v/2,r+r/2, textPaint);
                if (is_draw_down_bitmap&&style!=2)
                canvas.drawBitmap(normal_bt,cx-width_bitmap/2,2*r+margin_top,line_paint);
            }
        }
    }
    public interface CallBack{
        void onCallBack(int postion);
    }
}

补充说明

将 setClickable(true);//令自己可点击,从而获取触摸事件
加到自定义view中即可获取事件监听,就可以不在xml中添加clickble

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,928评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,077评论 4 62
  • 大家好, 我是汉家松鼠cg,汉家松鼠工作室创始人/CEO,《金庸群侠传X》、《江湖X》制作人。 上一篇写了我们团队...
    汉家松鼠阅读 1,109评论 1 2
  • 九重山后是什么?一片祥和,一份安稳,还是一份荆棘?不知如何下笔,不懂该如何让文字稳重,拙笔之下确实难表达出心虚的万...
    十二柒柒阅读 287评论 0 0
  • 时间过得飞快,《通往财富自由之路》第二季都要开始了! 还记得一年前那会,订阅这个专栏时的情景。当时读了几篇笑来老师...
    一切才刚刚开始阅读 183评论 1 1