Android 6.0+ TelephonyManager 使用示例(2)監(jiān)聽手機(jī)來電

本文參考文獻(xiàn):《瘋狂Android講義 : 第2版
本文參考文章: 《Android 6.0 - 動(dòng)態(tài)權(quán)限管理的解決方案

檢測(cè)權(quán)限工具類

檢測(cè)權(quán)限工具類握截,我們可以將其在多個(gè)應(yīng)用中復(fù)用:

package com.toby.personal.testlistview;

import android.content.Context;
import android.content.pm.PackageManager;
import android.support.v4.content.ContextCompat;
import android.util.Log;

/**
 * Created by toby on 2017/4/29.
 */

class PermissionsChecker {
    private final Context mContext;

    PermissionsChecker(Context context) {
        mContext = context.getApplicationContext();
    }

    // 判斷權(quán)限集合
    boolean lacksPermissions(String... permissions) {
        for (String permission : permissions) {
            if (lacksPermission(permission)) {
                Log.d("","jkdfksahdfkhsadkjghdfskg: " + permission);
                return true;
            }
        }
        return false;
    }

    // 判斷是否缺少權(quán)限
    private boolean lacksPermission(String permission) {
        return ContextCompat.checkSelfPermission(mContext, permission) ==
                PackageManager.PERMISSION_DENIED;
    }
}

授權(quán)頁

授權(quán)頁, 首先使用系統(tǒng)默認(rèn)的授權(quán)頁, 當(dāng)用戶拒絕時(shí), 指導(dǎo)用戶手動(dòng)設(shè)置, 當(dāng)用戶再次操作失敗后, 返回繼續(xù)提示儒飒。用戶手動(dòng)退出授權(quán)頁時(shí), 給使用頁發(fā)送授權(quán)失敗的通知。我們可以保存這個(gè)授權(quán)頁并將其在多個(gè)應(yīng)用中復(fù)用箩祥。

package com.toby.personal.testlistview;

import android.app.Activity;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Bundle;
import android.provider.Settings;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.app.ActivityCompat;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.AppCompatActivity;

/**
 * Created by toby on 2017/4/29.
 */

public class PermissionsActivity extends AppCompatActivity {
    public static final int PERMISSIONS_GRANTED = 0; // 權(quán)限授權(quán)
    public static final int PERMISSIONS_DENIED = 1; // 權(quán)限拒絕

    private static final int PERMISSION_REQUEST_CODE = 0; // 系統(tǒng)權(quán)限管理頁面的參數(shù)
    private static final String EXTRA_PERMISSIONS =
            "com.toby.personal.permission.extra_permission"; // 權(quán)限參數(shù)
    private static final String PACKAGE_URL_SCHEME = "package:"; // 方案

    private PermissionsChecker mChecker; // 權(quán)限檢測(cè)器
    private boolean isRequireCheck; // 是否需要系統(tǒng)權(quán)限檢測(cè), 防止和系統(tǒng)提示框重疊

    // 啟動(dòng)當(dāng)前權(quán)限頁面的公開接口
    public static void startActivityForResult(Activity activity, int requestCode, String... permissions) {
        Intent intent = new Intent(activity, PermissionsActivity.class);
        intent.putExtra(EXTRA_PERMISSIONS, permissions);
        ActivityCompat.startActivityForResult(activity, intent, requestCode, null);
    }

    @Override protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (getIntent() == null || !getIntent().hasExtra(EXTRA_PERMISSIONS)) {
            throw new RuntimeException("PermissionsActivity需要使用靜態(tài)startActivityForResult方法啟動(dòng)!");
        }
        setContentView(R.layout.activity_permissions);

