Android6.0指纹识别开发案例_Android

Android M指纹的资料太少,经过一段时间阅读原生Android代码,写了以下例子,贡献出来给需要帮助的人。
以下内容基于64位的高通CPU,搭载fpc1020芯片,此部分代码在原生android上做了更改,以应付工厂指纹的测试。原生android指纹录入需要采集10次(因不同的芯片而定)。

代码简单说明:

1. FingerPrintEnrollBase类:
重要的是

public interface Listener {
  void onEnrollmentHelp(CharSequence helpString);
  void onEnrollmentError(int errMsgId, CharSequence errString);
  void onEnrollmentProgressChange(int steps, int remaining);
}

这个接口。
  1)onEnrollmentHelp函数:是下层返回的指纹操作帮助信息。
  2)onEnrollmentError函数是下层返回的错误信息,errMsgId是错误信息类型,其中1表示指纹硬件不可用,3表示超时,5表示操作已取消。errString是errMsgId对应的文字信息,有需要的自己实验就知道了。
  3)onEnrollmentProgressChange()函数是显示当前指纹的采集情况,steps:总共需要录入多少次;remaining是剩余多少次,我在此函数中有打印Log。

2. FingerPrintActivity是主Activity类,FingerPrintEnrollBase是指纹Base类。
需要特别说明的是,在FingerPrintEnrollBase类中有一句:private byte[] mToken = new byte[69];这个数组正常情况是通过输入密码后经由framework层向hal层申请的,我这里为了省去验证那一块的代码,通过sizeof(hw_auth_token_t)计算出来的为69,需要的请依此计算。我在代码里实验过,此数组如果不是69(在64位CPU上),将直接导致程序crach!
fpc指纹的采集流程图:

/*
 * Copyright (C) 2015 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *  http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License
 */ 

package com.mediatek.test; 

import android.app.Activity;
import android.app.Fragment;
import android.content.Context;
import android.hardware.fingerprint.FingerprintManager;
import android.os.Bundle;
import android.os.CancellationSignal;
import android.os.Handler;
import android.util.Log;
/**
 * local fragment to handle the state around fingerprint enrollment.
 */
public class FingerPrintEnrollBase extends Fragment { 

 private int mEnrollmentSteps = -1;
 private int mEnrollmentRemaining = 0;
 private Listener mListener;
 private boolean mEnrolling;
 private CancellationSignal mEnrollmentCancel;
 private Handler mHandler = new Handler();
 private byte[] mToken = new byte[69];//if byte length != sizeof(hw_auth_token_t), throws Exception
 private boolean mDone;
 private static final String TAG = "FingerPrintEnrollBase"; 

 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setRetainInstance(true);
 } 

 @Override
 public void onAttach(Activity activity) {
  super.onAttach(activity);
 } 

 @Override
 public void onStart() {
  super.onStart();
  Log.d(TAG, "mToken length="+mToken.length);
  if (!mEnrolling) {
   FingerprintManager mFpM = (FingerprintManager) getActivity().getSystemService(Context.FINGERPRINT_SERVICE);
   startEnrollment(mFpM);
  }
 } 

 @Override
 public void onStop() {
  super.onStop();
  if (!getActivity().isChangingConfigurations()) {
   cancelEnrollment();
  }
 } 

 protected void startEnrollment(FingerprintManager mFpM) {
  mHandler.removeCallbacks(mTimeoutRunnable);
  mEnrollmentSteps = -1;
  mEnrollmentCancel = new CancellationSignal();
  if(mFpM != null){
   mFpM.enroll(mToken, mEnrollmentCancel, 0, mEnrollmentCallback);
  }
  mEnrolling = true;
 } 

 protected void cancelEnrollment() {
  mHandler.removeCallbacks(mTimeoutRunnable);
  if (mEnrolling) {
   mEnrollmentCancel.cancel();
   mEnrolling = false;
   mEnrollmentSteps = -1;
  }
 } 

 public void setListener(Listener listener) {
  mListener = listener;
 } 

 public int getEnrollmentSteps() {
  return mEnrollmentSteps;
 } 

 public int getEnrollmentRemaining() {
  return mEnrollmentRemaining;
 } 

 public boolean isDone() {
  return mDone;
 } 

 private FingerprintManager.EnrollmentCallback mEnrollmentCallback
   = new FingerprintManager.EnrollmentCallback() { 

  @Override
  public void onEnrollmentProgress(int remaining) {
   if (mEnrollmentSteps == -1) {
    mEnrollmentSteps = remaining;
   }
   mEnrollmentRemaining = remaining;
   mDone = remaining == 0;
   if (mListener != null) {
    mListener.onEnrollmentProgressChange(mEnrollmentSteps, remaining);
   }
  } 

  @Override
  public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
   if (mListener != null) {
    mListener.onEnrollmentHelp(helpString);
   }
  } 

  @Override
  public void onEnrollmentError(int errMsgId, CharSequence errString) {
   if (mListener != null) {
    mListener.onEnrollmentError(errMsgId, errString);
   }
  }
 }; 

 private final Runnable mTimeoutRunnable = new Runnable() {
  @Override
  public void run() {
   cancelEnrollment();
  }
 }; 

 public interface Listener {
  void onEnrollmentHelp(CharSequence helpString);
  void onEnrollmentError(int errMsgId, CharSequence errString);
  void onEnrollmentProgressChange(int steps, int remaining);
 }
} 

 FingerPrintActivity:

package com.mediatek.test; 

import java.io.FileInputStream;
import java.io.IOException;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask; 

import android.R.color;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.graphics.Color;
import android.os.Bundle;
import android.os.CancellationSignal;
import android.os.Handler;
import android.os.Message;
import android.text.Spannable;
import android.text.style.ForegroundColorSpan;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ProgressBar;
import android.widget.TableRow;
import android.widget.TextView;
import android.widget.Toast; 

import com.mediatek.test.R;
import android.hardware.fingerprint.Fingerprint;
import android.hardware.fingerprint.FingerprintManager; 

public class FingerPrintActivity extends Activity implements FingerPrintEnrollBase.Listener{
 private static final String TAG = "FingerPrintActivity";
 private MyHandler mHandler = new MyHandler();
 private TextView fingerPrintAcquisitionTextView, fingerPrintEnrollmentErrorView, fingerPrintEnrollmentHelpView;
 private FingerPrintEnrollBase fpeb;
 private FingerprintManager mFingerprintManager; 

 @Override
 protected int setTitleId() {
  return R.string.tittle_fingerprint;
 } 

 @Override
 protected Activity getActivity() {
  return this;
 } 

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.fingerprint);
  fingerPrintAcquisitionTextView = (TextView)findViewById(R.id.fingerprint_message);
  fingerPrintEnrollmentErrorView= (TextView)findViewById(R.id.fingerprint_enrollment_error);
  fingerPrintEnrollmentHelpView= (TextView)findViewById(R.id.fingerprint_enrollment_help); 

  mFingerprintManager = (FingerprintManager) getActivity().getSystemService(
    Context.FINGERPRINT_SERVICE); 

 } 

 private void sendMsgToHandler(int msgType, String remark){
  Message msg = Message.obtain();
  msg.what = msgType;
  Bundle bundle = new Bundle();
  bundle.putString("result", remark);
  msg.setData(bundle);
  mHandler.sendMessage(msg);
 } 

 @Override
 public void onEnrollmentHelp(CharSequence helpString) {
  Log.d(TAG, "==onEnrollmentHelp="+helpString.toString());
  sendMsgToHandler(3, helpString.toString());
 } 

 @Override
 public void onEnrollmentError(int errMsgId, CharSequence errString) {
  Log.d(TAG, "==onEnrollmentError errMsgId="+errMsgId+" errString="+errString.toString());
  if(errMsgId == 3){//time out
   postEnrollment();
  }else if(errMsgId == 1){//hardware lose efficacy
   startEnrollment();
  }else if(errMsgId == 5){//operation cancel
  }else{
   sendMsgToHandler(4, errString.toString());
  }
 } 

 @Override
 public void onEnrollmentProgressChange(int steps, int remaining) {
  Log.d(TAG, "===============onEnrollmentProgressChange=======");
  Log.d(TAG, "steps" + steps + "||remaining=" + remaining);
  fingerPrintEnrollmentHelpView.setText("");
  fingerPrintEnrollmentErrorView.setText("");
  sendMsgToHandler(2, getString(R.string.fingerprint_validate_success));
  buttonS.setEnabled(true);
  stopFingerprint();
 } 

 private void startEnrollment(){
  if (fpeb == null) {
   fpeb = new FingerPrintEnrollBase();
  }
  fpeb.setListener(this);
  fpeb.startEnrollment(mFingerprintManager);
 }
 private void postEnrollment(){
  if(mFingerprintManager != null){
   mFingerprintManager.postEnroll();
  }
 } 

 private void stopFingerprint() {
  if (fpeb != null){
   fpeb.cancelEnrollment();
   onStop();
  }
 } 

 @Override
 protected void onStart() {
  super.onStart();
 } 

 @Override
 protected void onStop() {
  super.onStop();
  if(fpeb != null){
   fpeb.setListener(null);
  }
 } 

 @Override
 protected void onResume() {
  super.onResume();
  if(mFingerprintManager == null){
   mFingerprintManager = (FingerprintManager) getActivity().getSystemService(
    Context.FINGERPRINT_SERVICE);
  } 

  startEnrollment();
 } 

 @Override
 protected void onPause() {
  super.onPause();
  postEnrollment();
  mFingerprintManager = null;
  stopFingerprint();
  fpeb = null;
 } 

 class MyHandler extends Handler{ 

  @Override
  public void handleMessage(Message msg) {
   Bundle bundle = null;
   super.handleMessage(msg);
   switch (msg.what) {
   case 0:
    removeMessages(0);
    bundle = msg.getData();
    String result = bundle.getString("result");
    selfTestView.setText(result);
    selfTestView.setTextColor(result.contains("PASS")?Color.GREEN:Color.RED);
    break;
   case 1:
    removeMessages(1);
    bundle = msg.getData();
    String rs = bundle.getString("result");
    checkBroadView.setText(bundle.getString("result"));
    checkBroadView.setTextColor(rs.contains("PASS")?Color.GREEN:Color.RED);
    break;
   case 2:
    removeMessages(2);
    bundle = msg.getData();
    String fingerprint = bundle.getString("result");
    fingerPrintAcquisitionTextView.setText(bundle.getString("result"));
    fingerPrintAcquisitionTextView.setTextColor(fingerprint.contains("PASS")?Color.GREEN:Color.WHITE);
    mButtonSelftest.setEnabled(true);
    mButtonCheckerboardTest.setEnabled(true);
    break;
   case 3:
    removeMessages(3);
    bundle = msg.getData();
    String enrollmentHelp = bundle.getString("result");
    fingerPrintEnrollmentHelpView.setText(bundle.getString("result"));
    break;
   case 4:
    removeMessages(4);
    bundle = msg.getData();
    String enrollmentError = bundle.getString("result");
    fingerPrintEnrollmentErrorView.setText(bundle.getString("result"));
    fingerPrintAcquisitionTextView.setText("");
    break;
   default:
    break;
   }
  }
 } 

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
指纹识别
android 6.0指纹识别、安卓6.0指纹识别、s5 6.0 指纹识别、android 指纹识别、android 指纹识别demo,以便于您获取更多的相关知识。

