วิธีการตรวจจับสายเรียกเข้าในอุปกรณ์ Android?


130

ฉันกำลังพยายามสร้างแอพเช่นเมื่อมีสายเข้ามาในโทรศัพท์ฉันต้องการตรวจจับหมายเลข ด้านล่างเป็นสิ่งที่ฉันพยายาม แต่ไม่พบสายเรียกเข้า

ฉันต้องการเรียกใช้เป็นMainActivityพื้นหลังฉันจะทำอย่างไร

ฉันได้รับอนุญาตในmanifestไฟล์

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

มีอะไรอีกบ้างที่ฉันควรให้ในรายการ?

public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.test_layout);
   }

   public class myPhoneStateChangeListener extends PhoneStateListener {
       @Override
       public void onCallStateChanged(int state, String incomingNumber) {
           super.onCallStateChanged(state, incomingNumber);
           if (state == TelephonyManager.CALL_STATE_RINGING) {
               String phoneNumber =   incomingNumber;
           }
       }
   }
}

เราควรทำอย่างไรสำหรับ Android P
Ahmad Arslan

คำตอบ:


336

นี่คือสิ่งที่ฉันใช้ทำสิ่งนี้:

Manifest:

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

<!--This part is inside the application-->
    <receiver android:name=".CallReceiver" >
        <intent-filter>
            <action android:name="android.intent.action.PHONE_STATE" />
        </intent-filter>
        <intent-filter>
            <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
        </intent-filter>
    </receiver>

เครื่องตรวจจับการโทรที่นำมาใช้ใหม่ของฉันได้

package com.gabesechan.android.reusable.receivers;

import java.util.Date;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.telephony.TelephonyManager;

public abstract class PhonecallReceiver extends BroadcastReceiver {

    //The receiver will be recreated whenever android feels like it.  We need a static variable to remember data between instantiations

    private static int lastState = TelephonyManager.CALL_STATE_IDLE;
    private static Date callStartTime;
    private static boolean isIncoming;
    private static String savedNumber;  //because the passed incoming is only valid in ringing


    @Override
    public void onReceive(Context context, Intent intent) {

        //We listen to two intents.  The new outgoing call only tells us of an outgoing call.  We use it to get the number.
        if (intent.getAction().equals("android.intent.action.NEW_OUTGOING_CALL")) {
            savedNumber = intent.getExtras().getString("android.intent.extra.PHONE_NUMBER");
        }
        else{
            String stateStr = intent.getExtras().getString(TelephonyManager.EXTRA_STATE);
            String number = intent.getExtras().getString(TelephonyManager.EXTRA_INCOMING_NUMBER);
            int state = 0;
            if(stateStr.equals(TelephonyManager.EXTRA_STATE_IDLE)){
                state = TelephonyManager.CALL_STATE_IDLE;
            }
            else if(stateStr.equals(TelephonyManager.EXTRA_STATE_OFFHOOK)){
                state = TelephonyManager.CALL_STATE_OFFHOOK;
            }
            else if(stateStr.equals(TelephonyManager.EXTRA_STATE_RINGING)){
                state = TelephonyManager.CALL_STATE_RINGING;
            }


            onCallStateChanged(context, state, number);
        }
    }

    //Derived classes should override these to respond to specific events of interest
    protected abstract void onIncomingCallReceived(Context ctx, String number, Date start);
    protected abstract void onIncomingCallAnswered(Context ctx, String number, Date start);
    protected abstract void onIncomingCallEnded(Context ctx, String number, Date start, Date end);

    protected abstract void onOutgoingCallStarted(Context ctx, String number, Date start);      
    protected abstract void onOutgoingCallEnded(Context ctx, String number, Date start, Date end);

    protected abstract void onMissedCall(Context ctx, String number, Date start);

    //Deals with actual events

