Android自定义SeekBar实现滑动验证且不可点击
作者:小猪阿正 发布时间:2023-11-08 04:35:45
最近公司因为短信接口被盗刷的比较严重,需要做一个类似于淘宝的滑动验证,用于特定环境,以增加一层保障。拿到需求首先想到的是自定义ViewGroup来实现,里面放一个seekbar和TextView即可。但是有更简单的方法,直接在布局中放入seekbar和TextView,不就ok了?用最简单快捷的方法实现需求,才是硬道理。
值得一提的是,seekbar默认情况下是支持点击事件的,也就是说,用户可以直接点击进度条以实现滑动验证这是不允许的,因此,自定义seekbar,屏蔽点击事件。下面我们先从seekbar + textxiew实现滑动验证效果开始,最后实现seekbar点击事件的屏蔽。
滑动验证实现:
先上一张效果图:
不太美观,UI还没设计,只是个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的进度条-->
<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,监听seekbar。
简单介绍下几个回调方法的作用:
onProgressChanged :当progress进度改变时调用;
onStartTrackingTouch :开始滑动时调用;
onStopTrackingTouch : 滑动结束时调用;
public class NoClickSeekbar extends SeekBar{
private int oldsign = 0;
private int mTemp = 10;//点击最大值,超过这个值则不响应
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的时候,设置供用户的回调监听,
public interface OnNoClickSeekBarChangeListener {
void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser);
void onStartTrackingTouch(SeekBar seekBar);
void onStopTrackingTouch(SeekBar seekBar);
}
并在seekbar中重写监听时,重写对应的事件回调时,将上面对应的接口方法对应的执行。用户在使用自定义seekbar时,执行监听,加入我们需要实现的需求。
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("验证通过");
} 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点击事件的屏蔽
1、解决办法一:
在我们滑动seekbar的时候,是可以监听到progress的。因此,我们用一个变量记录上一次的progress,当点击事件发生时,计算点击的进度与之前的进度是否超过一定范围,从而判断是否需要响应。比较简单,直接上代码:
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的事件监听,重写view的onTouchEvent事件,在MotionEvent.ACTION_DOWN的时候,同样判断前后两次事件之间的距离,判断是否要处理该点击事件。
@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来说,在屏蔽点击事件上,还是有瑕疵的。是能设定一定的范围,小于了该范围,比如用户小范围的点击,是会响应的。把问题都在这儿,后面解决了再补充!
来源:https://blog.csdn.net/Mr_azheng/article/details/81745491


猜你喜欢
- 在Android Studio 2.1 Preview 3之后,官方开始支持双向绑定了。可惜目前Google并没有在Data Binding
- 问题描述:解决:检查以上是否版本一致。不行就再检查下面的:如果上面的方法还是不行的话,就建议你改一下pom文件中的maven插件编译级别,可
- 今天没事跟群里面侃大山,有个哥们说道Android Wheel这个控件,以为是Andriod内置的控件,google一把,发现是个githu
- 学会了Paint,Canvas的基本用法之后,我们就可以动手开始实践了,先写个简单的图片加载进度条看看。按照惯例,先看效果图,再决定要不要往
- TCP/IP通信协议是一种可靠的网络协议,它在通信的两端各建立一个Socket,从而在通信的两端之间形成网络虚拟链路。一旦建立了虚拟的网络链
- 本文实例分析了C# SQlite操作方法。分享给大家供大家参考,具体如下:最近项目需求用C#保存一些数据,如此先总结一下。需要下载Sqlit
- 上文对数据结构与算法,有了一个简单的概述与介绍,这篇文章,我们介绍一中典型数据结构——线性结构。什么是线性结构,线性结构是最简单、最基本、最
- 今天在做一个联系人管理的C#设计时,遇到了这个问题,我需要将父窗体中的textBox中的值传到子窗体并进行数据库查询操作,我用了new 父窗
- 本文实例讲述了C#计算程序执行过程花费时间的方法。分享给大家供大家参考。具体如下:计算执行完程序花费的时间:void AddInfo(){
- 井字棋游戏要求在3乘3棋盘上,每行都相同或者每列都相同再或者对角线相同,则胜出.因此我们可以使用一个二维数组来表示棋盘,判断胜负只需要判断数
- 我们先假设一个场景想象一下,当一个项目出现bug的时候,恰巧这个时候需要你去修改,而当你打开项目之后,眼前的代码让你有一种特别严重的陌生感,
- 首先给大家来讲一个我们遇到的一个奇怪的问题:1.我的一个springboot项目,用mvn install打包成jar,换一台有jdk的机器
- SpringBoot默认加载的是application.yml文件,所以想要引入其他配置的yml文件,就要在application.yml中
- 一、前言在日常工作中,如果涉及到与第三方进行接口对接,有的会使用WebService的方式,这篇文章主要讲解在.NET Framework中
- 文中涉及到文件名称排序,固定根目录以及返回上一层在上面,方便选择等。根据文件后缀,筛选文件还没做。先看效果。1、效果图2、核心代码如下是通过
- 实现方案:我们直接参考实例代码:private String pattern = "((http|ftp
- 本文实例讲述了Android游戏开发学习①弹跳小球实现方法。分享给大家供大家参考。具体如下:在学习了一点点Android之后,觉得有必要记录
- 介绍try-with-resources是Java中的环绕语句之一,旨在减轻开发人员释放try块中使用的资源的义务。它最初在Java 7中引
- 一、什么是代理?指为一个目标对象提供一个代理对象, 并由代理对象控制对目标对象的引用. 使用代理对象, 是为了在不修改目标对象的基础上,增强
- 先看看下面的代码能不能编译通过:public static void main(String[] args) {List l1 = new