热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android自定义双向进度条的实现代码

想整个双向的进度条,就是可以选取播放范围的。 像这样: 然而官方控件里只有单向的。不要慌,我们自己画一个。 绘制一个进

想整个双向的进度条,就是可以选取播放范围的。

像这样:


然而官方控件里只有单向的。不要慌,我们自己画一个。

绘制一个进度条主要是三方面。1.样式,2.尺寸,3.操作监听。

完整代码来一遍:

注释基本上就把原理说明了一下。

package util;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.drawable.Drawable;
import android.support.v4.content.ContextCompat;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

import com.example.qzd.utildemo.R;

import java.math.BigDecimal;

/**
 * 双向滑块的进度条(区域选择)
 */
public class SeekRangeBar extends View {
  private Context _context;
  private static final int CLICK_ON_LOW = 1;    //手指在前滑块上滑动
  private static final int CLICK_ON_HIGH = 2;    //手指在后滑块上滑动
  private static final int CLICK_IN_LOW_AREA = 3;  //手指点击离前滑块近
  private static final int CLICK_IN_HIGH_AREA = 4; //手指点击离后滑块近
  private static final int CLICK_OUT_AREA = 5;   //手指点击在view外
  private static final int CLICK_INVAILD = 0;
  private static final int[] STATE_NORMAL = {};
  private static final int[] STATE_PRESSED =
 {android.R.attr.state_pressed,android.R.attr.state_window_focused,};
  private static int mThumbMarginTop = 0;  //滑动块顶部离view顶部的距离
  private static int mTextViewMarginTop = 0;  //当前滑块文字距离view顶部距离
  private Drawable hasScrollBarBg;    //滑动条滑动后背景图
  private Drawable notScrollBarBg;    //滑动条未滑动背景图
  private Drawable mThumbLow;     //前滑块
  private Drawable mThumbHigh;    //后滑块
  private int mScollBarWidth;   //控件宽度 = 滑动条宽度 + 滑动块宽度
  private int mScollBarHeight;  //控件高度
  private int mThumbWidth;    //滑动块直径
  private double mOffsetLow = 0;   //前滑块中心坐标
  private double mOffsetHigh = 0;  //后滑块中心坐标
  private int mDistance=0;   //总刻度是固定距离 两边各去掉半个滑块距离
  private int mFlag = CLICK_INVAILD;  //手指按下的类型
  private double defaultScreenLow = 0;  //默认前滑块位置百分比
  private double defaultScreenHigh = 100; //默认后滑块位置百分比
  private OnSeekBarChangeListener mBarChangeListener;
  private boolean editable=false;//是否处于可编辑状态
  private int miniGap=5;//AB的最小间隔
  private double progressLow;//起点(百分比)
  private double progressHigh;//终点

  public SeekRangeBar(Context context) {
    this(context, null);
  }
  public SeekRangeBar(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
  }
  public SeekRangeBar(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    _cOntext=context;
    //这里设置背景图及滑块图,自定义过进度条的同学应该很熟悉了
    notScrollBarBg = ContextCompat.getDrawable(_context,R.mipmap.hp_wbf);
    hasScrollBarBg = ContextCompat.getDrawable(_context, R.mipmap.hp_ybf);
    mThumbLow = ContextCompat.getDrawable(_context,R.mipmap.hp_a);
    mThumbHigh = ContextCompat.getDrawable(_context,R.mipmap.hp_b);
    mThumbLow.setState(STATE_NORMAL);
    mThumbHigh.setState(STATE_NORMAL);
    //设置滑动条高度
    mScollBarHeight = notScrollBarBg.getIntrinsicHeight();
    //设置滑动块直径
    mThumbWidth = mThumbLow.getIntrinsicWidth();
  }