    //Incoming call-  goes from IDLE to RINGING when it rings, to OFFHOOK when it's answered, to IDLE when its hung up
    //Outgoing call-  goes from IDLE to OFFHOOK when it dials out, to IDLE when hung up
    public void onCallStateChanged(Context context, int state, String number) {
        if(lastState == state){
            //No change, debounce extras
            return;
        }
        switch (state) {
            case TelephonyManager.CALL_STATE_RINGING:
                isIncoming = true;
                callStartTime = new Date();
                savedNumber = number;
                onIncomingCallReceived(context, number, callStartTime);
                break;
            case TelephonyManager.CALL_STATE_OFFHOOK:
                //Transition of ringing->offhook are pickups of incoming calls.  Nothing done on them
                if(lastState != TelephonyManager.CALL_STATE_RINGING){
                    isIncoming = false;
                    callStartTime = new Date();
                    onOutgoingCallStarted(context, savedNumber, callStartTime);                     
                }
                else
                {
                    isIncoming = true;
                    callStartTime = new Date();
                    onIncomingCallAnswered(context, savedNumber, callStartTime); 
                }

                break;
            case TelephonyManager.CALL_STATE_IDLE:
                //Went to idle-  this is the end of a call.  What type depends on previous state(s)
                if(lastState == TelephonyManager.CALL_STATE_RINGING){
                    //Ring but no pickup-  a miss
                    onMissedCall(context, savedNumber, callStartTime);
                }
                else if(isIncoming){
                    onIncomingCallEnded(context, savedNumber, callStartTime, new Date());                       
                }
                else{
                    onOutgoingCallEnded(context, savedNumber, callStartTime, new Date());                                               
                }
                break;
        }
        lastState = state;
    }
}

จากนั้นให้ใช้เพียงแค่คลาสที่ได้รับมาจากนั้นใช้ฟังก์ชั่นง่าย ๆ ไม่กี่อย่าง

public class CallReceiver extends PhonecallReceiver {

    @Override
    protected void onIncomingCallReceived(Context ctx, String number, Date start)
    {
        //
    }

    @Override
    protected void onIncomingCallAnswered(Context ctx, String number, Date start)
    {
        //
    }

    @Override
    protected void onIncomingCallEnded(Context ctx, String number, Date start, Date end)
    {
        //
    }

    @Override
    protected void onOutgoingCallStarted(Context ctx, String number, Date start)
    {
        //
    } 

    @Override 
    protected void onOutgoingCallEnded(Context ctx, String number, Date start, Date end)
    {
        //
    }

    @Override
    protected void onMissedCall(Context ctx, String number, Date start)
    {
        //
    }

}

นอกจากนี้คุณสามารถดู writeup ฉันไม่เกี่ยวกับว่าทำไมรหัสเป็นเหมือนมันเป็นของฉันบล็อก ลิงค์ส่วนสำคัญ: https://gist.github.com/ftvs/e61ccb039f511eb288ee

แก้ไข: อัปเดตเป็นรหัสที่ง่ายขึ้นเนื่องจากฉันได้ทำคลาสใหม่สำหรับการใช้งานของฉันเอง


2
รหัสนี้ไม่แสดงอะไรเลย มันจะโทรหาคุณเมื่อการโทรออกเริ่มต้น / สิ้นสุดและส่งต่อหมายเลขเวลาเริ่มและเวลาที่คุณทำเสร็จแล้ว การแสดงจริงเป็นงานของคุณเพราะฉันไม่รู้วิธีที่คุณต้องการ
Gabe Sechan

3
@GabeSechan: ยอดเยี่ยม! คุณช่วยแนะนำฉันให้จัดการกับสถานการณ์การรอสายได้ไหม
Mehul Joisar

6
เพียงเพื่อเพิ่มสิ่งนี้มันไม่ทำงานเมื่อแอปไม่ได้อยู่เบื้องหน้าหรือพื้นหลังจนกว่าฉันจะเพิ่มสิ่งนี้ลงในผู้รับ: "android: enabled =" true "
Rajat Sharma