时间: 2024-09-26 10:24:01

Android6.0指纹识别开发案例_Android的相关文章

Android6.0指纹识别开发实例详解

Android6.0指纹识别开发实例详解 最近在做android指纹相关的功能,谷歌在android6.0及以上版本对指纹识别进行了官方支持.当时在FingerprintManager和FingerprintManagerCompat这两个之间纠结,其中使用FingerprintManager要引入com.android.support:appcompat-v7包,考虑到包的大小,决定使用v4兼容包FingerprintManagerCompat来实现. 主要实现的工具类FingerprintU

Android 指纹识别开发实例_Android

Android M指纹的资料太少,经过一段时间阅读原生Android代码,写了以下例子,贡献出来给需要帮助的人. 以下内容基于64位的高通CPU,搭载fpc1020芯片,此部分代码在原生android上做了更改,以应付工厂指纹的测试.原生android指纹录入需要采集10次(因不同的芯片而定). 代码简单说明: 1. FingerPrintEnrollBase类: 重要的是 Java代码 public interface Listener { void onEnrollmentHelp(Char

Android 6.0指纹识别App开发案例_Android

在android 6.0中google终于给android系统加上了指纹识别的支持,这个功能在iPhone上早就已经实现了,并且在很多厂商的定制的ROM中也都自己内部实现这个功能了,这个功能来的有点晚啊.在google全新发布的nexus设备:nexus 5x和nexus 6p中都携带了一颗指纹识别芯片在设备的背面,如下图(图片来自网络): 笔者手中的设备就是图上的那台黑色的nexus 5x,话说这台机器很是好看呢!手感超棒! 废话不多说,下面我出一个指纹识别的demo app,并且详细说明怎么

Android指纹识别API初试_Android

在android6.0之后谷歌对指纹识别进行了官方支持,今天还在放假,所以就随意尝试了一下这个api,但是遇到了各种各样的问题  ①在使用FingerPrintManager这个类实现的时候发现了很多问题,这个类里面的一些函数是被hide了的,也就是我们不能调用,比如enroll(),也就是说,当前的官方支持其实是有限的,我们能读取到本机已经存在的指纹(用于解锁的),然后验证这些指纹,但是不能让用户在app使用的时候录入一个指纹,用于app的其他功能,这个是一个缺陷吧目前来说,下面的图也是展示了

详解Android6.0运行时权限管理_Android

自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验,不会再强迫用户因拒绝不该拥有的权限而导致的无法安装的事情,也不会再不征求用户授权的情况下,就可以任意的访问用户隐私,而且即使在授权之后也可以及时的更改权限.这就是6.0版本做出的更拥护和注重用户的一大体现. 一.认知 今天我们就来学习下Android6.0的权限管理. Android6.0系统把权限分为两个级别: 一个是Normal Permiss

Android6.0仿微信权限设置_Android

Android 6.0版本对于程序员兄弟来说最不友好的就是权限的问题,动态权限的设置曾经让我很苦恼,目前大部分关于6.0权限设置的框架基本都是一次性访问多个权限(EasyPermissions),这样导致的问题就是如果我们申请了三种权限,而用户只同意了其中一种,下次再申请权限又是一次性申请三种,很不方便对于用户来说很不友好,偶然情况下发现了安卓猴的这篇文章, http://sunjiajia.com/2016/04/19/android-m-permissions/ 在此基础上做了修改,就实现了

Android6.0 消息机制原理解析_Android

消息都是存放在一个消息队列中去,而消息循环线程就是围绕这个消息队列进入一个无限循环的,直到线程退出.如果队列中有消息,消息循环线程就会把它取出来,并分发给相应的Handler进行处理:如果队列中没有消息,消息循环线程就会进入空闲等待状态,等待下一个消息的到来.在编写Android应用程序时,当程序执行的任务比较繁重时,为了不阻塞UI主线程而导致ANR的发生,我们通常的做法的创建一个子线程来完成特定的任务.在创建子线程时,有两种选择,一种通过创建Thread对象来创建一个无消息循环的子线程:还有一

Android手机闹钟服务AlarmManagerk开发案例_Android

AlarmManager通常用来开发手机闹钟,并且它是一个全局定时器,可在指定时间或指定周期启动其他组件(包括Activity,Service,BroadcastReceiver) 获取AlarmManager对象: getSystemService(Service.ALARM_SERVICE) 调用其中的方式设置定时器启动指定组件: set(int type,long triggerAtTime,PendingIntent operation)设置在TriggerAtTime时间启动由oper

Android 指纹识别详解及实现方法_Android

最近项目需要使用到指纹识别的功能,查阅了相关资料后,整理成此文. 指纹识别是在Android 6.0之后新增的功能,因此在使用的时候需要先判断用户手机的系统版本是否支持指纹识别.另外,实际开发场景中,使用指纹的主要场景有两种: 纯本地使用.即用户在本地完成指纹识别后,不需要将指纹的相关信息给后台. 与后台交互.用户在本地完成指纹识别后,需要将指纹相关的信息传给后台. 由于使用指纹识别功能需要一个加密对象(CryptoObject)该对象一般是由对称加密或者非对称加密获得.上述两种开发场景的实现大