it-roy-ru.com

Android - SMS Приемник вещания

Я пытался заставить эту программу работать, но пока безуспешно. Я не могу найти, где я делаю неправильно. Я не уверен, что что-то не так с кодом или отладкой.

Я пытаюсь получить уведомление о получении нового SMS.

Вот моя программа:

package Technicaljar.SMSBroadcastReceiver;

import Android.content.BroadcastReceiver;
import Android.content.Context;
import Android.content.Intent;
import Android.os.Bundle;
import Android.telephony.SmsMessage;
import Android.util.Log;

public class SMSBroadcastReceiver extends BroadcastReceiver {

        private static final String SMS_RECEIVED = "Android.provider.Telephony.SMS_RECEIVED";
        private static final String TAG = "SMSBroadcastReceiver";

        @Override
        public void onReceive(Context context, Intent intent) {
             Log.i(TAG, "Intent recieved: " + intent.getAction());

                if (intent.getAction() == SMS_RECEIVED) {
                    Bundle bundle = intent.getExtras();
                    if (bundle != null) {
                        Object[] pdus = (Object[])bundle.get("pdus");
                        final SmsMessage[] messages = new SmsMessage[pdus.length];
                        for (int i = 0; i < pdus.length; i++) {
                            messages[i] = SmsMessage.createFromPdu((byte[])pdus[i]);
                        }
                        if (messages.length > -1) {
                            Log.i(TAG, "Message recieved: " + messages[0].getMessageBody());
                        }
                    }
                }
           }
    }

И файл манифеста:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:Android="http://schemas.Android.com/apk/res/Android"
      package="Technicaljar.SMSBroadcastReceiver"
      Android:versionCode="1"
      Android:versionName="1.0">
    <application Android:icon="@drawable/icon" Android:label="@string/app_name" Android:debuggable="true" >
        <receiver Android:name=".SMSBroadcastReceiver">
            <intent-filter>
                <action Android:name="Android.provider.telephony.SMS_RECEIVED"></action>
            </intent-filter>
        </receiver>

    </application>
    <uses-sdk Android:minSdkVersion="7" />
    <uses-permission Android:name="Android.permission.INTERNET"></uses-permission>
    <uses-permission Android:name="Android.permission.RECEIVE_SMS"></uses-permission>
</manifest> 

Я отправляю SMS через Telnet, и не вижу сообщений, полученных Intent, в logcat. Вот мой logcat со времени установки.

D/AndroidRuntime(  478): 
D/AndroidRuntime(  478): >>>>>>>>>>>>>> AndroidRuntime START <<<<<<<<<<<<<<
D/AndroidRuntime(  478): CheckJNI is ON
D/AndroidRuntime(  478): --- registering native functions ---
D/AndroidRuntime(  478): Shutting down VM
D/dalvikvm(  478): Debugger has detached; object registry had 1 entries
I/AndroidRuntime(  478): NOTE: attach of thread 'Binder Thread #3' failed
D/Mms:app (  220): getSmsNewMessageNotificationInfo: count=14, first addr=12345, thread_id=4
D/dalvikvm(  151): GC_EXPLICIT freed 391 objects / 22552 bytes in 65ms
D/dalvikvm(  220): GC_EXPLICIT freed 926 objects / 44840 bytes in 73ms

Таким образом, SMS, кажется, получен эмулятором, но похоже, что нет намерений запускать .. Что я здесь не так делаю? После установки мне нужно как-то «запустить» этот ресивер? Потому что при установке я получаю

 [2010-11-07 21:24:41 - SMSBroadcastReceiver] No Launcher activity found!
[2010-11-07 21:24:41 - SMSBroadcastReceiver] The launch will only sync the application package on the device!

Поэтому мне интересно, если что-то здесь не так.

51
madu

Android.provider.Telephony.SMS_RECEIVED имеет прописную букву T, а ваша в манифесте - нет.

Помните, что это действие Intent не задокументировано.

70
CommonsWare

Я попробовал ваш код и обнаружил, что он не работает.

Я должен был изменить 

if (intent.getAction() == SMS_RECEIVED) {

в 

if (intent.getAction().equals(SMS_RECEIVED)) {

Теперь это работает. Это просто проблема с проверкой равенства Java.

17
anshul

Также обратите внимание, что приложение Hangouts в настоящее время блокирует мой BroadcastReceiver от получения сообщений SMS. Мне пришлось отключить функцию SMS в приложении Hangouts (Настройки-> SMS-> Включить SMS), прежде чем мой SMS BroadcastReceived начал работать.

Правка: Похоже, что некоторые приложения будут abortBroadcast () по намерению, что будет препятствовать другим намерениям получить намерение. Решением является увеличение атрибута Android:priority в теге intent-filter:

<receiver Android:name="com.company.application.SMSBroadcastReceiver" >
  <intent-filter Android:priority="500">
    <action Android:name="Android.provider.Telephony.SMS_RECEIVED" />
  </intent-filter>
</receiver>

Подробнее см. Здесь: Включение поддержки SMS в Hangouts 2.0 отключает BroadcastReceiver SMS_RECEIVED в моем приложении

5
TheIT

intent.getAction().equals(SMS_RECEIVED)

Я попробовал это успешно.

3
Cathy

Наткнулся сегодня на это. Для тех, кто сегодня кодирует SMS получатель, используйте этот код вместо устаревшего в OP:

SmsMessage[] msgs = Telephony.Sms.Intents.getMessagesFromIntent(intent);
SmsMessage smsMessage = msgs[0];
2
kjdion84

Получатель широковещательной рассылки должен указать Android:exported="true" для приема трансляций, созданных вне вашего собственного приложения. Мой широковещательный приемник определяется в манифесте следующим образом:

<receiver
    Android:name=".IncomingSmsBroadcastReceiver"
    Android:enabled="true"
    Android:exported="true" >
    <intent-filter>
        <action Android:name="Android.provider.Telephony.SMS_RECEIVED" />
    </intent-filter>
</receiver>

Как отмечено ниже, exported = "true" является значением по умолчанию, поэтому вы можете опустить эту строку. Я оставил это так, чтобы комментарии обсуждения имели смысл.

2
Phil

Для Android 19+ вы можете получить его в Telephony.Sms.Intents.SMS_RECEIVED_ACTION). В классе Intents есть и другие, которые стоит посмотреть At

1
peter

Обновленный код:

 private class SMSReceiver extends BroadcastReceiver {
                private Bundle bundle;
                private SmsMessage currentSMS;
                private String message;

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

                    if (intent.getAction().equals("Android.provider.Telephony.SMS_RECEIVED")) {
                        bundle = intent.getExtras();
                        if (bundle != null) {
                            Object[] pdu_Objects = (Object[]) bundle.get("pdus");
                            if (pdu_Objects != null) {

                                for (Object aObject : pdu_Objects) {

                                    currentSMS = getIncomingMessage(aObject, bundle);

                                    String senderNo = currentSMS.getDisplayOriginatingAddress();

                                    message = currentSMS.getDisplayMessageBody();
                                    Toast.makeText(OtpActivity.this, "senderNum: " + senderNo + " :\n message: " + message, Toast.LENGTH_LONG).show();
                                }
                                this.abortBroadcast();
                                // End of loop
                            }
                        }
                    } // bundle null
                }
            }

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

старый код был:

Object [] pdus = (Object[]) myBundle.get("pdus");

    messages = new SmsMessage[pdus.length];

    for (int i = 0; i < messages.length; i++)
    {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            String format = myBundle.getString("format");
            messages[i] = SmsMessage.createFromPdu((byte[]) pdus[i], format);
        }
        else {
            messages[i] = SmsMessage.createFromPdu((byte[]) pdus[i]);
        }
        strMessage += "SMS From: " + messages[i].getOriginatingAddress();
        strMessage += " : ";
        strMessage += messages[i].getMessageBody();
        strMessage += "\n";
    } 

Простой синтаксис кода:

private class SMSReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent.getAction().equals(Telephony.Sms.Intents.SMS_RECEIVED_ACTION)) {
            SmsMessage[] smsMessages = Telephony.Sms.Intents.getMessagesFromIntent(intent);
            for (SmsMessage message : smsMessages) {
                // Do whatever you want to do with SMS.
            }
        }
    }
}
1
Pradeep Sheoran

Я недавно столкнулся с такой проблемой. Хотя код был правильным, я не включил разрешения в настройках приложения. Таким образом, все разрешения не установлены по умолчанию на эмуляторах, поэтому вы должны сделать это самостоятельно. 

1
Dracontis

Android.provider.telephony.SMS_RECEIVED неверен, потому что телефония является классом и должна быть заглавной, как в Android.provider.Telephony.SMS_RECEIVED

0
Cool Java guy מוחמד