1
ตัวแปรแบบคงที่จะอยู่รอบ ๆ จนกว่าแอปพลิเคชันจะถูกเตะออกจากหน่วยความจำ (ซึ่งอาจเป็นเวลานานขึ้นอยู่กับว่าบริการกำลังทำงานอยู่และเงื่อนไขหน่วยความจำโทรศัพท์ทั่วไป) แต่ใช่พวกเขาอาจสูญหายได้ คุณสามารถเขียนลงดิสก์พูดผ่านการตั้งค่าแบบแบ่งใช้ แต่อาจทำให้คุณได้ผลลัพธ์ที่ผิดพลาดเช่นกันซึ่งจะช่วยป้องกันไม่ให้คุณล้างข้อมูลของคุณอย่างถูกต้องในหลาย ๆ กรณีเช่นรอบการรีบูตเครื่อง สำหรับกรณีการใช้งานของฉันข้อมูลที่หายากและข้อมูลที่หายไปนั้นดีกว่าข้อมูลที่ไม่ถูกต้อง อย่าลังเลที่จะเล่นกับสิ่งนี้ตามความต้องการของคุณ
Gabe Sechan

1
@GabeSechan: ดูเหมือนจะมีข้อผิดพลาดในนั้น ไม่ควรที่จะเริ่มต้นได้ที่lastState CALL_STATE_IDLEฉันหายไปไม่กี่สายเมื่อ app RINGINGของฉันถูกฆ่าตายในขณะที่รัฐปัจจุบันคือ เพราะเมื่อมันกลายเป็นIDLEสายอีกครั้งเมื่อสิ้นสุดการโทรตัวแปรแบบคงที่จะเริ่มต้นใหม่CALL_STATE_IDLEและมัน debounces ไม่ทำอะไรเลย lastStateดังนั้นเราจะสูญเสียการอ้างอิงถึง
ไฮเซนเบิร์ก

23
private MyPhoneStateListener phoneStateListener = new MyPhoneStateListener();

เพื่อลงทะเบียน

TelephonyManager telephonyManager = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_CALL_STATE);

และเพื่อยกเลิกการลงทะเบียน

TelephonyManager telephonyManager = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE);

ฉันควรใช้สิ่งนี้ในการทำกิจกรรมสำคัญ ๆ ที่ไหน?
Jesbin MJ

วางสิ่งนี้ไว้ในชั้นเรียนที่คุณฟังการเปลี่ยนแปลง .. โดยปกติแล้วการลงทะเบียนจะทำใน oncreate และ unregister ใน ondestroy .. ประกาศวัตถุทั่วโลกในชั้นเรียน
stinepike

ตัวเลือกนี้ไม่ต้องการการอนุญาต
Mike

ในขณะที่โซลูชันของ Gabe นั้นเหมาะสำหรับฟีเจอร์ที่น่ารำคาญมากกว่าเช่นแอพประเภท Viber นี่เป็นทางออกที่ดีที่สุดสำหรับผู้ที่ต้องการตอบสนองต่อการกระทำของผู้ใช้ในการโทรศัพท์ การขออนุญาตรันไทม์ในกรณีเช่นนี้น่าจะเป็นการ overkill และผู้ใช้อาจไม่ได้รับ
bosphere

1
ฉันต้องการทำบางสิ่งเมื่อมีสายเข้ามาจะทำอย่างไรกับรหัสนี้
Noor Hossain

14

ด้วย Android P - Api ระดับ 28: คุณต้องได้รับอนุญาต READ_CALL_LOG

การ จำกัด การเข้าถึงบันทึกการโทร

Android P ย้ายCALL_LOG, READ_CALL_LOG, WRITE_CALL_LOGและPROCESS_OUTGOING_CALLSสิทธิ์จากPHONEกลุ่มการอนุญาตใหม่CALL_LOGกลุ่มการอนุญาต กลุ่มนี้ช่วยให้ผู้ใช้สามารถควบคุมและเปิดเผยแอพที่ดีขึ้นซึ่งต้องการการเข้าถึงข้อมูลที่ละเอียดอ่อนเกี่ยวกับการโทรเช่นการอ่านบันทึกการโทรและการระบุหมายเลขโทรศัพท์

การอ่านตัวเลขจาก PHONE_STATE การกระทำที่เจตนาคุณจำเป็นต้องใช้ทั้งREAD_CALL_LOGได้รับอนุญาตและREAD_PHONE_STATEได้รับอนุญาต เพื่ออ่านตัวเลขจากonCallStateChanged()ตอนนี้คุณต้องREAD_CALL_LOGได้รับอนุญาตเท่านั้น คุณไม่ต้องการการREAD_PHONE_STATEอนุญาตอีกต่อไป


