Android - Как добавить последние смс в список разговоров во время активности? - PullRequest
0 голосов
/ 26 июня 2011

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

Я пытался использовать ContentObserver.Как только произошли некоторые изменения в базе данных SMS, я попытался перезагрузить ListView с новым набором данных, но он не работает.Каждый раз, чтобы увидеть новое SMS (отправлено на получение), я должен перезапустить приложение, чтобы список мог быть перезагружен.Я хочу, чтобы это новое сообщение отображалось в окне чата, как только я отправляю или получаю их так же, как мы общаемся через gtalk, msn или любое SMS-приложение.

Вот мой код: -

public class ThreadData extends ListActivity
{
private static final Uri SMS_URI = Uri.parse("content://sms");
HashMap<Long, BodyType> bodies = new HashMap<Long, BodyType>();
private String name, number;
private ListView listView;
private Activity activity;
ThreadDataAdapter adapter;
Handler handler;
ArrayList<BodyType> items;
private Context context;
private static final String PHONE_NUMBER_SEPARATORS = " ()-./";
static HashMap<String, ContactInfo.ContactDetail> info = new HashMap<String, ContactInfo.ContactDetail>();

public void onCreate(Bundle bundle)
{
    super.onCreate(bundle);
    setContentView(R.layout.chats);
    Bundle extras = getIntent().getExtras();
    activity = this;
    context = this;
    listView = getListView();
    if(extras != null)
    {
        number = extras.getString("address");
        ContactInfo.ContactDetail nameInfo = getContactsDetailWithNumber(number);
        if(nameInfo != null)
        {
            name = nameInfo.name;
        }
        else
        {
            name = number;
        }
    }
    TextView person = (TextView) findViewById(R.id.chat_person);
    person.setText(name);
    ImageButton callPerson = (ImageButton) findViewById(R.id.call_person);
    callPerson.setOnClickListener(new OnClickListener() {
        public void onClick(View view) {
            popUpDialerAlert();
        }
    });

    buildMessageList();
    items = sortBodies(bodies);
    adapter = new ThreadDataAdapter(this, items);
    listView.setAdapter(adapter);
    listView.setStackFromBottom(true);

    getContentResolver().registerContentObserver(SMS_URI, true, new MyContentObserver(handler));
    //Intent intent = new Intent(this, RBSMSService.class);
    //startService(intent);
}

public void loadListView()
{
    items.clear();
    items = sortBodies(bodies);
    adapter = new ThreadDataAdapter(this, items);
    listView.setAdapter(adapter);
    listView.setStackFromBottom(true);
}

public void buildMessageList()
{
    Cursor cursor = getContentResolver().query(SMS_URI, null, null, null, "date ASC");
    startManagingCursor(cursor);
    while (cursor.moveToNext())
    {
        BodyType bodyInfo = new BodyType();
        String address = cursor.getString(cursor.getColumnIndex("address"));
        bodyInfo.body = cursor.getString(cursor.getColumnIndexOrThrow("body"));
        Long date = cursor.getLong(cursor.getColumnIndexOrThrow("date"));
        bodyInfo.date = date;
        String type =  cursor.getString(cursor.getColumnIndexOrThrow("type"));
        if(type.equals("1"))
        {
            bodyInfo.type = "received";
        }
        else if(type.equals("2"))
        {
            bodyInfo.type = "sent";
        }
        else if(type.equals("3"))
        {
            bodyInfo.type = "draft";
        }

        String number = filterPhoneNumber(address);
        ContactInfo.ContactDetail nameInfo = getContactsDetailWithNumber(number);
        String personName = number;
        if(nameInfo != null)
        {
            personName = nameInfo.name;
        }
        else
        {
            personName = number;
        }
        if(personName.equals(name))
       {
           bodies.put(date, bodyInfo);
       }

    }
}

public void popUpDialerAlert()
{
    AlertDialog.Builder dialerAlert = new AlertDialog.Builder(this);
    dialerAlert.setTitle("Confirm");
    dialerAlert.setMessage("Call" + " " + name + "?");
    dialerAlert.setPositiveButton("Yes", new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialogInterface, int i) {
            Uri uri = Uri.parse("tel:" + number);
            Intent intent = new Intent((Intent.ACTION_CALL));
            intent.setData(uri);
            context.startActivity(intent);
        }
    });
    dialerAlert.setNegativeButton("No", new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialogInterface, int i) {

        }
    });
    dialerAlert.show();
}

public ArrayList<BodyType> sortBodies(HashMap<Long, BodyType> bodies)
{
    ArrayList<Long> dates = new ArrayList<Long>(bodies.keySet());
    Collections.sort(dates);
    ArrayList<BodyType> items = new ArrayList<BodyType>();
    for(Long date : dates)
    {
        for(Long key : bodies.keySet())
        {
            if(date.equals(key))
            {

                items.add(bodies.get(key));
            }
        }
    }
    return items;
}

public void printBodies(ArrayList<BodyType> items)
{
    for(BodyType bodyType : items)
    {
        log(bodyType.date.toString());
        log(bodyType.body);
        log(bodyType.type);
    }
}

static class BodyType
{
    public String type;
    public String body;
    public Long date;

}


public static ContactInfo.ContactDetail getContactsDetailWithNumber(String number)
{
    if(info.containsKey(number))
    {
        return info.get(number);
    }
    return null;
}

public static String filterPhoneNumber(String phoneNumber)
{
    if (phoneNumber == null)
    {
        return null;
    }

    int length = phoneNumber.length();
    StringBuilder builder = new StringBuilder(length);

    for (int i = 0; i < length; i++)
    {
        char character = phoneNumber.charAt(i);

        if (PHONE_NUMBER_SEPARATORS.indexOf(character) == -1)
        {
           builder.append(character);
        }
    }
    return builder.toString();
}

public class MyContentObserver extends ContentObserver
{

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

    @Override
    public void onChange(boolean selfChange)
    {
        bodies.clear();
        buildMessageList();
        items = sortBodies(bodies);
        runOnUiThread(new Runnable() {
            public void run() {
                loadListView();
            }
        });
        super.onChange(selfChange);
    }
}

public void log(String msg)
{
    Log.e("ThreadData", msg);
}
}

Пожалуйста, помогите!

Спасибо

Ответы [ 2 ]

1 голос
/ 26 июня 2011

Вам необходимо сохранить ссылку на свой список (предметы).

Когда провайдер контента вызывает OnChange, очистите тот же список (не создавайте новый объект, переработайте свой список) и снова добавьте новые письма. Поскольку этот список привязан к вашему адаптеру, он обновит ваш список при вызове notifyDataChanged.

Вы должны переработать тот же список.

С уважением Stéphane PS: пожалуйста, обратите внимание, что вы могли бы сохранить свою старую ветку, чтобы задать тот же вопрос снова.

0 голосов
/ 26 июня 2011

Итак, чтобы получить список для прокрутки до конца при добавлении новых элементов, вы должны установить его в « transcriptMode ». Однако тот факт, что вам нужно перезапустить ваше приложение для отображения нового контента, говорит о том, что что-то еще не так.

Изменить: Стефан прав, не звоните buildMessageList(); снова, когда вы получаете уведомление onChange, просто обновите данные в адаптере и позвоните notifyDatasetChanged()

...