  /**
   * 测量view尺寸(在onDraw()之前)
   * @param widthMeasureSpec
   * @param heightMeasureSpec
   */
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int width = MeasureSpec.getSize(widthMeasureSpec);
    mScollBarWidth = width;
    if(mDistance==0) {//这里滑块中心坐标初始化的时候测量一下(根据mDistance是否赋值判断),并不需要不停地去测量。后面会根据进度计算滑块位置。
      mOffsetLow = mThumbWidth / 2;
      mOffsetHigh = width - mThumbWidth / 2;
    }
    mDistance = width - mThumbWidth;
    if(defaultScreenLow != 0) {
      mOffsetLow = formatInt(defaultScreenLow / 100 * (mDistance)) + mThumbWidth / 2;
    }
    if(defaultScreenHigh != 100) {
      mOffsetHigh = formatInt(defaultScreenHigh / 100 * (mDistance)) + mThumbWidth / 2;
    }
    setMeasuredDimension(width, mThumbWidth + mThumbMarginTop + 2);
  }

  protected void onLayout(boolean changed, int l, int t, int r, int b) {
    super.onLayout(changed, l, t, r, b);
  }

  /**
   * 绘制进度条
   */
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    //设置绘制样式
    Paint text_Paint = new Paint();
    text_Paint.setTextAlign(Paint.Align.CENTER);
    text_Paint.setColor(Color.RED);
    text_Paint.setTextSize(20);

    int top = mThumbMarginTop + mThumbWidth / 2 - mScollBarHeight / 2;
    int bottom = top + mScollBarHeight;

    //绘制是否可操作状态的下的不同样式,仅可编辑状态下显示进度条
    if(editable) {
      //白色滑动条,两个滑块各两边部分
      notScrollBarBg.setBounds(mThumbWidth / 2, top, mScollBarWidth - mThumbWidth / 2, bottom);
      notScrollBarBg.draw(canvas);

      //红色滑动条,两个滑块中间部分
      hasScrollBarBg.setBounds((int) mOffsetLow, top, (int) mOffsetHigh, bottom);
      hasScrollBarBg.draw(canvas);
    }

    //前滑块
    mThumbLow.setBounds((int) (mOffsetLow - mThumbWidth / 2), mThumbMarginTop, (int) (mOffsetLow + mThumbWidth / 2), mThumbWidth + mThumbMarginTop);
    mThumbLow.draw(canvas);

    //后滑块
    mThumbHigh.setBounds((int) (mOffsetHigh - mThumbWidth / 2), mThumbMarginTop, (int) (mOffsetHigh + mThumbWidth / 2), mThumbWidth + mThumbMarginTop);
    mThumbHigh.draw(canvas);

    //当前滑块刻度
    progressLow = formatInt((mOffsetLow - mThumbWidth / 2) * 100 / mDistance);
    progressHigh = formatInt((mOffsetHigh - mThumbWidth / 2) * 100 / mDistance);
    canvas.drawText((int) progressLow + "", (int) mOffsetLow - 2 - 2, mTextViewMarginTop, text_Paint);
    canvas.drawText((int) progressHigh + "", (int) mOffsetHigh - 2, mTextViewMarginTop, text_Paint);

    if (mBarChangeListener != null) {
      mBarChangeListener.onProgressChanged(this, progressLow, progressHigh);
    }
  }

  //手势监听
  @Override
  public boolean onTouchEvent(MotionEvent e) {
    if(!editable) {
      return false;
    }
    if (e.getAction() == MotionEvent.ACTION_DOWN) {
      mFlag = getAreaFlag(e);
      if (mFlag == CLICK_ON_LOW) {
        mThumbLow.setState(STATE_PRESSED);
      } else if (mFlag == CLICK_ON_HIGH) {
        mThumbHigh.setState(STATE_PRESSED);
      } else if (mFlag == CLICK_IN_LOW_AREA) {
        mThumbLow.setState(STATE_PRESSED);
        mThumbHigh.setState(STATE_NORMAL);
        //如果点击0-mThumbWidth/2坐标
        if (e.getX() <0 || e.getX() <= mThumbWidth / 2) {
          mOffsetLow = mThumbWidth / 2;
        } else if (e.getX() > mScollBarWidth - mThumbWidth / 2) {
          mOffsetLow = mThumbWidth / 2 + mDistance;
        } else {
          mOffsetLow = formatInt(e.getX());

        }
      } else if (mFlag == CLICK_IN_HIGH_AREA) {
        mThumbHigh.setState(STATE_PRESSED);
        mThumbLow.setState(STATE_NORMAL);
        if (e.getX() >= mScollBarWidth - mThumbWidth / 2) {
          mOffsetHigh = mDistance + mThumbWidth / 2;
        } else {
          mOffsetHigh = formatInt(e.getX());
        }
      }
      //更新滑块
      invalidate();
    } else if (e.getAction() == MotionEvent.ACTION_MOVE) {
      if (mFlag == CLICK_ON_LOW) {
        if (e.getX() <0 || e.getX() <= mThumbWidth / 2) {
          mOffsetLow = mThumbWidth / 2;
        } else if (e.getX() >= mScollBarWidth - mThumbWidth / 2) {
          mOffsetLow = mThumbWidth / 2 + mDistance;
          mOffsetHigh = mOffsetLow;
        } else {
          mOffsetLow = formatInt(e.getX());
          if (mOffsetHigh - mOffsetLow <= 0) {
            mOffsetHigh = (mOffsetLow <= mDistance + mThumbWidth / 2) &#63; (mOffsetLow) : (mDistance + mThumbWidth / 2);
          }
        }
      } else if (mFlag == CLICK_ON_HIGH) {
        if (e.getX()  mScollBarWidth - mThumbWidth / 2) {
          mOffsetHigh = mThumbWidth / 2 + mDistance;
        } else {
          mOffsetHigh = formatInt(e.getX());
          if (mOffsetHigh - mOffsetLow <= 0) {
            mOffsetLow = (mOffsetHigh >= mThumbWidth / 2) &#63; (mOffsetHigh) : mThumbWidth / 2;
          }
        }
      }
      //更新滑块,每次滑块有动作都要执行此函数触发onDraw方法绘制新图片
      invalidate();
    } else if (e.getAction() == MotionEvent.ACTION_UP) {
      Log.d("LOGCAT","ACTION UP:"+progressHigh+"-"+progressLow);
      mThumbLow.setState(STATE_NORMAL);
      mThumbHigh.setState(STATE_NORMAL);
      if(miniGap>0 && progressHigh= top && e.getY() <= bottom && e.getX() >= (mOffsetLow - mThumbWidth / 2) && e.getX() <= mOffsetLow + mThumbWidth / 2) {
      return CLICK_ON_LOW;
    } else if (e.getY() >= top && e.getY() <= bottom && e.getX() >= (mOffsetHigh - mThumbWidth / 2) && e.getX() <= (mOffsetHigh + mThumbWidth / 2)) {
      return CLICK_ON_HIGH;
    } else if (e.getY() >= top
      && e.getY() <= bottom
      && ((e.getX() >= 0 && e.getX() <(mOffsetLow - mThumbWidth / 2)) || ((e.getX() > (mOffsetLow + mThumbWidth / 2))
      && e.getX() <= ((double) mOffsetHigh + mOffsetLow) / 2))) {
      return CLICK_IN_LOW_AREA;
    } else if (e.getY() >= top && e.getY() <= bottom && (((e.getX() > ((double) mOffsetHigh + mOffsetLow) / 2) && e.getX() <(mOffsetHigh - mThumbWidth / 2)) || (e.getX() > (mOffsetHigh + mThumbWidth / 2) && e.getX() <= mScollBarWidth))) {
      return CLICK_IN_HIGH_AREA;
    } else if (!(e.getX() >= 0 && e.getX() <= mScollBarWidth && e.getY() >= top && e.getY() <= bottom)) {
      return CLICK_OUT_AREA;
    } else {
      return CLICK_INVAILD;
    }
  }

  /**
   * 设置前滑块位置
   * @param progressLow
   */
  public void setProgressLow(double progressLow) {
    this.defaultScreenLow = progressLow;
    mOffsetLow = formatInt(progressLow / 100 * (mDistance)) + mThumbWidth / 2;
    invalidate();
  }

  /**
   * 设置后滑块位置
   * @param progressHigh
   */
  public void setProgressHigh(double progressHigh) {
    this.defaultScreenHigh = progressHigh;
    mOffsetHigh = formatInt(progressHigh / 100 * (mDistance)) + mThumbWidth / 2;
    invalidate();
  }

  /**
   * 设置滑动监听
   * @param mListener
   */
  public void setOnSeekBarChangeListener(OnSeekBarChangeListener mListener) {
    this.mBarChangeListener = mListener;
  }

  /**
   * 滑动监听,改变输入框的值
   */
  public interface OnSeekBarChangeListener {
    //滑动时
    public void onProgressChanged(SeekRangeBar seekBar, double progressLow, double progressHigh);
  }

  /**
   * 设置滑动结果为整数
   */
  private int formatInt(double value) {
    BigDecimal bd = new BigDecimal(value);
    BigDecimal bd1 = bd.setScale(0, BigDecimal.ROUND_HALF_UP);
    return bd1.intValue();
  }
}