สำหรับผู้ที่เป็นเพียงการเพิ่มREAD_CALL_LOGในการมุ่งเน้นการเพิ่มคำขอสิทธิ์ในAndroidManifest.xml MainActivity
Piyush

13

อัปเดต: โค้ดสุดยอดจริงๆที่โพสต์โดยGabe Sechanไม่ทำงานอีกต่อไปเว้นแต่คุณจะขอให้ผู้ใช้ให้สิทธิ์ที่จำเป็น นี่คือรหัสบางส่วนที่คุณสามารถวางในกิจกรรมหลักของคุณเพื่อขอสิทธิ์เหล่านี้:

    if (getApplicationContext().checkSelfPermission(Manifest.permission.READ_PHONE_STATE)
            != PackageManager.PERMISSION_GRANTED) {
        // Permission has not been granted, therefore prompt the user to grant permission
        ActivityCompat.requestPermissions(this,
                new String[]{Manifest.permission.READ_PHONE_STATE},
                MY_PERMISSIONS_REQUEST_READ_PHONE_STATE);
    }

    if (getApplicationContext().checkSelfPermission(Manifest.permission.PROCESS_OUTGOING_CALLS)
            != PackageManager.PERMISSION_GRANTED) {
        // Permission has not been granted, therefore prompt the user to grant permission
        ActivityCompat.requestPermissions(this,
                new String[]{Manifest.permission.PROCESS_OUTGOING_CALLS},
                MY_PERMISSIONS_REQUEST_PROCESS_OUTGOING_CALLS);
    }

นอกจากนี้: ตามที่มีคนพูดถึงในความคิดเห็นด้านล่างโพสต์ของ Gabeคุณต้องเพิ่มข้อมูลโค้ดเล็กน้อยandroid:enabled="trueเพื่อผู้รับเพื่อตรวจจับสายเรียกเข้าเมื่อแอพไม่ได้ทำงานอยู่เบื้องหน้า:

    <!--This part is inside the application-->
    <receiver android:name=".CallReceiver" android:enabled="true">
        <intent-filter>
            <action android:name="android.intent.action.PHONE_STATE" />
        </intent-filter>
        <intent-filter>
            <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
        </intent-filter>
    </receiver>

จะทำอย่างไรถ้าแอปพลิเคชันไม่มีกิจกรรมใด ๆ และมีเพียงผู้ส่งสัญญาณออกอากาศและบริการเท่านั้น จากนั้นเราจะเขียนรหัสนี้ที่ไหนเพื่อขออนุญาตจากผู้ใช้เนื่องจากผู้ออกอากาศจะไม่ถูกเรียกจนกว่าจะได้รับการอนุญาตนี้
user2779311

2
อย่างน้อยคุณต้องการ MainActivity แม้ว่าจะเปิดเพียงครั้งเดียว ใช้แอปปิดกั้นการโทรของฉัน RoboStop เช่น: เมื่อผู้ใช้ดาวน์โหลดแอปเป็นครั้งแรกจากนั้นคลิกที่ไอคอนแอพเพื่อเปิดแอพพวกเขาจะได้รับแจ้งให้อนุญาตแอพของฉัน แอพยังมีปุ่มเพื่อเปิด / ปิดการบล็อคการโทร แต่ผู้ใช้ไม่จำเป็นต้องเปิดแอพ / กิจกรรมอีกการบล็อคการโทรจะเกิดขึ้นในพื้นหลังโดยที่ผู้ใช้ไม่ต้องเปิดแอพ / กิจกรรมอีก
topherPedersen

ผู้ที่มีปัญหาในการใช้งานให้ทำตามกวดวิชานี้studytutorial.in/ …
Prasath

5

สิ่งนี้อาจช่วยคุณและเพิ่มต้องได้รับอนุญาต

public class PhoneListener extends PhoneStateListener
{
    private Context context;
    public static String getincomno;

    public PhoneListener(Context c) {
        Log.i("CallRecorder", "PhoneListener constructor");
        context = c;
    }