        mChecker = new PermissionsChecker(this);
        isRequireCheck = true;
    }

    @Override protected void onResume() {
        super.onResume();
        if (isRequireCheck) {
            String[] permissions = getPermissions();
            if (mChecker.lacksPermissions(permissions)) {
                requestPermissions(permissions); // 請(qǐng)求權(quán)限
            } else {
                allPermissionsGranted(); // 全部權(quán)限都已獲取
            }
        } else {
            isRequireCheck = true;
        }
    }

    // 返回傳遞的權(quán)限參數(shù)
    private String[] getPermissions() {
        return getIntent().getStringArrayExtra(EXTRA_PERMISSIONS);
    }

    // 請(qǐng)求權(quán)限兼容低版本
    private void requestPermissions(String... permissions) {
        ActivityCompat.requestPermissions(this, permissions, PERMISSION_REQUEST_CODE);
    }

    // 全部權(quán)限均已獲取
    private void allPermissionsGranted() {
        setResult(PERMISSIONS_GRANTED);
        finish();
    }

    /**
     * 用戶權(quán)限處理,
     * 如果全部獲取, 則直接過.
     * 如果權(quán)限缺失, 則提示Dialog.
     *
     * @param requestCode  請(qǐng)求碼
     * @param permissions  權(quán)限
     * @param grantResults 結(jié)果
     */
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if (requestCode == PERMISSION_REQUEST_CODE && hasAllPermissionsGranted(grantResults)) {
            isRequireCheck = true;
            allPermissionsGranted();
        } else {
            isRequireCheck = false;
            showMissingPermissionDialog();
        }
    }

    // 含有全部的權(quán)限
    private boolean hasAllPermissionsGranted(@NonNull int[] grantResults) {
        for (int grantResult : grantResults) {
            if (grantResult == PackageManager.PERMISSION_DENIED) {
                return false;
            }
        }
        return true;
    }

    // 顯示缺失權(quán)限提示
    private void showMissingPermissionDialog() {
        AlertDialog.Builder builder = new AlertDialog.Builder(PermissionsActivity.this);
        builder.setTitle(R.string.help);
        builder.setMessage(R.string.string_help_text);

        // 拒絕, 退出應(yīng)用
        builder.setNegativeButton(R.string.quit, new DialogInterface.OnClickListener() {
            @Override public void onClick(DialogInterface dialog, int which) {
                setResult(PERMISSIONS_DENIED);
                finish();
            }
        });

        builder.setPositiveButton(R.string.settings, new DialogInterface.OnClickListener() {
            @Override public void onClick(DialogInterface dialog, int which) {
                startAppSettings();
            }
        });

        builder.setCancelable(false);

        builder.show();
    }

    // 啟動(dòng)應(yīng)用的設(shè)置
    private void startAppSettings() {
        Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
        intent.setData(Uri.parse(PACKAGE_URL_SCHEME + getPackageName()));
        startActivity(intent);
    }
}

需要在 AndroidManifest.xml 中加入權(quán)限的申請(qǐng)內(nèi)容:

    <uses-permission android:name="android.permission.READ_PHONE_STATE" />

主布局文件的內(nèi)容如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/container"
    android:background="@color/colorGray"
    android:orientation="vertical"
    >

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="@string/start"
        android:onClick="start"
        />
</LinearLayout>

主程序文件的內(nèi)容如下:

package com.toby.personal.testlistview;

import android.Manifest;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.telephony.PhoneStateListener;
import android.telephony.TelephonyManager;
import android.util.Log;
import android.view.View;
import android.widget.ListView;
import android.widget.Toast;

import java.io.FileNotFoundException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.util.ArrayList;
import java.util.Date;

public class MainActivity extends AppCompatActivity {

    final private static String TAG = "Toby_Test";
    final private static int REQUEST_CODE = 29; // 請(qǐng)求碼


    static boolean listenerStart = false;

    static final String[] PERMISSIONS = new String[]{
            Manifest.permission.READ_PHONE_STATE
//            ,Manifest.permission.ACCESS_COARSE_LOCATION
//            ,Manifest.permission.ACCESS_FINE_LOCATION
    };

