Android自定義SeekBar實(shí)現(xiàn)滑動驗(yàn)證且不可點(diǎn)擊
最近公司因?yàn)槎绦沤涌诒槐I刷的比較嚴(yán)重,需要做一個類似于淘寶的滑動驗(yàn)證,用于特定環(huán)境,以增加一層保障。拿到需求首先想到的是自定義ViewGroup來實(shí)現(xiàn),里面放一個seekbar和TextView即可。但是有更簡單的方法,直接在布局中放入seekbar和TextView,不就ok了?用最簡單快捷的方法實(shí)現(xiàn)需求,才是硬道理。
值得一提的是,seekbar默認(rèn)情況下是支持點(diǎn)擊事件的,也就是說,用戶可以直接點(diǎn)擊進(jìn)度條以實(shí)現(xiàn)滑動驗(yàn)證這是不允許的,因此,自定義seekbar,屏蔽點(diǎn)擊事件。下面我們先從seekbar + textxiew實(shí)現(xiàn)滑動驗(yàn)證效果開始,最后實(shí)現(xiàn)seekbar點(diǎn)擊事件的屏蔽。
滑動驗(yàn)證實(shí)現(xiàn):
先上一張效果圖:

不太美觀,UI還沒設(shè)計,只是個demo。
1、布局
<RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@color/white" android:padding="10dp"> <com.dmlc.app.android.widget.NoClickSeekbar android:id="@+id/sb_bar" android:layout_width="match_parent" android:layout_height="wrap_content" android:max="100" android:progress="0" android:progressDrawable="@drawable/style_seekbar_verify" android:thumb="@drawable/style_seekbar_thumb" android:thumbOffset="0dp" /> <TextView android:id="@+id/sb_tv" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_centerInParent="true" android:gravity="center" android:text="請按住滑塊,拖動到最右邊" android:textColor="#888888" android:textSize="14dp" /> </RelativeLayout>
其中,android:progressDrawable用于定義滑動條背景,android:thumb定義滑塊樣式。
滑動條背景:
<?xml version="1.0" encoding="utf-8"?> <layer-list xmlns:android="http://schemas.android.com/apk/res/android"> <!--seekBar背景--> <item android:id="@android:id/background"> <!--形狀--> <shape android:shape="rectangle"> <!--大小--> <size android:height="30dp" /> <!--圓角--> <corners android:radius="5dp" /> <!--背景--> <solid android:color="#E7EAE9" /> <!--邊框--> <stroke android:width="1dp" android:color="#C3C5C4" /> </shape> </item> <!--seekBar的進(jìn)度條--> <item android:id="@android:id/progress"> <clip> <shape> <corners android:radius="5dp" /> <solid android:color="#7AC23C" /> <stroke android:width="1dp" android:color="#C3C5C4" /> </shape> </clip> </item> </layer-list>
滑塊樣式:
<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"> <item android:state_focused="true" android:state_pressed="false" android:drawable="@drawable/seekbar_thumb_normal" /> <item android:state_focused="true" android:state_pressed="true" android:drawable="@drawable/seekbar_thumb_pressed" /> <item android:state_focused="false" android:state_pressed="true" android:drawable="@drawable/seekbar_thumb_pressed" /> <item android:drawable="@drawable/seekbar_thumb_normal" /> </selector>
2、自定義seekbar
重寫setOnSeekBarChangeListener,監(jiān)聽seekbar。
簡單介紹下幾個回調(diào)方法的作用:
- onProgressChanged :當(dāng)progress進(jìn)度改變時調(diào)用;
- onStartTrackingTouch :開始滑動時調(diào)用;
- onStopTrackingTouch : 滑動結(jié)束時調(diào)用;
public class NoClickSeekbar extends SeekBar{
private int oldsign = 0;
private int mTemp = 10;//點(diǎn)擊最大值,超過這個值則不響應(yīng)
private int mStep = 0;
OnNoClickSeekBarChangeListener mOnSeekBarChangeListener;
public NoClickSeekbar(Context context) {
this(context,null);
}
public NoClickSeekbar(Context context, AttributeSet attrs) {
this(context, attrs,0);
}
public NoClickSeekbar(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
private void init() {
setOnSeekBarChangeListener(new OnSeekBarChangeListener(){
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
// TODO 自動生成的方法存根
if(Math.abs(progress - oldsign) > mTemp){
seekBar.setProgress(oldsign);
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onProgressChanged(seekBar,oldsign,fromUser);
}
return;
}
seekBar.setProgress(progress);
oldsign = progress;
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onProgressChanged(seekBar,oldsign,fromUser);
}
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
// TODO 自動生成的方法存根
seekBar.setProgress(oldsign);
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onStartTrackingTouch(seekBar);
}
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
// TODO 自動生成的方法存根
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onStopTrackingTouch(seekBar);
}
}
});
}
public interface OnNoClickSeekBarChangeListener {
void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser);
void onStartTrackingTouch(SeekBar seekBar);
void onStopTrackingTouch(SeekBar seekBar);
}
public void setNoClickSeekBarChangeListener(OnNoClickSeekBarChangeListener l) {
mOnSeekBarChangeListener = l;
}
}
在自定義seekbar的時候,設(shè)置供用戶的回調(diào)監(jiān)聽,
public interface OnNoClickSeekBarChangeListener {
void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser);
void onStartTrackingTouch(SeekBar seekBar);
void onStopTrackingTouch(SeekBar seekBar);
}
并在seekbar中重寫監(jiān)聽時,重寫對應(yīng)的事件回調(diào)時,將上面對應(yīng)的接口方法對應(yīng)的執(zhí)行。用戶在使用自定義seekbar時,執(zhí)行監(jiān)聽,加入我們需要實(shí)現(xiàn)的需求。
mSeekBar.setNoClickSeekBarChangeListener(new NoClickSeekbar.OnNoClickSeekBarChangeListener() {
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
if (progress == seekBar.getMax()){
mSeekbarTV.setVisibility(View.VISIBLE);
mSeekbarTV.setText("驗(yàn)證通過");
} else {
mSeekbarTV.setVisibility(View.INVISIBLE);
if (progress < 10){
mSeekbarTV.setVisibility(View.VISIBLE);
mSeekbarTV.setText("請按住滑塊,拖動到最右邊");
}
}
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
}
});
SeekBar點(diǎn)擊事件的屏蔽
1、解決辦法一:
在我們滑動seekbar的時候,是可以監(jiān)聽到progress的。因此,我們用一個變量記錄上一次的progress,當(dāng)點(diǎn)擊事件發(fā)生時,計算點(diǎn)擊的進(jìn)度與之前的進(jìn)度是否超過一定范圍,從而判斷是否需要響應(yīng)。比較簡單,直接上代碼:
setOnSeekBarChangeListener(new OnSeekBarChangeListener(){
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
// TODO 自動生成的方法存根
if(Math.abs(progress - oldsign) > mTemp){
seekBar.setProgress(oldsign);
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onProgressChanged(seekBar,oldsign,fromUser);
}
return;
}
seekBar.setProgress(progress);
oldsign = progress;
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onProgressChanged(seekBar,oldsign,fromUser);
}
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
// TODO 自動生成的方法存根
seekBar.setProgress(oldsign);
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onStartTrackingTouch(seekBar);
}
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
// TODO 自動生成的方法存根
if (mOnSeekBarChangeListener != null) {
mOnSeekBarChangeListener.onStopTrackingTouch(seekBar);
}
}
});
2、解決辦法二:
通過view的事件監(jiān)聽,重寫view的onTouchEvent事件,在MotionEvent.ACTION_DOWN的時候,同樣判斷前后兩次事件之間的距離,判斷是否要處理該點(diǎn)擊事件。
@Override
public boolean onTouchEvent(MotionEvent event) {
int x = (int) event.getX();
switch (event.getAction()){
case MotionEvent.ACTION_DOWN:
if (Math.abs(x - mStep) > 100) {
return false;
}
break;
case MotionEvent.ACTION_MOVE:
break;
case MotionEvent.ACTION_UP:
mStep = x;
break;
}
return super.onTouchEvent(event);
}
對于上面自定義SeekBar來說,在屏蔽點(diǎn)擊事件上,還是有瑕疵的。是能設(shè)定一定的范圍,小于了該范圍,比如用戶小范圍的點(diǎn)擊,是會響應(yīng)的。把問題都在這兒,后面解決了再補(bǔ)充!
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android SeekBar控制視頻播放進(jìn)度實(shí)現(xiàn)過程講解
- Android?SeekBar充當(dāng)Progress實(shí)現(xiàn)兔兔進(jìn)度條Plus
- Android開發(fā)雙向滑動選擇器范圍SeekBar實(shí)現(xiàn)
- Android開發(fā)自定義雙向SeekBar拖動條控件
- Android開發(fā)手冊SeekBar拖動條使用實(shí)例
- Android通過SeekBar調(diào)節(jié)布局背景顏色
- Android SeekBar實(shí)現(xiàn)平滑滾動
- Android SeekBar在刷新使用中需要注意的問題
相關(guān)文章
如何給Flutter界面切換實(shí)現(xiàn)點(diǎn)特效
這篇文章主要給大家介紹了關(guān)于如何給Flutter界面切換實(shí)現(xiàn)點(diǎn)特效的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用Flutter具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
Android自定義PasswordInputView密碼輸入
這篇文章主要為大家詳細(xì)介紹了Android自定義PasswordInputView密碼輸入功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-08-08
Android WebView實(shí)現(xiàn)文件下載功能
這篇文章主要為大家詳細(xì)介紹了Android WebView實(shí)現(xiàn)文件下載功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-05-05
Android利用Intent啟動和關(guān)閉Activity
這篇文章主要為大家詳細(xì)介紹了Android利用Intent啟動和關(guān)閉Activity的相關(guān)操作,感興趣的小伙伴們可以參考一下2016-06-06
Flutter實(shí)現(xiàn)滾動選擇數(shù)字
這篇文章主要為大家詳細(xì)介紹了Flutter實(shí)現(xiàn)滾動選擇數(shù)字,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03
關(guān)于Touch Panel AA區(qū)要做外擴(kuò)的原因解析
今天小編就為大家分享一篇關(guān)于Touch Panel AA區(qū)要做外擴(kuò)的原因解析,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-12-12
Android報錯Didn‘t?find?class?“android.view.x“問題解決原理剖析
這篇文章主要為大家介紹了Android報錯Didn‘t?find?class?“android.view.x“問題解決及原理剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03