然后就可以在程序中使用了。

布局中

调用

private SeekRangeBar doubleSeekbar;//双向进度条
doubleSeekbar = (SeekRangeBar) findViewById(R.id.doubleSeekbar);
//监听进度范围变化
doubleSeekbar.setOnSeekBarChangeListener(new SeekRangeBar.OnSeekBarChangeListener() {
  @Override
  public void onProgressChanged(SeekRangeBar seekBar, double progressLow, double progressHigh) {
    Log.d("LOGCAT","低:" + progressLow + "高:" + progressHigh);
  }
});

相关GitHub项目地址:https://github.com/codeqian/android-class-lib

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 本文详细介绍了在Centos7上部署安装zabbix5.0的步骤和注意事项,包括准备工作、获取所需的yum源、关闭防火墙和SELINUX等。提供了一步一步的操作指南,帮助读者顺利完成安装过程。 ... [详细]
  • 大坑|左上角_pycharm连接服务器同步写代码(图文详细过程)
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了pycharm连接服务器同步写代码(图文详细过程)相关的知识,希望对你有一定的参考价值。pycharm连接服务 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • imx6ull开发板驱动MT7601U无线网卡的方法和步骤详解
    本文详细介绍了在imx6ull开发板上驱动MT7601U无线网卡的方法和步骤。首先介绍了开发环境和硬件平台,然后说明了MT7601U驱动已经集成在linux内核的linux-4.x.x/drivers/net/wireless/mediatek/mt7601u文件中。接着介绍了移植mt7601u驱动的过程,包括编译内核和配置设备驱动。最后,列举了关键词和相关信息供读者参考。 ... [详细]
  • FeatureRequestIsyourfeaturerequestrelatedtoaproblem?Please ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • 原理:dismiss再弹出,把dialog设为全局对象。if(dialog!null&&dialog.isShowing()&&!(Activity.)isFinishing()) ... [详细]
  • GSIOpenSSH PAM_USER 安全绕过漏洞
    漏洞名称:GSI-OpenSSHPAM_USER安全绕过漏洞CNNVD编号:CNNVD-201304-097发布时间:2013-04-09 ... [详细]
  •     这里使用自己编译的hadoop-2.7.0版本部署在windows上,记得几年前,部署hadoop需要借助于cygwin,还需要开启ssh服务,最近发现,原来不需要借助cy ... [详细]
author-avatar
-林之涵_396
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有