conversations-classic/src/de/gultsch/chat/services/XmppConnectionService.java

271 lines
8.5 KiB
Java
Raw Normal View History

2014-01-24 01:04:05 +00:00
package de.gultsch.chat.services;
import java.io.IOException;
import java.util.ArrayList;
2014-02-01 14:07:20 +00:00
import java.util.Hashtable;
import java.util.List;
import de.gultsch.chat.entities.Account;
import de.gultsch.chat.entities.Contact;
import de.gultsch.chat.entities.Conversation;
import de.gultsch.chat.entities.Message;
import de.gultsch.chat.persistance.DatabaseBackend;
2014-02-03 17:38:47 +00:00
import de.gultsch.chat.ui.ConversationActivity;
2014-02-01 14:07:20 +00:00
import de.gultsch.chat.ui.OnConversationListChangedListener;
import de.gultsch.chat.ui.OnRosterFetchedListener;
2014-02-03 17:38:47 +00:00
import de.gultsch.chat.utils.UIHelper;
import de.gultsch.chat.xml.Element;
import de.gultsch.chat.xmpp.IqPacket;
2014-02-01 14:07:20 +00:00
import de.gultsch.chat.xmpp.MessagePacket;
import de.gultsch.chat.xmpp.OnIqPacketReceived;
2014-02-01 14:07:20 +00:00
import de.gultsch.chat.xmpp.OnMessagePacketReceived;
import de.gultsch.chat.xmpp.XmppConnection;
2014-02-03 17:38:47 +00:00
import android.R;
import android.R.dimen;
import android.app.NotificationManager;
import android.app.PendingIntent;
2014-01-24 01:04:05 +00:00
import android.app.Service;
import android.content.Context;
2014-01-24 01:04:05 +00:00
import android.content.Intent;
2014-02-03 17:38:47 +00:00
import android.content.res.Resources;
2014-01-24 01:04:05 +00:00
import android.os.Binder;
import android.os.IBinder;
import android.os.PowerManager;
2014-02-03 17:38:47 +00:00
import android.support.v4.app.NotificationCompat;
import android.support.v4.app.TaskStackBuilder;
import android.util.Log;
2014-01-24 01:04:05 +00:00
public class XmppConnectionService extends Service {
2014-02-03 17:38:47 +00:00
protected static final String LOGTAG = "xmppService";
protected DatabaseBackend databaseBackend;
2014-02-03 17:38:47 +00:00
public long startDate;
2014-02-03 17:38:47 +00:00
private List<Account> accounts;
2014-02-01 14:07:20 +00:00
private List<Conversation> conversations = null;
2014-02-03 17:38:47 +00:00
private Hashtable<Account, XmppConnection> connections = new Hashtable<Account, XmppConnection>();
2014-01-24 01:04:05 +00:00
2014-02-01 14:07:20 +00:00
private OnConversationListChangedListener convChangedListener = null;
2014-02-03 17:38:47 +00:00
private final IBinder mBinder = new XmppConnectionBinder();
2014-02-01 14:07:20 +00:00
private OnMessagePacketReceived messageListener = new OnMessagePacketReceived() {
2014-02-03 17:38:47 +00:00
2014-02-01 14:07:20 +00:00
@Override
2014-02-03 17:38:47 +00:00
public void onMessagePacketReceived(Account account,
MessagePacket packet) {
if (packet.getType() == MessagePacket.TYPE_CHAT) {
2014-02-02 16:53:34 +00:00
String fullJid = packet.getFrom();
String jid = fullJid.split("/")[0];
String name = jid.split("@")[0];
2014-02-03 17:38:47 +00:00
Contact contact = new Contact(account, name, jid, null); // dummy
// contact
Conversation conversation = findOrCreateConversation(account,
contact);
Message message = new Message(conversation, fullJid,
packet.getBody(), Message.ENCRYPTION_NONE,
Message.STATUS_RECIEVED);
2014-02-02 16:53:34 +00:00
conversation.getMessages().add(message);
databaseBackend.createMessage(message);
if (convChangedListener != null) {
convChangedListener.onConversationListChanged();
2014-02-03 17:38:47 +00:00
} else {
NotificationManager mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
mNotificationManager.notify(2342, UIHelper
.getUnreadMessageNotification(
getApplicationContext(), conversation));
2014-02-02 16:53:34 +00:00
}
2014-02-01 14:07:20 +00:00
}
}
};
2014-01-24 01:04:05 +00:00
2014-02-03 17:38:47 +00:00
public class XmppConnectionBinder extends Binder {
public XmppConnectionService getService() {
return XmppConnectionService.this;
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
for (Account account : accounts) {
if (!connections.containsKey(account)) {
XmppConnection connection = new XmppConnection(account, pm);
connection
.setOnMessagePacketReceivedListener(this.messageListener);
Thread thread = new Thread(connection);
thread.start();
this.connections.put(account, connection);
}
}
return START_STICKY;
}
@Override
public void onCreate() {
databaseBackend = DatabaseBackend.getInstance(getApplicationContext());
this.accounts = databaseBackend.getAccounts();
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
public void sendMessage(final Account account, final Message message) {
new Thread() {
@Override
public void run() {
Log.d(LOGTAG, "sending message for " + account.getJid()
+ " to: " + message.getCounterpart());
databaseBackend.createMessage(message);
MessagePacket packet = new MessagePacket();
packet.setType(MessagePacket.TYPE_CHAT);
packet.setTo(message.getCounterpart());
packet.setFrom(account.getJid());
packet.setBody(message.getBody());
try {
connections.get(account).sendMessagePacket(packet);
message.setStatus(Message.STATUS_SEND);
databaseBackend.updateMessage(message);
} catch (IOException e) {
2014-02-03 17:38:47 +00:00
Log.d(LOGTAG,
"io exception during send. message is in database. will try again later");
}
2014-02-03 17:38:47 +00:00
}
}.start();
}
public void getRoster(final Account account,
final OnRosterFetchedListener listener) {
new Thread() {
@Override
public void run() {
IqPacket iqPacket = new IqPacket(IqPacket.TYPE_GET);
Element query = new Element("query");
query.setAttribute("xmlns", "jabber:iq:roster");
query.setAttribute("ver", "");
iqPacket.addChild(query);
try {
connections.get(account).sendIqPacket(iqPacket,
new OnIqPacketReceived() {
@Override
public void onIqPacketReceived(Account account,
IqPacket packet) {
Element roster = packet.findChild("query");
List<Contact> contacts = new ArrayList<Contact>();
for (Element item : roster.getChildren()) {
String name = item.getAttribute("name");
String jid = item.getAttribute("jid");
if (name == null) {
name = jid.split("@")[0];
}
Contact contact = new Contact(account,
name, jid, null);
contacts.add(contact);
}
if (listener != null) {
listener.onRosterFetched(contacts);
}
}
});
} catch (IOException e) {
Log.d(LOGTAG, "io error during roster fetch");
}
}
}.start();
}
public void addConversation(Conversation conversation) {
databaseBackend.createConversation(conversation);
}
public List<Conversation> getConversations() {
if (this.conversations == null) {
Hashtable<String, Account> accountLookupTable = new Hashtable<String, Account>();
for (Account account : this.accounts) {
accountLookupTable.put(account.getUuid(), account);
}
this.conversations = databaseBackend
.getConversations(Conversation.STATUS_AVAILABLE);
for (Conversation conv : this.conversations) {
conv.setAccount(accountLookupTable.get(conv.getAccountUuid()));
}
}
return this.conversations;
}
public List<Account> getAccounts() {
return this.accounts;
}
public List<Message> getMessages(Conversation conversation) {
return databaseBackend.getMessages(conversation, 100);
}
public Conversation findOrCreateConversation(Account account,
Contact contact) {
// Log.d(LOGTAG,"was asked to find conversation for "+contact.getJid());
for (Conversation conv : this.getConversations()) {
if ((conv.getAccount().equals(account))
&& (conv.getContactJid().equals(contact.getJid()))) {
// Log.d(LOGTAG,"found one in memory");
return conv;
}
}
Conversation conversation = databaseBackend.findConversation(account,
contact.getJid());
if (conversation != null) {
Log.d("gultsch", "found one. unarchive it");
conversation.setStatus(Conversation.STATUS_AVAILABLE);
conversation.setAccount(account);
this.databaseBackend.updateConversation(conversation);
} else {
Log.d(LOGTAG, "didnt find one in archive. create new one");
conversation = new Conversation(contact.getDisplayName(),
contact.getProfilePhoto(), account, contact.getJid());
this.databaseBackend.createConversation(conversation);
}
this.conversations.add(conversation);
if (this.convChangedListener != null) {
this.convChangedListener.onConversationListChanged();
}
return conversation;
}
public void archiveConversation(Conversation conversation) {
this.databaseBackend.updateConversation(conversation);
this.conversations.remove(conversation);
if (this.convChangedListener != null) {
this.convChangedListener.onConversationListChanged();
}
}
public int getConversationCount() {
return this.databaseBackend.getConversationCount();
}
2014-01-28 18:21:54 +00:00
public void createAccount(Account account) {
databaseBackend.createAccount(account);
}
2014-02-03 17:38:47 +00:00
2014-01-28 18:21:54 +00:00
public void updateAccount(Account account) {
databaseBackend.updateAccount(account);
}
public void deleteAccount(Account account) {
databaseBackend.deleteAccount(account);
}
2014-02-03 17:38:47 +00:00
public void setOnConversationListChangedListener(
OnConversationListChangedListener listener) {
2014-02-01 14:07:20 +00:00
this.convChangedListener = listener;
}
2014-02-03 17:38:47 +00:00
2014-02-01 14:07:20 +00:00
public void removeOnConversationListChangedListener() {
this.convChangedListener = null;
}
2014-01-24 01:04:05 +00:00
}