Android開發之自動填充簡訊驗證碼

yungfan發表於2016-04-10

筆者發現在很多應用中,都有自動獲取驗證碼的功能:點選獲取驗證碼按鈕,收到簡訊,當前應用不需要退出程式就可以獲取到簡訊中的驗證碼,並自動填充。覺得這種使用者體驗很贊,無須使用者在簡訊和App之間來回切換,這個功能是如何實現的呢?其實很簡單,就是用到了Android中的一個叫ContentObserver的東西,下面手動簡單實現一個~~~

</br>

一、開發ContentObserver

主要是用來監聽收件箱的內容,一旦有新訊息過來,就去監聽是否是我想要的那個號碼發過來的簡訊,如果是,就直接用正規表示式擷取

/**
 * 簡訊驗證碼擷取
 * 
 * @author yangfan
 * 
 */
public class SMSContentObserver extends ContentObserver
{

    private Context mContext; // 上下文
    private Handler mHandler; // 更新UI執行緒
    private String code; // 驗證碼

    public SMSContentObserver(Context context, Handler handler)
    {
        super(handler);
        mContext = context;
        mHandler = handler;
    }

    /**
     * 回撥函式, 當所監聽的Uri發生改變時,就會回撥此方法
     * 
     * 注意當收到簡訊的時候會回撥兩次
     * 
     * @param selfChange
     *            此值意義不大 一般情況下該回撥值false
     */
    @Override
    public void onChange(boolean selfChange, Uri uri)
    {

        Log.e("XXXXXXXXXXXXXXXX", uri.toString());

        // 第一次回撥 不是我們想要的 直接返回
        if (uri.toString().equals("content://sms/raw"))
        {
            return;
        }

        // 第二次回撥 查詢收件箱裡的內容
        Uri inboxUri = Uri.parse("content://sms/inbox");

        // 按時間順序排序簡訊資料庫
        Cursor c = mContext.getContentResolver().query(inboxUri, null, null,
                null, "date desc");
        if (c != null)
        {
            if (c.moveToFirst())
            {

                // 獲取手機號
                String address = c.getString(c.getColumnIndex("address"));
                // 獲取簡訊內容
                String body = c.getString(c.getColumnIndex("body"));
                // 判斷手機號是否為目標號碼
                if (!address.equals("你的目標號碼"))
                {
                    return;
                }

                // 正規表示式擷取簡訊中的6位驗證碼
                Pattern pattern = Pattern.compile("(\d{6})");
                Matcher matcher = pattern.matcher(body);

                // 如果找到通過Handler傳送給主執行緒
                if (matcher.find())
                {
                    code = matcher.group(0);
                    mHandler.obtainMessage(1, code).sendToTarget();
                }
            }

        }
        c.close();

    }

}

二、佈局和Activity
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:padding="50dp" >

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="您的驗證碼為:" />

    <EditText
        android:id="@+id/smsCode"
        android:focusable="false"
        android:focusableInTouchMode="false"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>
public class MainActivity extends ActionBarActivity
{
    private EditText code;

    @SuppressLint("HandlerLeak")
    Handler handler = new Handler()
    {
        @Override
        public void handleMessage(Message msg)
        {
            if (msg.what == 1)
            {
                code.setText(msg.obj.toString());
            }
        }
    };

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

        code = (EditText) findViewById(R.id.smsCode);

        SMSContentObserver smsContentObserver = new SMSContentObserver(
                MainActivity.this, handler);

        MainActivity.this.getContentResolver().registerContentObserver(
                Uri.parse("content://sms/"), true, smsContentObserver);
    }
}

三、不要忘記加許可權
  <uses-permission android:name="android.permission.READ_SMS" />

四、執行測試

本人是拿兩部手機來測試的 ~~~

img_bdd9e23b91d8e84abb0e3c1f79debd0d.png
自動獲取簡訊驗證碼.png

程式碼地址:https://github.com/yungfan/SMSSMSContentObserver


相關文章