    public void onCallStateChanged (int state, String incomingNumber)
    {

        if(!TextUtils.isEmpty(incomingNumber)){
        // here for Outgoing number make null to get incoming number
        CallBroadcastReceiver.numberToCall = null;
        getincomno = incomingNumber;
        }

        switch (state) {
        case TelephonyManager.CALL_STATE_IDLE:

            break;
        case TelephonyManager.CALL_STATE_RINGING:
            Log.d("CallRecorder", "CALL_STATE_RINGING");
            break;
        case TelephonyManager.CALL_STATE_OFFHOOK:

            break;
        }
    }
}

2
ดูเหมือนว่าจะดี แต่ฉันจะใช้มันจากกิจกรรมได้อย่างไร โปรดบอกรายละเอียดกับฉัน
อาเมียร์

4

เพียงเพื่ออัปเดตคำตอบของ Gabe Sechan หากรายการของคุณขอสิทธิ์ในการ READ_CALL_LOG และ READ_PHONE_STATE จะเรียกว่า onReceiveสองครั้ง หนึ่งในนั้นมี EXTRA_INCOMING_NUMBER อยู่และอีกอันหนึ่งไม่มี คุณต้องทดสอบว่ามันมีอะไรและมันสามารถเกิดขึ้นได้ในลำดับใด ๆ

https://developer.android.com/reference/android/telephony/TelephonyManager.html#ACTION_PHONE_STATE_CHANGED


2

นี่คือวิธีการง่ายๆที่สามารถหลีกเลี่ยงการใช้PhonestateListenerและภาวะแทรกซ้อนอื่น ๆ
ดังนั้นที่นี่เราจะได้รับเหตุการณ์ที่ 3 จากหุ่นยนต์เช่นRINGING, และOFFHOOK IDLEและเพื่อที่จะได้รับสถานะเป็นไปได้ทั้งหมดของการโทรเราจำเป็นต้องกำหนดรัฐของเราเองชอบRINGING, OFFHOOK, IDLE, ,FIRST_CALL_RINGING SECOND_CALL_RINGINGมันสามารถจัดการทุกรัฐในการโทร
โปรดคิดในวิธีที่เรารับกิจกรรมจาก android และเราจะกำหนดสถานะการโทรของเรา ดูรหัส

public class CallListening  extends BroadcastReceiver {
    private static final String TAG ="broadcast_intent";
    public static String incoming_number;
    private String current_state,previus_state,event;
    public static Boolean dialog= false;
    private Context context;
    private SharedPreferences sp,sp1;
    private SharedPreferences.Editor spEditor,spEditor1;
    public void onReceive(Context context, Intent intent) {
        //Log.d("intent_log", "Intent" + intent);
        dialog=true;
        this.context = context;
        event = intent.getStringExtra(TelephonyManager.EXTRA_STATE);
        incoming_number = intent.getStringExtra(TelephonyManager.EXTRA_INCOMING_NUMBER);
        Log.d(TAG, "The received event : "+event+", incoming_number : " + incoming_number);
        previus_state = getCallState(context);
        current_state = "IDLE";
        if(incoming_number!=null){
            updateIncomingNumber(incoming_number,context);
        }else {
            incoming_number=getIncomingNumber(context);
        }
        switch (event) {
            case "RINGING":
                Log.d(TAG, "State : Ringing, incoming_number : " + incoming_number);
            if((previus_state.equals("IDLE")) || (previus_state.equals("FIRST_CALL_RINGING"))){
                    current_state ="FIRST_CALL_RINGING";
                }
                if((previus_state.equals("OFFHOOK"))||(previus_state.equals("SECOND_CALL_RINGING"))){
                    current_state = "SECOND_CALL_RINGING";
                }

                break;
            case "OFFHOOK":
                Log.d(TAG, "State : offhook, incoming_number : " + incoming_number);
                if((previus_state.equals("IDLE")) ||(previus_state.equals("FIRST_CALL_RINGING")) || previus_state.equals("OFFHOOK")){
                    current_state = "OFFHOOK";
                }
                if(previus_state.equals("SECOND_CALL_RINGING")){
                    current_state ="OFFHOOK";
                    startDialog(context);
                }
                break;
            case "IDLE":
                Log.d(TAG, "State : idle and  incoming_number : " + incoming_number);
                if((previus_state.equals("OFFHOOK")) || (previus_state.equals("SECOND_CALL_RINGING")) || (previus_state.equals("IDLE"))){
                    current_state="IDLE";
                }
                if(previus_state.equals("FIRST_CALL_RINGING")){
                    current_state = "IDLE";
                    startDialog(context);
                }
                updateIncomingNumber("no_number",context);
                Log.d(TAG,"stored incoming number flushed");
                break;
        }
        if(!current_state.equals(previus_state)){
            Log.d(TAG, "Updating  state from "+previus_state +" to "+current_state);
            updateCallState(current_state,context);

        }
    }
    public void startDialog(Context context) {
        Log.d(TAG,"Starting Dialog box");
        Intent intent1 = new Intent(context, NotifyHangup.class);
        intent1.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(intent1);

    }
    public void updateCallState(String state,Context context){
        sp = PreferenceManager.getDefaultSharedPreferences(context);
        spEditor = sp.edit();
        spEditor.putString("call_state", state);
        spEditor.commit();
        Log.d(TAG, "state updated");

    }
    public void updateIncomingNumber(String inc_num,Context context){
        sp = PreferenceManager.getDefaultSharedPreferences(context);
        spEditor = sp.edit();
        spEditor.putString("inc_num", inc_num);
        spEditor.commit();
        Log.d(TAG, "incoming number updated");
    }
    public String getCallState(Context context){
        sp1 = PreferenceManager.getDefaultSharedPreferences(context);
        String st =sp1.getString("call_state", "IDLE");
        Log.d(TAG,"get previous state as :"+st);
        return st;
    }
    public String getIncomingNumber(Context context){
        sp1 = PreferenceManager.getDefaultSharedPreferences(context);
        String st =sp1.getString("inc_num", "no_num");
        Log.d(TAG,"get incoming number as :"+st);
        return st;
    }
}

