Как получить отправленное / доставленное SMS-сообщение для отдельного SMS-сообщения после того, как пользователь оставил активность
У меня есть приложение, которое обеспечивает обмен SMS-сообщениями. Данные SMS хранятся в базе данных приложения. Приложение поддерживает отправку нескольким контактам, поэтому при отправке SMS я динамически регистрирую другой BroadcastReceiver для прослушивания, если каждое SMS было отправлено (я использую каждый номер телефона в строке, определяющий действие в IntentFilter).
Я пишу SMS в базу данных после получения подтверждения, что оно было отправлено.
Проблема заключается в том, что если пользователь покидает действие до того, как "отправленный" Намерение будет передано, BroadcastReceivers будут потеряны, и я больше не смогу перехватить "отправленное" Намерение, поэтому база данных не будет обновлена. Одним из решений, которое я выбрал, было внедрение onKeyDown(), чтобы пользователь не мог закрыть действие до того, как все Intents были переданы, но это решение работает только с кнопкой "Назад" - события кнопки "Домой" не могут быть перехвачены.
Вот мой код:
public void sendSMS(String[] phoneNumbers, String message){
final String currentMessage = message;
SmsManager sms = SmsManager.getDefault();
ArrayList<String> parts = sms.divideMessage(message);
for(int i=0; i<phoneNumbers.length; i++){
for(int j=0; j<parts.size(); j++){
BroadcastReceiver sent = new BroadcastReceiver(){
public void onReceive(Context arg0, Intent arg1) {
String[] arg = arg1.getAction().split(KEY_SMS_SENT);
String phoneNo = Utils.setSimpleFormatNumber(arg[1]);
String count = arg[0];
String parts = count.split(KEY_SMS_PART_NO)[0];
String partNo = count.split(KEY_SMS_PART_NO)[1];
sentReceivers.remove(this);
unregisterReceiver(this);
switch (getResultCode())
{
case Activity.RESULT_OK:
if(parts.equals(partNo)){
Toast.makeText(getBaseContext(), context.getString(R.string.sms_sent_message),
Toast.LENGTH_SHORT).show();
dbAdapter.createSentSMS(KEY_SMS_TYPE_SENT, phoneNo, currentMessage, Utils.getTimeStamp());
}
break;
case SmsManager.RESULT_ERROR_GENERIC_FAILURE:
if(parts.equals(partNo)){
Toast.makeText(getBaseContext(), context.getString(R.string.sms_generic_failure_message),
Toast.LENGTH_SHORT).show();
dbAdapter.createSentSMS(KEY_SMS_TYPE_FAILED, phoneNo, currentMessage, Utils.getTimeStamp());
}
break;
case SmsManager.RESULT_ERROR_NO_SERVICE:
if(parts.equals(partNo)){
Toast.makeText(getBaseContext(), context.getString(R.string.sms_no_service_message),
Toast.LENGTH_SHORT).show();
dbAdapter.createSentSMS(KEY_SMS_TYPE_FAILED, phoneNo, currentMessage, Utils.getTimeStamp());
}
break;
case SmsManager.RESULT_ERROR_NULL_PDU:
if(parts.equals(partNo)){
Toast.makeText(getBaseContext(), context.getString(R.string.sms_null_pdu_message),
Toast.LENGTH_SHORT).show();
dbAdapter.createSentSMS(KEY_SMS_TYPE_FAILED, phoneNo, currentMessage, Utils.getTimeStamp());
}
break;
case SmsManager.RESULT_ERROR_RADIO_OFF:
if(parts.equals(partNo)){
Toast.makeText(getBaseContext(), context.getString(R.string.sms_radio_off_message),
Toast.LENGTH_SHORT).show();
dbAdapter.createSentSMS(KEY_SMS_TYPE_FAILED, phoneNo, currentMessage, Utils.getTimeStamp());
}
break;
}
}
};
registerReceiver(sent, new IntentFilter(String.valueOf(parts.size()-1) + KEY_SMS_PART_NO + String.valueOf(j) + KEY_SMS_SENT + phoneNumbers[i]));
//I add the BroadcastReceiver-s to a Vector in order to keep track of them
sentReceivers.add(sent);
}
}
//I use an IntentService to do the actual sending
Intent intent = new Intent(context, SMSSendService.class);
intent.putExtra(KEY_SELECTED_PHONE_NUMBERS, phoneNumbers);
intent.putExtra(KEY_SMS_MESSAGE, message);
intent.putExtra(KEY_ACTION, KEY_REQUEST_SEND_SMS);
startService(intent);
}
и это код в IntentService:
public void sendSMS(String phoneNo, String message, SmsManager sms)
{
ArrayList<String> parts = sms.divideMessage(message);
ArrayList<PendingIntent> sentPIs = new ArrayList<PendingIntent>();
for(int i=0; i<parts.size(); i++){
sentPIs.add(PendingIntent.getBroadcast(context, 0, new Intent(String.valueOf(parts.size()-1) + KEY_SMS_PART_NO + String.valueOf(i) + KEY_SMS_SENT + phoneNo), 0));
}
sms.sendMultipartTextMessage(phoneNo, null, parts, sentPIs, deliveredPIs);
}
1 ответ
Вы могли бы создать Службу (не IntentService), в которой вы могли бы создать экземпляр всех BroadcastReceivers и выполнить логику отправки SMS. Таким образом, ваши SMS-сообщения будут отправлены в фоновом режиме, и вы сможете перехватить отправленные / доставленные трансляции.