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;
}
}
}
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