0

@Gabe Sechan ขอบคุณสำหรับรหัสของคุณ มันทำงานได้ดียกเว้นonOutgoingCallEnded()มันทำงานได้ดียกเว้นมันไม่เคยถูกประหารชีวิต โทรศัพท์ทดสอบคือ Samsung S5 & Trendy ฉันคิดว่ามีข้อบกพร่อง 2 ข้อ

1: วงเล็บขาดหายไปหนึ่งคู่

case TelephonyManager.CALL_STATE_IDLE: 
    // Went to idle-  this is the end of a call.  What type depends on previous state(s)
    if (lastState == TelephonyManager.CALL_STATE_RINGING) {
        // Ring but no pickup-  a miss
        onMissedCall(context, savedNumber, callStartTime);
    } else {
        // this one is missing
        if(isIncoming){
            onIncomingCallEnded(context, savedNumber, callStartTime, new Date());                       
        } else {
            onOutgoingCallEnded(context, savedNumber, callStartTime, new Date());                                               
        }
    }
    // this one is missing
    break;

2: lastStateไม่ได้รับการอัพเดตstateถ้าอยู่ที่ท้ายฟังก์ชั่น มันควรจะถูกแทนที่เป็นบรรทัดแรกของฟังก์ชั่นนี้โดย

public void onCallStateChanged(Context context, int state, String number) {
    int lastStateTemp = lastState;
    lastState = state;
    // todo replace all the "lastState" by lastStateTemp from here.
    if (lastStateTemp  == state) {
        //No change, debounce extras
        return;
    }
    //....
}

เพิ่มเติมฉันได้ตั้งค่าlastStateและsavedNumberแบ่งปันการตั้งค่าตามที่คุณแนะนำ

เพิ่งทดสอบกับการเปลี่ยนแปลงข้างต้น แก้ไขข้อผิดพลาดอย่างน้อยในโทรศัพท์ของฉัน


0

กรุณาใช้รหัสด้านล่าง มันจะช่วยให้คุณได้รับหมายเลขเข้ากับรายละเอียดการโทร

activity_main.xml

<RelativeLayout 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"
tools:context=".MainActivity" >

<TextView
    android:id="@+id/call"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_centerHorizontal="true"
    android:layout_centerVertical="true"
    android:text="@string/hello_world" />

</RelativeLayout>

