溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何在Android中實現(xiàn)一個滑塊拼圖驗證碼功能

發(fā)布時間:2021-02-23 16:28:15 來源:億速云 閱讀:203 作者:Leah 欄目:開發(fā)技術(shù)

本篇文章給大家分享的是有關(guān)如何在Android中實現(xiàn)一個滑塊拼圖驗證碼功能,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

一、實現(xiàn)步驟:

1、定義自定義屬性; 2、確認目標位置,這里使用的是陰影圖片來遮蓋背景圖片; 3、創(chuàng)建與目標位置相結(jié)合的滑塊圖片; 4、設置目標陰影圖片和滑塊圖片可以隨機旋轉(zhuǎn),并保持一致; 5、創(chuàng)建拖拽條,使滑塊隨著拖拽條的拖拽而移動; 6、判斷是否驗證成功。

二、實現(xiàn)流程:

1、定義自定義屬性 創(chuàng)建一個attr文件來定義一些自定義屬性

   <declare-styleable name="ImageAuthenticationView">
    <!--滑塊的高度-->
    <attr name="unitHeight" format="dimension" />
    <!--滑塊的寬度-->
    <attr name="unitWidth" format="dimension" />
    <!--滑塊占圖片高度的比例-->
    <attr name="unitHeightScale" format="integer" />
    <!--滑塊占圖片寬度的比例-->
    <attr name="unitWidthScale" format="integer" />
    <!--滑塊邊框的圖片資源-->
    <attr name="unitShadeSrc" format="reference" />
    <!--陰影部分的圖片資源-->
    <attr name="unitShowSrc" format="reference" />
    <!--是否需要旋轉(zhuǎn)-->
    <attr name="needRotate" format="boolean" />
    <!--驗證時的誤差值-->
    <attr name="deviate" format="integer" />
  </declare-styleable>

2、確認目標位置,這里使用的是陰影圖片來遮蓋背景圖片

 /**
   * 創(chuàng)建目標圖片(陰影部分)
   */
  private Bitmap drawTargetBitmap() {
    // 繪制圖片
    Bitmap showB;
    if (null != mShowBp) {
      showB = handleBitmap(mShowBp, mUintWidth, mUintHeight);
    } else {
      showB = handleBitmap(BitmapFactory.decodeResource(getResources(), R.mipmap.puzzle_show), mUintWidth, mUintHeight);
    }
    // 如果需要旋轉(zhuǎn)圖片,進行旋轉(zhuǎn),旋轉(zhuǎn)后為了保持和滑塊大小一致,需要重新縮放比例
    if (needRotate) {
      showB = handleBitmap(rotateBitmap(rotate, showB), mUintWidth, mUintHeight);
    }
    return showB;
  }

3、創(chuàng)建與目標位置相結(jié)合的滑塊圖片

 /**
   * 創(chuàng)建結(jié)滑塊圖片
   *
   * @param bp
   */
  private Bitmap drawResultBitmap(Bitmap bp) {
    // 繪制圖片
    Bitmap shadeB;
    if (null != mShadeBp) {
      shadeB = handleBitmap(mShadeBp, mUintWidth, mUintHeight);
    } else {
      shadeB = handleBitmap(BitmapFactory.decodeResource(getResources(), R.mipmap.puzzle_shade), mUintWidth, mUintHeight);
    }
    // 如果需要旋轉(zhuǎn)圖片,進行旋轉(zhuǎn),旋轉(zhuǎn)后為了和畫布大小保持一致,避免出現(xiàn)圖像顯示不全,需要重新縮放比例
    if (needRotate) {
      shadeB = handleBitmap(rotateBitmap(rotate, shadeB), mUintWidth, mUintHeight);
    }
    Bitmap resultBmp = Bitmap.createBitmap(mUintWidth, mUintHeight,
        Bitmap.Config.ARGB_8888);
    Paint paint = new Paint();
    paint.setAntiAlias(true);
    Canvas canvas = new Canvas(resultBmp);
    canvas.drawBitmap(shadeB, new Rect(0, 0, mUintWidth, mUintHeight),
        new Rect(0, 0, mUintWidth, mUintHeight), paint);
    // 選擇交集去上層圖片
    paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.MULTIPLY));
    canvas.drawBitmap(bp, new Rect(0, 0, mUintWidth, mUintHeight),
        new Rect(0, 0, mUintWidth, mUintHeight), paint);
    return resultBmp;
  }

