我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android如何自定义带有圆形进度条的可长按控件功能

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android如何自定义带有圆形进度条的可长按控件功能

今天小编给大家分享一下Android如何自定义带有圆形进度条的可长按控件功能的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

(一)继承自View并实现构造方法,代码如下:

public class LongClickView extends View {    public int DEFAULT_MAX_SECONDS = 15;    public int DEFAULT_ANNULUS_WIDTH = 5;    public int DEFAULT_ANNULUS_COLOR;    public int DEFAULT_RATE = 50;    private Paint mSmallCirclePaint;    private Paint mMiddenCirclePaint;    private Paint mBigCirclePaint;    private Paint mAngleCirclePaint;    private int mWidthSize;    private Timer mTimer;//计时器    private AtomicInteger mCount = new AtomicInteger(0);    private MyClickListener mMyClickListener;    private boolean mIsFinish = true;    private long mStartTime;//点击的时间    private long mEndTime;//点击结束的时间    private int mMaxSeconds;    private int mDelayMilliseconds;    private int mAnnulusColor;    private float mAnnulusWidth;    public interface MyClickListener {        void longClickFinish();//长按结束        void singleClickFinish();//单击结束    }    public void setMyClickListener(MyClickListener myClickListener) {        mMyClickListener = myClickListener;    }    public LongClickView(Context context) {        this(context, null);    }    public LongClickView(Context context, @Nullable AttributeSet attrs) {        this(context, attrs, 0);    }    public LongClickView(Context context, @Nullable AttributeSet attrs, int defStyleAttr)                 {        super(context, attrs, defStyleAttr);        getAttrs(context, attrs);        initView();    }}

(二)定义并获取自定义属性,属性以及获取属性代码如下:

attr_long_click_view.xml

<?xml version="1.0" encoding="utf-8"?><resources>    <declare-styleable name="LongClickView">        <attr name="maxSeconds" format="integer" />        <attr name="annulusWidth" format="integer" />        <attr name="annulusColor" format="color" />        <attr name="delayMilliseconds" format="integer" />    </declare-styleable></resources>
 private void getAttrs(Context context, @Nullable AttributeSet attrs) {        TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.LongClickView);        //maxSeconds 最大的秒数        mMaxSeconds = typedArray.getInt(R.styleable.LongClickView_maxSeconds, DEFAULT_MAX_SECONDS);        //annulusWidth 圆环的宽度        mAnnulusWidth = typedArray.getInt(R.styleable.LongClickView_annulusWidth, DEFAULT_ANNULUS_WIDTH);        //annulusColor 圆环的颜色        DEFAULT_ANNULUS_COLOR = context.getResources().getColor(R.color.color_grey);        mAnnulusColor = typedArray.getColor(R.styleable.LongClickView_annulusColor, DEFAULT_ANNULUS_COLOR);        //delayMilliseconds 进度条隔多少时间走一次,值越小走的越快,显得更流畅        mDelayMilliseconds = typedArray.getInt(R.styleable.LongClickView_delayMilliseconds, DEFAULT_RATE);    }

(三)定义画笔工具 的代码如下:

    private void initView() {        mBigCirclePaint = new Paint();        mSmallCirclePaint = new Paint();        mMiddenCirclePaint = new Paint();        mAngleCirclePaint = new Paint();        mBigCirclePaint.setStyle(Paint.Style.FILL);        mBigCirclePaint.setColor(Color.LTGRAY);        mBigCirclePaint.setAntiAlias(true);        mBigCirclePaint.setStrokeWidth(5);        mSmallCirclePaint.setStrokeWidth(5);        mSmallCirclePaint.setAntiAlias(true);        mSmallCirclePaint.setColor(Color.WHITE);        mSmallCirclePaint.setStyle(Paint.Style.FILL);        mMiddenCirclePaint.setStrokeWidth(5);        mMiddenCirclePaint.setAntiAlias(true);        mMiddenCirclePaint.setColor(Color.LTGRAY);        mMiddenCirclePaint.setStyle(Paint.Style.FILL);        mAngleCirclePaint.setStrokeWidth(5);        mAngleCirclePaint.setAntiAlias(true);        mAngleCirclePaint.setColor(mAnnulusColor);        mAngleCirclePaint.setStyle(Paint.Style.FILL);        ...//这里是长按监听    }

(四)onMeasure中测量大小,onDraw中绘制圆与扇形,代码如下:

onMeasure中,如果没有定义实际宽高就会使用父组件的宽高,如果有实际宽高便会使用自己的宽高

 @Override    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        super.onMeasure(widthMeasureSpec, heightMeasureSpec);        mWidthSize = MeasureSpec.getSize(widthMeasureSpec);        setMeasuredDimension(mWidthSize, mWidthSize);    }

onDraw中,一共有三层圆形填充绘制以及一层扇形填充绘制,先绘制最外层的灰色圆形,再根据此时的进度绘制一定角度的扇形,然后覆盖一层灰色的圆形,最后在覆盖上一层白色的中心圆,并且在绘制过程以及绘制结束时的中心圆半径不同。代码如下:

 @Override    protected void onDraw(Canvas canvas) {        canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2, mBigCirclePaint);//最外层的填充圆        RectF rectF = new RectF(0, 0, mWidthSize, mWidthSize);//进度扇形        if (mCount.get() > 0) {            //求出每一次定时器执行所绘制的扇形度数            float perAngle = 360f / mMaxSeconds / (1000f / mDelayMilliseconds);            canvas.drawArc(rectF, 0, perAngle * mCount.get(), true, mAngleCirclePaint);        }        canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2 - mAnnulusWidth, mMiddenCirclePaint);//中间一层灰色的圆        //最后绘制中心圆        if (mIsFinish) {            canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2 - mAnnulusWidth, mSmallCirclePaint);        } else {            canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 8, mSmallCirclePaint);        }        super.onDraw(canvas);    }

(五)监听长按监听开始定时器并刷新画布,监听触摸事件进行结束的回调,定时器使用的是Timer类,当时间超过自定义的最大秒数时就会自动停止,并定时刷新画布,代码如下:

        this.setOnLongClickListener(new OnLongClickListener() {            @Override            public boolean onLongClick(View v) {                mIsFinish = false;                mCount.set(0);                mTimer = new Timer();                mTimer.schedule(new TimerTask() {                    @Override                    public void run() {                        mCount.addAndGet(1);                        invalidate();                        if (mCount.get() * mDelayMilliseconds >= mMaxSeconds * 1000) {                            mCount.set(0);                            this.cancel();                            invalidate();                            mIsFinish = true;                            if (mMyClickListener != null) {                                mMyClickListener.longClickFinish();                            }                        }                    }                }, 0, mDelayMilliseconds);                return true;            }        });
 @Override    public boolean onTouchEvent(MotionEvent event) {        if (event.getAction() == MotionEvent.ACTION_UP) {            mEndTime = System.currentTimeMillis();            new MyAsyncTask().execute();        } else if (event.getAction() == MotionEvent.ACTION_DOWN) {            mStartTime = System.currentTimeMillis();        }        return super.onTouchEvent(event);    }

将定时器停止与停止后的判断逻辑放在AsyncTask中编写,确保定时器不会继续处理逻辑之后再做判断

    public class MyAsyncTask extends AsyncTask<Void, Void, Void> {        @Override        protected Void doInBackground(Void... voids) {            if (mTimer != null) {                mTimer.cancel();            }            return null;        }        @Override        protected void onPostExecute(Void aVoid) {            //使用时间戳的差来判断是单击或者长按            if (mEndTime - mStartTime > 1000) {                //防止在自动结束后松开手指又重新调用了一次长按结束的回调                if (!mIsFinish) {                    if (mMyClickListener != null) {                        mMyClickListener.longClickFinish();                    }                }            } else {                //若是单击就清除进度条                mCount.set(0);                invalidate();                if (mMyClickListener != null) {                    mMyClickListener.singleClickFinish();                }            }            mIsFinish = true;        }    }

结束后的回调类代码如下:

   public interface MyClickListener {        void longClickFinish();//长按结束        void singleClickFinish();//单击结束    }

最后,完整的代码如下,自定义属性上方有贴出来代码:

public class LongClickView extends View {    public int DEFAULT_MAX_SECONDS = 15;    public int DEFAULT_ANNULUS_WIDTH = 5;    public int DEFAULT_ANNULUS_COLOR;    public int DEFAULT_RATE = 50;    private Paint mSmallCirclePaint;    private Paint mMiddenCirclePaint;    private Paint mBigCirclePaint;    private Paint mAngleCirclePaint;    private int mWidthSize;    private Timer mTimer;//计时器    private AtomicInteger mCount = new AtomicInteger(0);    private MyClickListener mMyClickListener;    private boolean mIsFinish = true;    private long mStartTime;//点击的时间    private long mEndTime;//点击结束的时间    private int mMaxSeconds;    private int mDelayMilliseconds;    private int mAnnulusColor;    private float mAnnulusWidth;    public interface MyClickListener {        void longClickFinish();//长按结束        void singleClickFinish();//单击结束    }    public void setMyClickListener(MyClickListener myClickListener) {        mMyClickListener = myClickListener;    }    public LongClickView(Context context) {        this(context, null);    }    public LongClickView(Context context, @Nullable AttributeSet attrs) {        this(context, attrs, 0);    }    public LongClickView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {        super(context, attrs, defStyleAttr);        getAttrs(context, attrs);        initView();    }    private void getAttrs(Context context, @Nullable AttributeSet attrs) {        TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.LongClickView);        //maxSeconds 最大的秒数        mMaxSeconds = typedArray.getInt(R.styleable.LongClickView_maxSeconds, DEFAULT_MAX_SECONDS);        //annulusWidth 圆环的宽度        mAnnulusWidth = typedArray.getInt(R.styleable.LongClickView_annulusWidth, DEFAULT_ANNULUS_WIDTH);        //annulusColor 圆环的颜色        DEFAULT_ANNULUS_COLOR = context.getResources().getColor(R.color.color_grey);        mAnnulusColor = typedArray.getColor(R.styleable.LongClickView_annulusColor, DEFAULT_ANNULUS_COLOR);        //delayMilliseconds 进度条隔多少时间走一次,值越小走的越快,显得更流畅        mDelayMilliseconds = typedArray.getInt(R.styleable.LongClickView_delayMilliseconds, DEFAULT_RATE);    }    private static final String TAG = "LongClickView";    private void initView() {        mBigCirclePaint = new Paint();        mSmallCirclePaint = new Paint();        mMiddenCirclePaint = new Paint();        mAngleCirclePaint = new Paint();        mBigCirclePaint.setStyle(Paint.Style.FILL);        mBigCirclePaint.setColor(Color.LTGRAY);        mBigCirclePaint.setAntiAlias(true);        mBigCirclePaint.setStrokeWidth(5);        mSmallCirclePaint.setStrokeWidth(5);        mSmallCirclePaint.setAntiAlias(true);        mSmallCirclePaint.setColor(Color.WHITE);        mSmallCirclePaint.setStyle(Paint.Style.FILL);        mMiddenCirclePaint.setStrokeWidth(5);        mMiddenCirclePaint.setAntiAlias(true);        mMiddenCirclePaint.setColor(Color.LTGRAY);        mMiddenCirclePaint.setStyle(Paint.Style.FILL);        mAngleCirclePaint.setStrokeWidth(5);        mAngleCirclePaint.setAntiAlias(true);        mAngleCirclePaint.setColor(mAnnulusColor);        mAngleCirclePaint.setStyle(Paint.Style.FILL);        this.setOnLongClickListener(new OnLongClickListener() {            @Override            public boolean onLongClick(View v) {                mIsFinish = false;                mCount.set(0);                mTimer = new Timer();                mTimer.schedule(new TimerTask() {                    @Override                    public void run() {                        mCount.addAndGet(1);                        invalidate();                        if (mCount.get() * mDelayMilliseconds >= mMaxSeconds * 1000) {                            mCount.set(0);                            this.cancel();                            invalidate();                            mIsFinish = true;                            if (mMyClickListener != null) {                                mMyClickListener.longClickFinish();                            }                        }                    }                }, 0, mDelayMilliseconds);                return true;            }        });    }    @Override    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        super.onMeasure(widthMeasureSpec, heightMeasureSpec);        mWidthSize = MeasureSpec.getSize(widthMeasureSpec);        setMeasuredDimension(mWidthSize, mWidthSize);    }    @Override    protected void onDraw(Canvas canvas) {        canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2, mBigCirclePaint);//最外层的填充圆        RectF rectF = new RectF(0, 0, mWidthSize, mWidthSize);//进度扇形        if (mCount.get() > 0) {            //求出每一次定时器执行所绘制的扇形度数            float perAngle = 360f / mMaxSeconds / (1000f / mDelayMilliseconds);            canvas.drawArc(rectF, 0, perAngle * mCount.get(), true, mAngleCirclePaint);        }        canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2 - mAnnulusWidth, mMiddenCirclePaint);//中间一层灰色的圆        //最后绘制中心圆        if (mIsFinish) {            canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 2 - mAnnulusWidth, mSmallCirclePaint);        } else {            canvas.drawCircle(mWidthSize / 2, mWidthSize / 2, mWidthSize / 8, mSmallCirclePaint);        }        super.onDraw(canvas);    }    @Override    public boolean onTouchEvent(MotionEvent event) {        if (event.getAction() == MotionEvent.ACTION_UP) {            mEndTime = System.currentTimeMillis();            new MyAsyncTask().execute();        } else if (event.getAction() == MotionEvent.ACTION_DOWN) {            mStartTime = System.currentTimeMillis();        }        return super.onTouchEvent(event);    }    public class MyAsyncTask extends AsyncTask<Void, Void, Void> {        @Override        protected Void doInBackground(Void... voids) {            if (mTimer != null) {                mTimer.cancel();            }            return null;        }        @Override        protected void onPostExecute(Void aVoid) {            //使用时间戳的差来判断是单击或者长按            if (mEndTime - mStartTime > 1000) {                //防止在结束后松开手指有重新调用了一次长按结束的回调                if (!mIsFinish) {                    if (mMyClickListener != null) {                        mMyClickListener.longClickFinish();                    }                }            } else {                mCount.set(0);                invalidate();                if (mMyClickListener != null) {                    mMyClickListener.singleClickFinish();                }            }            mIsFinish = true;        }    }}

使用的代码如下:

activity_long_click_view.xml

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:app="http://schemas.android.com/apk/res-auto"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:gravity="center"    android:orientation="vertical">    <com.example.customerview.long_click_view.LongClickView        android:id="@+id/long_click_view"        android:layout_width="100dp"        android:layout_height="wrap_content"        app:annulusColor="@color/color_2196F3"        app:annulusWidth="20"        app:delayMilliseconds="40"        app:maxSeconds="4" />    <TextView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:layout_marginTop="20dp"        android:text="长按录制视频,单击拍照"        android:textColor="@color/colorBlack"        android:textSize="20dp" /></LinearLayout>

LongClickViewActivity.java

        mLongClickView.setMyClickListener(new LongClickView.MyClickListener() {            @Override            public void longClickFinish() {                runOnUiThread(new Runnable() {                    @Override                    public void run() {                        Toast.makeText(LongClickViewActivity.this, "长按结束", Toast.LENGTH_SHORT).show();                    }                });            }            @Override            public void singleClickFinish() {                runOnUiThread(new Runnable() {                    @Override                    public void run() {                        Toast.makeText(LongClickViewActivity.this, "单击结束", Toast.LENGTH_SHORT).show();                    }                });            }        });

以上就是“Android如何自定义带有圆形进度条的可长按控件功能”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android如何自定义带有圆形进度条的可长按控件功能

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android如何自定义带有圆形进度条的可长按控件功能

今天小编给大家分享一下Android如何自定义带有圆形进度条的可长按控件功能的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。(
2023-07-02

Android 实现自定义圆形进度条的功能

Android 实现自定义圆形进度条: Android 自定义view,在大多数项目中根据客户需求及用户的体验度来说,都要重新写控件的来展示漂亮的界面,这里就对圆形进度条说下如何实现。 绘制自定义的圆形进度条
2022-06-06

Android自定义View圆形进度条控件的方法

这篇文章主要讲解了Android自定义View圆形进度条控件的方法,内容清晰明了,对此有兴趣的小伙伴可以学习一下,相信大家阅读完之后会有帮助。继续练习自定义View,这次带来的圆形进度条控件与之前的圆形百分比控件大同小异,这次涉及到了渐变渲
2023-05-31

Android编程基于自定义View实现绚丽的圆形进度条功能示例

本文实例讲述了Android编程基于自定义View实现绚丽的圆形进度条功能。分享给大家供大家参考,具体如下: 本文包含两个组件,首先上效果图: 1.ProgressBarView1(支持拖动):2.ProgressBarView2(不同进度
2022-06-06

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录