    private PermissionsChecker permissionsChecker;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        Log.d(TAG, "=========== onCreate ===========");
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        permissionsChecker = new PermissionsChecker(this);
    }

    @Override
    protected void onResume() {
        super.onResume();
        if (permissionsChecker.lacksPermissions(PERMISSIONS)) {
            startPermissionsActivity();
        }
    }

    private void startPermissionsActivity() {
        PermissionsActivity.startActivityForResult(this, REQUEST_CODE, PERMISSIONS);
    }

    public void start(View view) {

        if (!listenerStart) {

            final TelephonyManager telephonyManager =
                    (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);

            PhoneStateListener listener = new PhoneStateListener() {
                @Override
                public void onCallStateChanged(int state, String incomingNumber) {

                    switch (state) {
                        case TelephonyManager.CALL_STATE_RINGING:
                            OutputStream os = null;
                            try {
                                os = openFileOutput("phoneList", MODE_APPEND);
                            } catch (FileNotFoundException e) {
                                e.printStackTrace();
                            }
                            PrintStream ps = new PrintStream(os);
                            ps.println(new Date() + " 來電: " + incomingNumber);
                            ps.close();
                            break;
                        default:
                            break;
                    }

                    super.onCallStateChanged(state, incomingNumber);
                }
            };

            telephonyManager.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);

            listenerStart = true;
        }

    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        // 拒絕時(shí), 關(guān)閉頁面, 缺少主要權(quán)限, 無法運(yùn)行
        if (requestCode == REQUEST_CODE && resultCode == PermissionsActivity.PERMISSIONS_DENIED) {
            finish();
        }
    }
}

我們可以在 /data/data/com.toby.personal.myapplication/files/phoneList 文件中查看監(jiān)聽的內(nèi)容:

顯示效果

程序的運(yùn)行效果志电,各位可以自行嘗試骇笔,如果有什么問題彻消,可以留言一起討論竿拆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末宾尚,一起剝皮案震驚了整個(gè)濱河市丙笋,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌煌贴,老刑警劉巖御板,帶你破解...
    沈念sama閱讀 211,817評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異牛郑,居然都是意外死亡怠肋,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,329評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門淹朋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來笙各,“玉大人,你說我怎么就攤上這事础芍¤厩溃” “怎么了?”我有些...
    開封第一講書人閱讀 157,354評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵者甲,是天一觀的道長春感。 經(jīng)常有香客問我,道長虏缸,這世上最難降的妖魔是什么鲫懒? 我笑而不...
    開封第一講書人閱讀 56,498評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮刽辙,結(jié)果婚禮上窥岩,老公的妹妹穿的比我還像新娘。我一直安慰自己宰缤,他們只是感情好颂翼,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,600評(píng)論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著慨灭,像睡著了一般朦乏。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上氧骤,一...
    開封第一講書人閱讀 49,829評(píng)論 1 290
  • 那天呻疹,我揣著相機(jī)與錄音,去河邊找鬼筹陵。 笑死刽锤,一個(gè)胖子當(dāng)著我的面吹牛镊尺,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播并思,決...
    沈念sama閱讀 38,979評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼庐氮,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了宋彼?” 一聲冷哼從身側(cè)響起弄砍,我...
    開封第一講書人閱讀 37,722評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎宙暇,沒想到半個(gè)月后输枯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,189評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡占贫,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,519評(píng)論 2 327
  • 正文 我和宋清朗相戀三年桃熄,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片型奥。...
    茶點(diǎn)故事閱讀 38,654評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡瞳收,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出厢汹,到底是詐尸還是另有隱情螟深,我是刑警寧澤,帶...
    沈念sama閱讀 34,329評(píng)論 4 330
  • 正文 年R本政府宣布烫葬,位于F島的核電站界弧,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏搭综。R本人自食惡果不足惜垢箕,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,940評(píng)論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望兑巾。 院中可真熱鬧条获,春花似錦、人聲如沸蒋歌。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,762評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽堂油。三九已至修档,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間府框,已是汗流浹背萍悴。 一陣腳步聲響...
    開封第一講書人閱讀 31,993評(píng)論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留寓免,地道東北人癣诱。 一個(gè)月前我還...
    沈念sama閱讀 46,382評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像袜香,于是被迫代替她去往敵國和親撕予。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,543評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,790評(píng)論 25 707
  • 周日送兒子回老家蜈首,發(fā)現(xiàn)不是他沒了媽媽不行实抡,是我沒了他不行。 自己回來欢策,睡他的床吆寨,想著他跟我說再見時(shí)候的交待,好像他...
    藍(lán)天虹閱讀 208評(píng)論 0 0
  • 我雖然骨子里是個(gè)懶的踩寇,卻鮮少做賴床的事情啄清。或許這跟出身農(nóng)村有關(guān)俺孙,小學(xué)三年級(jí)前都是母親或者爺爺送我上學(xué)辣卒,他們都是早起...
    小紅帽_8c52閱讀 162評(píng)論 0 0