Android Studio BroadcastReceiver не работает в Android 6.0

Поэтому я пытаюсь получить информацию о входящих вызовах и SMS в своем приложении. Я попробовал код, показанный ниже, и он работает на Oneplus 6t (android 10), Moto g plus (android 8) и Samsung M51 (android 10), и он работает так, как предполагалось. но при использовании на Redmi Note 3 (android 6.0.1) часть SMS работает, но часть входящего вызова не работает. Я пытался искать в Google и Stack Overflow, но пока ничего не решило проблему.

AndroidManifest.xml

Разрешения

<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.READ_CONTACTS"/>
<uses-permission android:name="android.permission.READ_CALL_LOG"/>
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
<uses-permission android:name="android.permission.READ_SMS" />

Получатель

<receiver android:name="com.earthenergy.earthenergyev.AnswerCallBroadcastReceiver" android:enabled="true">
        <intent-filter android:priority="100">
            <action android:name="android.intent.action.PHONE_STATE" />
            <action android:name="android.intent.action.NEW_OUTGOING_CALL" />
            <action android:name="android.provider.Telephony.SMS_RECEIVED" />
        </intent-filter>
</receiver>

Класс BroadcastReceiver

public class AnswerCallBroadcastReceiver extends BroadcastReceiver {

static PhonecallStartEndDetector listener;
String contactName;
Context context;
private static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED";


private SmsMessage getIncomingMessage(Object aObject, Bundle bundle) {
    SmsMessage currentSMS;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        String format = bundle.getString("format");
        currentSMS = SmsMessage.createFromPdu((byte[]) aObject, format);
    } else {
        currentSMS = SmsMessage.createFromPdu((byte[]) aObject);
    }
    return currentSMS;
}

@Override
public void onReceive(Context context, Intent intent) {
    this.context = context;

    if (listener == null) {
        listener = new PhonecallStartEndDetector();

    }

    if (intent.getAction()
            .equals("android.intent.action.NEW_OUTGOING_CALL")) {
        listener.setOutgoingNumber(intent.getExtras().getString(
                "android.intent.extra.PHONE_NUMBER"));

        return;
    }

    if(intent.getAction().equals(SMS_RECEIVED)) {
        Bundle bundle = intent.getExtras();
        if(bundle != null) {
            Object[] pdus = (Object[]) bundle.get("pdus");
            SmsMessage currentSms;
            for(Object aObject: pdus) {
                currentSms = getIncomingMessage(aObject, bundle);
                String sender = currentSms.getDisplayOriginatingAddress();
                String message = currentSms.getDisplayMessageBody();
                Log.d("============","Message: "+ message);
                Log.d("============","Sender: "+ sender);
                Toast.makeText(context,"Number: "+sender+"\nMessage: "+message,Toast.LENGTH_SHORT).show();
            }
        }
        return;
    }

    TelephonyManager telephony = (TelephonyManager) context
            .getSystemService(Context.TELEPHONY_SERVICE);
    telephony.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);
}

public class PhonecallStartEndDetector extends PhoneStateListener {
    int lastState = TelephonyManager.CALL_STATE_IDLE;
    boolean isIncoming;
    boolean isIncomingPicked;
    boolean isOutgoingStarted;
    String savedNumber; 

    public PhonecallStartEndDetector() {

    }

    public void setOutgoingNumber(String number) {
        savedNumber = number;
    }

    @Override
    public void onCallStateChanged(int state, String incomingNumber) {
        super.onCallStateChanged(state, incomingNumber);
        // state is changed
        if (lastState == state) {
            return;
        }

        switch (state) {
            case TelephonyManager.CALL_STATE_RINGING:
                //an incoming call has been started
                String name = getContactName(incomingNumber, context);
                Toast.makeText(context, "Incoming Number: " + incomingNumber + "\nContact Name: " + name, Toast.LENGTH_SHORT).show();
                Log.d("==============", "Number: " + incomingNumber);
                Log.d("==============", "Name: " + name);
                isIncoming = true;
                savedNumber = incomingNumber;

                break;

        }

    }
}

person Chinmay Todankar    schedule 21.12.2020    source источник
comment
github.com/devggaurav/ взгляните на это   -  person Danial clarc    schedule 21.12.2020


Ответы (1)


В случае версии Marshmallow у нас есть концепция, называемая разрешением Runtime, которое должно быть сделано внутри Activity, чтобы работать с разрешением. Разрешение во время выполнения предоставляет способ запросить у пользователя конкретное разрешение во время выполнения, когда он запускает действие в первый раз.

Это две вещи, которые вы должны указать:

// указываем любой постоянный номер для разрешения

public final static int MY_PERMISSIONS_REQUEST_READ_PHONE_STATE = 11;

// Указываем следующий бит кода в методе OnCreate

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

    // Here, thisActivity is the current activity
    if (ContextCompat.checkSelfPermission(getApplicationContext(),
            Manifest.permission.READ_PHONE_STATE)
            != PackageManager.PERMISSION_GRANTED) {

        // Should we show an explanation?
        if (ActivityCompat.shouldShowRequestPermissionRationale(this,
                Manifest.permission.READ_CONTACTS)) {
        } else {
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.READ_PHONE_STATE},
                    MY_PERMISSIONS_REQUEST_READ_PHONE_STATE);
        }
    }
}

//указать этот метод, который будет всплывающее окно с запросом разрешения пользователя во время выполнения

@Override
public void onRequestPermissionsResult(int requestCode,
                                       String permissions[], int[] grantResults) {
    switch (requestCode) {
        case MY_PERMISSIONS_REQUEST_READ_PHONE_STATE: {
            // If request is cancelled, the result arrays are empty.
            if (grantResults.length > 0
                    && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                } else {
            }
            return;
        }
    }
}

это обеспечит способ работы с устройствами Marshmallow

person Usama Altaf    schedule 21.12.2020
comment
Привет, я добавил часть для проверки и запроса разрешения, но она не работает на Android 6.0.1 (Redmi Note 3), она работает на других устройствах. - person Chinmay Todankar; 22.12.2020
comment
Покажи свой логкэт - person Usama Altaf; 22.12.2020