MainActivity.java

public class MainActivity extends Activity {

private static final int MISSED_CALL_TYPE = 0;
private TextView txtcall;

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

    txtcall = (TextView) findViewById(R.id.call);

    StringBuffer sb = new StringBuffer();
    Cursor managedCursor = managedQuery(CallLog.Calls.CONTENT_URI, null,
            null, null, null);
    int number = managedCursor.getColumnIndex(CallLog.Calls.NUMBER);
    int type = managedCursor.getColumnIndex(CallLog.Calls.TYPE);
    int date = managedCursor.getColumnIndex(CallLog.Calls.DATE);
    int duration = managedCursor.getColumnIndex(CallLog.Calls.DURATION);
    sb.append("Call Details :");
    while (managedCursor.moveToNext()) {
        String phNumber = managedCursor.getString(number);
        String callType = managedCursor.getString(type);
        String callDate = managedCursor.getString(date);
        Date callDayTime = new Date(Long.valueOf(callDate));
        String callDuration = managedCursor.getString(duration);
        String dir = null;
        int dircode = Integer.parseInt(callType);
        switch (dircode) {

        case CallLog.Calls.OUTGOING_TYPE:
            dir = "OUTGOING";
            break;

        case CallLog.Calls.INCOMING_TYPE:
            dir = "INCOMING";
            break;

        case CallLog.Calls.MISSED_TYPE:
            dir = "MISSED";
            break;
        }
        sb.append("\nPhone Number:--- " + phNumber + " \nCall Type:--- "
                + dir + " \nCall Date:--- " + callDayTime
                + " \nCall duration in sec :--- " + callDuration);
        sb.append("\n----------------------------------");
    }
    managedCursor.close();
    txtcall.setText(sb);
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu; this adds items to the action bar if it is present.
    getMenuInflater().inflate(R.menu.activity_main, menu);
    return true;
}

} 

และในคำขอของคุณสำหรับการอนุญาตดังต่อไปนี้:

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

READ_LOGS ดำเนินการเพื่อให้แอปของคุณถูกแบนจาก Play Store
Duna

0

คุณต้องใช้ BroadcastReceiver สำหรับACTION_PHONE_STATE_CHANGEDสิ่งนี้จะโทรหาคุณได้รับเมื่อใดก็ตามที่สถานะโทรศัพท์เปลี่ยนจากการไม่ได้ใช้งาน, เปิดเสียง, offhook ดังนั้นจากค่าก่อนหน้าและค่าใหม่ที่คุณสามารถตรวจสอบได้ว่าเป็นการโทรเข้า / ออก

การอนุญาตที่จำเป็นจะเป็น:

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

แต่ถ้าคุณต้องการรับ EXTRA_INCOMING_NUMBER ในการออกอากาศนั้นคุณจะต้องได้รับอนุญาตอีกครั้ง: "android.permission.READ_CALL_LOG"

และรหัสบางอย่างเช่นนี้

val receiver: BroadcastReceiver = object : BroadcastReceiver() {
    override fun onReceive(context: Context, intent: Intent) {
        Log.d(TAG, "onReceive")
    }
}

override fun onResume() {
    val filter = IntentFilter()
    filter.addAction("android.intent.action.PHONE_STATE")
    registerReceiver(receiver, filter)
    super.onResume()
}

override fun onPause() {
    unregisterReceiver(receiver)
    super.onPause()
}

และในระดับผู้รับเราสามารถรับสถานะปัจจุบันโดยการอ่านเจตนาเช่นนี้

intent.extras["state"]

ผลลัพธ์ของการเสริมอาจเป็น:

เสียงกริ่ง -> หากโทรศัพท์ของคุณดังขึ้น

OFFHOOK -> หากคุณกำลังคุยกับใครบางคน (สายเรียกเข้าหรือโทรออก)

IDLE -> ถ้าการโทรสิ้นสุดลง (สายเรียกเข้าหรือสายเรียกเข้า)

ด้วยการถ่ายทอด PHONE_STATE เราไม่จำเป็นต้องใช้สิทธิ์ PROCESS_OUTGOING_CALLS หรือเลิกใช้ NEW_OUTGOING_CALL การกระทำ

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.