BroadcastReceiver для входящих звонков не срабатывает через некоторое время
У меня есть приложение для отправки текстовых сообщений на каждый входящий звонок. BroadcastReceiver зарегистрирован в службе, которая находится на переднем плане. По некоторым причинам широковещательный приемник для входящих вызовов не срабатывает. Если пользователь уничтожает и запускает службу, приложение работает некоторое время. Через некоторое время приемник входящего звонка не срабатывает.
Вот класс обслуживания:
public class SendSmsService extends Service {
private static Logger LOGGER = LoggerFactory.getLogger(SendSmsService.class);
private static final String SMS_SENT = "com.sms.sendsms.SMS_SENT";
private static final String SMS_DELIVERED = "com.sms.sendsms.SMS_DELIVERED";
private static final int MAX_SMS_MESSAGE_LENGTH = 160;
private static final int SMS_PORT = 16001;
private int mLastState = -1;
private static String recipientNumber;
private int smsNumber=1;
private NotificationManager mNotificationManager;
private User user;
private SharedPreferences sharedPref;
private AlarmManager alarmManager;
private TelephonyManager telephony;
private PendingIntent keepAlivePendingIntent;
private SharedPreferences prefs;
@Override
public void onCreate() {
super.onCreate();
LOGGER.info("Service is onCreate");
mNotificationManager
= (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
sharedPref = getSharedPreferences(
getString(R.string.preference_file_key), Context.MODE_PRIVATE);
prefs = PreferenceManager.getDefaultSharedPreferences(SendSmsService.this);
LOGGER.info("registering incomingCallReceiver receiver");
LOGGER.info("registering receiver: SMS_SEND");
registerReceiver(sendSmsReceiver, new IntentFilter(SMS_SENT));
registerReceiver(incomingCallReceiver, new IntentFilter(
TelephonyManager.ACTION_PHONE_STATE_CHANGED));
alarmManager = (AlarmManager) this
.getSystemService(Context.ALARM_SERVICE);
telephony = (TelephonyManager) this.getSystemService(Context.TELEPHONY_SERVICE);
Intent intent = new Intent(this, KeepAliveAlarmReceiver.class);
keepAlivePendingIntent = PendingIntent.getBroadcast(this, 0, intent, 0);
alarmManager.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP,
SystemClock.elapsedRealtime(), AlarmManager.INTERVAL_HALF_DAY,
keepAlivePendingIntent);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
LOGGER.info("Service is onStartCommand");
sharedPref.edit().putBoolean(getString(R.string.is_service_running), true).apply();
user = ApplicationLoader.getApplication(SendSmsService.this)
.getDaoSession()
.getUserDao()
.queryBuilder()
.unique();
int iconId = R.drawable.icon;
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(SendSmsService.this)
.setContentTitle(getResources().getText(R.string.app_name))
.setSmallIcon(iconId)
.setWhen(System.currentTimeMillis());
Intent nIntent = getPackageManager().
getLaunchIntentForPackage(ContextConstants.PACKAGE_NAME);
PendingIntent pendingIntent = PendingIntent.getActivity(SendSmsService.this, 0, nIntent,
PendingIntent.FLAG_UPDATE_CURRENT);
notificationBuilder.setContentIntent(pendingIntent);
startForeground(NotificationConstants.LAUNCHER_SERVICE_NOTE_ID,
notificationBuilder.build());
return START_STICKY;
}
private BroadcastReceiver sendSmsReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
LOGGER.info("sendSmsReceiver is received");
String message;
switch (getResultCode()) {
case Activity.RESULT_OK:
message = context.getResources().getString(R.string.msg_sent) + " " + recipientNumber;
SmsLog smsLog = new SmsLog();
String dateString = (String) DateFormat.format("dd/MM/yyyy h:mm a", new java.util.Date());
smsLog.setSentDate(dateString);
smsLog.setSentNumber(recipientNumber);
ApplicationLoader.getApplication(SendSmsService.this)
.getDaoSession()
.getSmsLogDao()
.insert(smsLog);
break;
case SmsManager.RESULT_ERROR_GENERIC_FAILURE:
message = context.getResources().getString(R.string.error_msg);
break;
case SmsManager.RESULT_ERROR_NO_SERVICE:
message = context.getResources().getString(R.string.no_service);
break;
case SmsManager.RESULT_ERROR_NULL_PDU:
message = context.getResources().getString(R.string.error_pdu);
break;
case SmsManager.RESULT_ERROR_RADIO_OFF:
message = context.getResources().getString(R.string.radio_off);
break;
default: {
message = context.getResources().getString(R.string.unknown_error_sent_msg);
break;
}
}
showSmsSendNotification(message);
LOGGER.info("sendSmsReceiver message = " + message);
}
};
@Override
public void onDestroy() {
super.onDestroy();
LOGGER.info("Service is onDestroy");
sharedPref.edit().putBoolean(getString(R.string.is_service_running), false).apply();
telephony.listen(phoneListener, PhoneStateListener.LISTEN_NONE);
unregisterReceiver(incomingCallReceiver);
unregisterReceiver(sendSmsReceiver);
alarmManager.cancel(keepAlivePendingIntent);
mNotificationManager.cancel(NotificationConstants.SEND_SMS_MSG);
mNotificationManager.cancel(NotificationConstants.LAUNCHER_SERVICE_NOTE_ID);
LOGGER.info("Unregistered all receiver and stopped successfully.");
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
private BroadcastReceiver incomingCallReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
LOGGER.info("IncomingCallReceiver is received.");
telephony.listen(phoneListener, PhoneStateListener.LISTEN_CALL_STATE);
}
};
private PhoneStateListener phoneListener = new PhoneStateListener() {
@Override
public void onCallStateChanged(int state, String incomingNumber) {
super.onCallStateChanged(state, incomingNumber);
LOGGER.info("phoneListener state : " + state);
if (state == mLastState) {
return;
}
if (state == TelephonyManager.CALL_STATE_RINGING) {
mLastState = state;
sendSms(incomingNumber, user.getMessageBody(), false);
}
/*after finishing call state android returns CALL_STATE_IDLE*/
if (state == TelephonyManager.CALL_STATE_IDLE) {
LOGGER.info("incomingCall finished state = " + state);
mLastState = -1;
}
}
};
private void showSmsSendNotification(String message) {
boolean isEnabledNty = sharedPref.getBoolean(getResources().getString(R.string.is_enable_notifying), true);
LOGGER.info("Enable notification = " + isEnabledNty);
if (isEnabledNty) {
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(SendSmsService.this)
.setContentTitle(getResources().getText(R.string.app_name))
.setContentText(message)
.setSmallIcon(R.drawable.icon)
.setWhen(System.currentTimeMillis())
.setDefaults(Notification.DEFAULT_ALL);
Notification notification = notificationBuilder.build();
notification.defaults = Notification.DEFAULT_ALL;
notification.flags |= Notification.FLAG_ONLY_ALERT_ONCE;
mNotificationManager.notify(NotificationConstants.SEND_SMS_MSG, notification);
}
}
private void sendSms(String phoneNumber,String messageText, boolean isBinary) {
LOGGER.info("Sending sms to phoneNumber = " + phoneNumber);
if (phoneNumber == null || phoneNumber.isEmpty()) {
LOGGER.info("Failed: Couldn't sent message to number = " + phoneNumber);
mLastState = -1;
return;
}
if (ApplicationLoader.getApplication(this)
.getDaoSession()
.getBlackListDao()
.queryBuilder()
.where(BlackListDao.Properties.Number.eq(phoneNumber)).buildCount().count() != 0) {
LOGGER.info("Failed: Number in blockList = " + phoneNumber);
mLastState = -1;
return;
}
if (messageText == null || messageText.isEmpty()) {
LOGGER.info("Error: Sending msg content empty or null message = " + messageText);
mLastState = -1;
return;
}
List<SmsLog> smsList = ApplicationLoader.getApplication(this)
.getDaoSession()
.getSmsLogDao()
.queryBuilder()
.where(SmsLogDao.Properties.SentNumber.eq(phoneNumber))
.build()
.list();
int smsListSize = smsList.size();
if (smsListSize > 0) {
Resources res = getResources();
String period = prefs.getString("send_sms_period", res.getString(R.string.everytime));
String everytime = res.getString(R.string.everytime);
String onceADay = res.getString(R.string.once_day);
String onceAWeek = res.getString(R.string.once_week);
String onceAMonth = res.getString(R.string.once_month);
long settingsPeriod = 0;
if (period.equals(everytime)) {
settingsPeriod = ContextConstants.EVERYTIME_PERIOD;
} else if (period.equals(onceADay)) {
settingsPeriod = ContextConstants.ONE_DAY_PERIOD;
} else if (period.equals(onceAWeek)) {
settingsPeriod = ContextConstants.WEEK_DAY_PERIOD;
} else {
settingsPeriod = ContextConstants.MONTH_DAY_PERIOD;
}
try {
SimpleDateFormat format = new SimpleDateFormat("dd/MM/yyyy h:mm a");
Date date = format.parse(smsList.get(smsListSize - 1).getSentDate());
Calendar thatDay = Calendar.getInstance();
thatDay.setTime(date);
if (settingsPeriod != ContextConstants.EVERYTIME_PERIOD &&
!DateUtil.isMoreThanSelectedDays(thatDay, settingsPeriod)) {
LOGGER.info("SEND SMS PERIOD = " + settingsPeriod + " LAST SENT SMS date = " + date.toString());
return;
}
} catch (ParseException e) {
e.printStackTrace();
}
} else {
LOGGER.info("It is the first time to send sms size = " + smsList.size());
}
LOGGER.info("Sending probable success ");
recipientNumber = phoneNumber;
SmsManager smsManager = SmsManager.getDefault();
PendingIntent piSend = PendingIntent.getBroadcast(this, 0, new Intent(SMS_SENT), 0);
PendingIntent piDelivered = PendingIntent.getBroadcast(this, 0, new Intent(SMS_DELIVERED), 0);
messageText = messageText+" "+smsNumber;
smsNumber++;
ArrayList<String> parts = smsManager.divideMessage(messageText);
int partsCount = parts.size();
ArrayList<PendingIntent> sentIntents = new ArrayList<>(partsCount);
ArrayList<PendingIntent> deliveryIntents = new ArrayList<>(partsCount);
for (int i = 0; i < partsCount; i++) {
sentIntents.add(i, piSend);
deliveryIntents.add(i, piDelivered);
}
smsManager.sendMultipartTextMessage(phoneNumber, null, parts, sentIntents, deliveryIntents);
}
}
Я провел четыре дня. Я не могу найти ошибку и схожу с ума. Буду очень признателен за вашу помощь.
1 ответ
Эй, вам нужно зарегистрировать этот приемник в Manifest
и сделать это GLOBAL RECEIVER
так что он может работать, даже если приложение не на переднем плане.