Android自定義雙向進度條的實現(xiàn)代碼
想整個雙向的進度條,就是可以選取播放范圍的。
像這樣:
然而官方控件里只有單向的。不要慌,我們自己畫一個。
繪制一個進度條主要是三方面。1.樣式,2.尺寸,3.操作監(jiān)聽。
完整代碼來一遍:
注釋基本上就把原理說明了一下。
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; /** * 雙向滑塊的進度條(區(qū)域選擇) */ 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; //當(dāng)前滑塊文字距離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; //前滑塊中心坐標(biāo) private double mOffsetHigh = 0; //后滑塊中心坐標(biāo) private int mDistance=0; //總刻度是固定距離 兩邊各去掉半個滑塊距離 private int mFlag = CLICK_INVAILD; //手指按下的類型 private double defaultScreenLow = 0; //默認(rèn)前滑塊位置百分比 private double defaultScreenHigh = 100; //默認(rèn)后滑塊位置百分比 private OnSeekBarChangeListener mBarChangeListener; private boolean editable=false;//是否處于可編輯狀態(tài) 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; //這里設(shè)置背景圖及滑塊圖,自定義過進度條的同學(xué)應(yīng)該很熟悉了 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); //設(shè)置滑動條高度 mScollBarHeight = notScrollBarBg.getIntrinsicHeight(); //設(shè)置滑動塊直徑 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) {//這里滑塊中心坐標(biāo)初始化的時候測量一下(根據(jù)mDistance是否賦值判斷),并不需要不停地去測量。后面會根據(jù)進度計算滑塊位置。 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); //設(shè)置繪制樣式 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; //繪制是否可操作狀態(tài)的下的不同樣式,僅可編輯狀態(tài)下顯示進度條 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); //當(dāng)前滑塊刻度 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); } } //手勢監(jiān)聽 @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坐標(biāo) 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) ? (mOffsetLow) : (mDistance + mThumbWidth / 2); } } } else if (mFlag == CLICK_ON_HIGH) { if (e.getX() < mThumbWidth / 2) { mOffsetHigh = mThumbWidth / 2; mOffsetLow = mThumbWidth / 2; } else if (e.getX() > mScollBarWidth - mThumbWidth / 2) { mOffsetHigh = mThumbWidth / 2 + mDistance; } else { mOffsetHigh = formatInt(e.getX()); if (mOffsetHigh - mOffsetLow <= 0) { mOffsetLow = (mOffsetHigh >= mThumbWidth / 2) ? (mOffsetHigh) : mThumbWidth / 2; } } } //更新滑塊,每次滑塊有動作都要執(zhí)行此函數(shù)觸發(fā)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<progressLow+miniGap){ progressHigh=progressLow+miniGap; this.defaultScreenHigh = progressHigh; mOffsetHigh = formatInt(progressHigh / 100 * (mDistance)) + mThumbWidth / 2; invalidate(); } } return true; } /** * 設(shè)置是否可編輯狀態(tài),非可編輯狀態(tài)將不能對AB點進行操作 * @param _b */ public void setEditable(boolean _b){ editable=_b; invalidate(); } /** * 獲取當(dāng)前手指位置 */ public int getAreaFlag(MotionEvent e) { int top = mThumbMarginTop; int bottom = mThumbWidth + mThumbMarginTop; if (e.getY() >= 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; } } /** * 設(shè)置前滑塊位置 * @param progressLow */ public void setProgressLow(double progressLow) { this.defaultScreenLow = progressLow; mOffsetLow = formatInt(progressLow / 100 * (mDistance)) + mThumbWidth / 2; invalidate(); } /** * 設(shè)置后滑塊位置 * @param progressHigh */ public void setProgressHigh(double progressHigh) { this.defaultScreenHigh = progressHigh; mOffsetHigh = formatInt(progressHigh / 100 * (mDistance)) + mThumbWidth / 2; invalidate(); } /** * 設(shè)置滑動監(jiān)聽 * @param mListener */ public void setOnSeekBarChangeListener(OnSeekBarChangeListener mListener) { this.mBarChangeListener = mListener; } /** * 滑動監(jiān)聽,改變輸入框的值 */ public interface OnSeekBarChangeListener { //滑動時 public void onProgressChanged(SeekRangeBar seekBar, double progressLow, double progressHigh); } /** * 設(shè)置滑動結(jié)果為整數(shù) */ private int formatInt(double value) { BigDecimal bd = new BigDecimal(value); BigDecimal bd1 = bd.setScale(0, BigDecimal.ROUND_HALF_UP); return bd1.intValue(); } }
然后就可以在程序中使用了。
布局中
<util.SeekRangeBar android:id="@+id/doubleSeekbar" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_centerVertical="true"/>
調(diào)用
private SeekRangeBar doubleSeekbar;//雙向進度條 doubleSeekbar = (SeekRangeBar) findViewById(R.id.doubleSeekbar); //監(jiān)聽進度范圍變化 doubleSeekbar.setOnSeekBarChangeListener(new SeekRangeBar.OnSeekBarChangeListener() { @Override public void onProgressChanged(SeekRangeBar seekBar, double progressLow, double progressHigh) { Log.d("LOGCAT","低:" + progressLow + "高:" + progressHigh); } });
相關(guān)GitHub項目地址:https://github.com/codeqian/android-class-lib
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android編程自定義進度條顏色的方法詳解
- Android 自定義view實現(xiàn)進度條加載效果實例代碼
- Android自定義View仿華為圓形加載進度條
- Android進度條控件progressbar使用方法詳解
- Android實現(xiàn)文件解壓帶進度條功能
- Android實現(xiàn)蝸牛進度條效果
- android 中win10 使用uwp控件實現(xiàn)進度條Marquez效果
- Android自定義圓形進度條
- Android自定義View實現(xiàn)環(huán)形進度條的思路與實例
- android自定義進度條漸變色View的實例代碼
- Android編程實現(xiàn)對話框形式進度條功能示例
相關(guān)文章
解決webview 第二次調(diào)用loadUrl頁面不刷新的問題
這篇文章主要介紹了解決webview 第二次調(diào)用loadUrl頁面不刷新的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03Flutter源碼分析之自定義控件(RenderBox)指南
寫了兩天的flutter,發(fā)現(xiàn)控件樣式很多,flutter資源很少,本文在于實用性,可以減少頁面代碼,下面這篇文章主要介紹了Flutter源碼分析之自定義控件(RenderBox)的相關(guān)資料,需要的朋友可以參考下2021-08-08Android中使用ZXing生成二維碼(支持添加Logo圖案)
ZXing是谷歌的一個開源庫,可以用來生成二維碼、掃描二維碼。接下來通過本文給大家介紹Android中使用ZXing生成二維碼(支持添加Logo圖案),需要的朋友參考下2017-01-01Android程序打開和對輸入法的操作(打開/關(guān)閉)
整理了一下Android下對輸入法的操作:打開輸入法窗口、關(guān)閉出入法窗口、如果輸入法打開則關(guān)閉,如果沒打開則打開、獲取輸入法打開的狀態(tài)2013-05-05Android開發(fā)之Sqliteopenhelper用法實例分析
這篇文章主要介紹了Android開發(fā)之Sqliteopenhelper用法,實例分析了SQLiteOpenHelper類操作數(shù)據(jù)庫的相關(guān)技巧,需要的朋友可以參考下2015-05-05Android SDK Manager更新、下載速度慢問題解決辦法
這篇文章主要介紹了Android SDK Manager更新、下載速度慢問題解決辦法的相關(guān)資料,需要的朋友可以參考下2017-05-05Android BroadcastReceiver廣播機制概述
這篇文章主要為大家詳細(xì)介紹了Android BroadcastReceiver廣播機制,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-08-08Android熱更新開源項目Tinker集成實踐總結(jié)
最近項目集成了Tinker,開始認(rèn)為集成會比較簡單,但是在實際操作的過程中還是遇到了一些問題,本文就會介紹在集成過程大家基本會遇到的主要問題。下面跟著小編一起來看下吧2017-01-01