4、設置目標陰影圖片和滑塊圖片可以隨機旋轉(zhuǎn),并保持一致

 /**
   * 旋轉(zhuǎn)圖片
   *
   * @param degree
   * @param bitmap
   * @return
   */
  public Bitmap rotateBitmap(int degree, Bitmap bitmap) {
    Matrix matrix = new Matrix();
    matrix.postRotate(degree);
    Bitmap bm = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(),
        bitmap.getHeight(), matrix, true);
    return bm;
  }

5、創(chuàng)建拖拽條,使滑塊隨著拖拽條的拖拽而移動

  //滑塊監(jiān)聽
    mSeekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
      @Override
      public void onProgressChanged(SeekBar seekBar, int i, boolean b) {
        //設置滑塊移動距離
        mDY.setUnitMoveDistance(mDY.getAverageDistance(seekBar.getMax()) * i);
      }

      @Override
      public void onStartTrackingTouch(SeekBar seekBar) {

      }

      @Override
      public void onStopTrackingTouch(SeekBar seekBar) {
        //驗證是否拼接成功
        mDY.testPuzzle();
      }
    });

6、判斷是否驗證成功

/**
   * 驗證是否拼接成功
   */
  public void testPuzzle() {
    if (Math.abs(mUnitMoveDistance - mUnitRandomX) <= DEFAULT_DEVIATE) {
      if (null != mlistener) {
        mlistener.onSuccess();
      }
    } else {
      if (null != mlistener) {
        mlistener.onFail();
      }
    }
  }

三、完整代碼

1、自定義控件內(nèi)容太多這里就不放出來了,完整Demo源碼會放在文章后面;

2、代碼邏輯

public class MainActivity extends Activity {

  //滑塊
  private SeekBar mSeekBar;
  //自定義的控件
  private ImageAuthenticationView mDY;
  private Button btn;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    initView();
    initListener();
  }

  private void initView() {
    mDY = findViewById(R.id.dy_v);
    mSeekBar = findViewById(R.id.sb_dy);
    btn = findViewById(R.id.btn);
  }

  private void initListener() {
    //滑塊監(jiān)聽
    mSeekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
      @Override
      public void onProgressChanged(SeekBar seekBar, int i, boolean b) {
        //設置滑塊移動距離
        mDY.setUnitMoveDistance(mDY.getAverageDistance(seekBar.getMax()) * i);
      }

      @Override
      public void onStartTrackingTouch(SeekBar seekBar) {

      }

      @Override
      public void onStopTrackingTouch(SeekBar seekBar) {
        //驗證是否拼接成功
        mDY.testPuzzle();
      }
    });

    //控件監(jiān)聽
    mDY.setPuzzleListener(new ImageAuthenticationView.onPuzzleListener() {
      @Override
      public void onSuccess() {
        //mSeekBar.setEnabled(false);//禁止滑動
        Toast.makeText(MainActivity.this, "驗證成功", Toast.LENGTH_SHORT).show();
      }

      @Override
      public void onFail() {
        Toast.makeText(MainActivity.this, "驗證失敗", Toast.LENGTH_SHORT).show();
        mSeekBar.setProgress(0);
      }
    });

    //重置
    btn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        //mSeekBar.setEnabled(true);
        mSeekBar.setProgress(0);
        mDY.reSet();
      }
    });
  }
}

3、布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:dy="http://schemas.android.com/apk/res-auto"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:gravity="center"
  android:orientation="vertical"
  android:paddingLeft="10dp"
  android:paddingTop="10dp"
  android:paddingRight="10dp"
  android:paddingBottom="10dp"
  tools:context=".MainActivity">

  <com.sjl.keeplive.slideImg.ImageAuthenticationView
    android:id="@+id/dy_v"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:scaleType="centerCrop"
    android:layout_marginBottom="10dp"
    android:src="@mipmap/test"
    dy:needRotate="true"
    dy:unitHeight="60dp"
    dy:unitShadeSrc="@mipmap/puzzle_shade"
    dy:unitShowSrc="@mipmap/puzzle_show"
    dy:unitWidth="80dp" />

  <SeekBar
    android:id="@+id/sb_dy"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="@drawable/bg_seekbar"
    android:max="100" />

  <Button
    android:id="@+id/btn"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="重置"/>
</LinearLayout>

以上就是如何在Android中實現(xiàn)一個滑塊拼圖驗證碼功能,小編相信有部分知識點可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學到更多知識。更多詳情敬請關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI