Подтвердить что ты не робот

Слушайте исходящие SMS или отправленные сообщения на Android

Я разрабатываю приложение, которое будет хранить все входящие и исходящие sms в текстовом файле на SD-карте.

Я могу слушать входящие сообщения с помощью широковещательного приемника. Мне очень сложно слушать исходящие SMS.

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

Как это можно сделать?

4b9b3361

Ответ 1

В принципе, вам нужно зарегистрировать наблюдателя контента... что-то вроде этого:

ContentResolver contentResolver = context.getContentResolver();
contentResolver.registerContentObserver(Uri.parse("content://sms/out"),true, yourObserver);

yourObserver - это объект (new YourObserver(new Handler())), который может выглядеть следующим образом:

class YourObserver extends ContentObserver {

    public YourObserver(Handler handler) {
        super(handler);
    }

    @Override
    public void onChange(boolean selfChange) {
        super.onChange(selfChange);
        // save the message to the SD card here
    }
}

Итак, как именно вы получаете содержание SMS? Вы должны использовать Cursor:

// save the message to the SD card here
Uri uriSMSURI = Uri.parse("content://sms/out");
Cursor cur = this.getContentResolver().query(uriSMSURI, null, null, null, null);
 // this will make it point to the first record, which is the last SMS sent
cur.moveToNext();
String content = cur.getString(cur.getColumnIndex("body"));
// use cur.getColumnNames() to get a list of all available columns...
// each field that compounds a SMS is represented by a column (phone number, status, etc.)
// then just save all data you want to the SDcard :)

Ответ 2

Это мой подход к решению этого

  • Создайте службу, вызванную из другой активности.
  • Создайте в нем наблюдателя содержимого

     @Override
     public int onStartCommand(Intent intent, int flag, int startId) {
     MyObserver myObserver = new MyObserver(new Handler());
     ContentResolver contentResolver = this.getApplicationContext().getContentResolver();
     contentResolver.registerContentObserver(Uri.parse("content://sms/sent"), true, myObserver);
     return START_STICKY;
     }
    
  • Создайте класс наблюдателя

    class MyObserver extends ContentObserver {
    
    public MyObserver(Handler handler) {
        super(handler);
    }
    
    @Override
    public void onChange(boolean selfChange) {
        super.onChange(selfChange);
        Uri uriSMSURI = Uri.parse("content://sms/sent");
        Cursor cur = getContentResolver().query(uriSMSURI, null, null, null, null);
        cur.moveToNext();
        String content = cur.getString(cur.getColumnIndex("body"));
        String smsNumber = cur.getString(cur.getColumnIndex("address"));
        if (smsNumber == null || smsNumber.length() <= 0) {
            smsNumber = "Unknown";
        }
        cur.close();
    
        if(smsChecker( "OutgoingSMS to " + smsNumber + ": " + content)) {
            //save data into database/sd card here
        }
    }
    }
    
  • Я добавил метод smsChecker(), чтобы проверить, совпадает ли новое сообщение с последним сообщением

    public boolean smsChecker(String sms) {
    boolean flagSMS = true;
    
    if (sms.equals(lastSMS)) {
        flagSMS = false;
    }
    else {
        lastSMS = sms;
    }
    //if flagSMS = true, those 2 messages are different
    return flagSMS;
    }
    

Если я не ошибаюсь, мы используем "content://sms/sent", если ТОЛЬКО хотим проверить все отправленные сообщения "content://sms/out", если мы ТОЛЬКО хотим проверить все сообщения внутри outbox, и "content://sms", если мы хотим проверить ВСЕ сообщения.

Ответ 3

Это моя версия, которая была проверена в Android 6.0 +

class smsObserver extends ContentObserver {

    private String lastSmsId;

    public smsObserver(Handler handler) {
        super(handler);
    }

    @Override
    public void onChange(boolean selfChange) {
        super.onChange(selfChange);
        Uri uriSMSURI = Uri.parse("content://sms/sent");
        Cursor cur = getContentResolver().query(uriSMSURI, null, null, null, null);
        cur.moveToNext();
        String id = cur.getString(cur.getColumnIndex("_id"));
        if (smsChecker(id)) {
            String address = cur.getString(cur.getColumnIndex("address"));
            // Optional: Check for a specific sender
            if (address.equals(phoneNumber)) {
                String message = cur.getString(cur.getColumnIndex("body"));
                // Use message content for desired functionality
            }
        }
    }

    // Prevent duplicate results without overlooking legitimate duplicates
    public boolean smsChecker(String smsId) {
        boolean flagSMS = true;

        if (smsId.equals(lastSmsId)) {
            flagSMS = false;
        }
        else {
            lastSmsId = smsId;
        }

        return flagSMS;
    }
}

Поместите этот код, где наблюдатель должен быть включен

ContentResolver contentResolver = getContentResolver();
contentResolver.registerContentObserver(Uri.parse("content://sms"), true, new smsObserver(new Handler()));

Предполагается, что вы используете операцию. Помните, что вам понадобится ссылка Context для вызова getContentResolver() из службы или получателя.

Ответ 4

Я видел, что пошло не так. его на линии:

 contentResolver.registerContentObserver(Uri.parse("content://sms/sent"), true, _myObserver);

вы должны удалить '/sent' и просто написать 'content://sms' его уже указано в ContentObserver для просмотра отправленных смс.