experimental message merging

This commit is contained in:
iNPUTmice 2014-08-31 14:29:12 +02:00
parent 9eafb10086
commit 8f8d4e320d
3 changed files with 112 additions and 44 deletions

View File

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="wrap_content">
</RelativeLayout>

View File

@ -7,7 +7,7 @@ import android.content.Context;
import android.database.Cursor; import android.database.Cursor;
public class Message extends AbstractEntity { public class Message extends AbstractEntity {
public static final String TABLENAME = "messages"; public static final String TABLENAME = "messages";
public static final int STATUS_RECEPTION_FAILED = -3; public static final int STATUS_RECEPTION_FAILED = -3;
@ -28,7 +28,7 @@ public class Message extends AbstractEntity {
public static final int ENCRYPTION_OTR = 2; public static final int ENCRYPTION_OTR = 2;
public static final int ENCRYPTION_DECRYPTED = 3; public static final int ENCRYPTION_DECRYPTED = 3;
public static final int ENCRYPTION_DECRYPTION_FAILED = 4; public static final int ENCRYPTION_DECRYPTION_FAILED = 4;
public static final int TYPE_TEXT = 0; public static final int TYPE_TEXT = 0;
public static final int TYPE_IMAGE = 1; public static final int TYPE_IMAGE = 1;
public static final int TYPE_AUDIO = 2; public static final int TYPE_AUDIO = 2;
@ -58,27 +58,32 @@ public class Message extends AbstractEntity {
protected String remoteMsgId = null; protected String remoteMsgId = null;
protected transient Conversation conversation = null; protected transient Conversation conversation = null;
protected transient JingleConnection jingleConnection = null; protected transient JingleConnection jingleConnection = null;
private Message() { private Message() {
} }
public Message(Conversation conversation, String body, int encryption) { public Message(Conversation conversation, String body, int encryption) {
this(java.util.UUID.randomUUID().toString(), conversation.getUuid(), this(java.util.UUID.randomUUID().toString(), conversation.getUuid(),
conversation.getContactJid(), null, body, System.currentTimeMillis(), encryption, conversation.getContactJid(), null, body, System
Message.STATUS_UNSEND,TYPE_TEXT,null); .currentTimeMillis(), encryption,
Message.STATUS_UNSEND, TYPE_TEXT, null);
this.conversation = conversation; this.conversation = conversation;
} }
public Message(Conversation conversation, String counterpart, String body, int encryption, int status) { public Message(Conversation conversation, String counterpart, String body,
this(java.util.UUID.randomUUID().toString(), conversation.getUuid(),counterpart, null, body, System.currentTimeMillis(), encryption,status,TYPE_TEXT,null); int encryption, int status) {
this(java.util.UUID.randomUUID().toString(), conversation.getUuid(),
counterpart, null, body, System.currentTimeMillis(),
encryption, status, TYPE_TEXT, null);
this.conversation = conversation; this.conversation = conversation;
} }
public Message(String uuid, String conversationUUid, String counterpart, String trueCounterpart, public Message(String uuid, String conversationUUid, String counterpart,
String body, long timeSent, int encryption, int status, int type, String remoteMsgId) { String trueCounterpart, String body, long timeSent, int encryption,
int status, int type, String remoteMsgId) {
this.uuid = uuid; this.uuid = uuid;
this.conversationUuid = conversationUUid; this.conversationUuid = conversationUUid;
this.counterpart = counterpart; this.counterpart = counterpart;
@ -97,20 +102,20 @@ public class Message extends AbstractEntity {
values.put(UUID, uuid); values.put(UUID, uuid);
values.put(CONVERSATION, conversationUuid); values.put(CONVERSATION, conversationUuid);
values.put(COUNTERPART, counterpart); values.put(COUNTERPART, counterpart);
values.put(TRUE_COUNTERPART,trueCounterpart); values.put(TRUE_COUNTERPART, trueCounterpart);
values.put(BODY, body); values.put(BODY, body);
values.put(TIME_SENT, timeSent); values.put(TIME_SENT, timeSent);
values.put(ENCRYPTION, encryption); values.put(ENCRYPTION, encryption);
values.put(STATUS, status); values.put(STATUS, status);
values.put(TYPE, type); values.put(TYPE, type);
values.put(REMOTE_MSG_ID,remoteMsgId); values.put(REMOTE_MSG_ID, remoteMsgId);
return values; return values;
} }
public String getConversationUuid() { public String getConversationUuid() {
return conversationUuid; return conversationUuid;
} }
public Conversation getConversation() { public Conversation getConversation() {
return this.conversation; return this.conversation;
} }
@ -118,7 +123,7 @@ public class Message extends AbstractEntity {
public String getCounterpart() { public String getCounterpart() {
return counterpart; return counterpart;
} }
public Contact getContact() { public Contact getContact() {
if (this.conversation.getMode() == Conversation.MODE_SINGLE) { if (this.conversation.getMode() == Conversation.MODE_SINGLE) {
return this.conversation.getContact(); return this.conversation.getContact();
@ -127,7 +132,8 @@ public class Message extends AbstractEntity {
return null; return null;
} else { } else {
Account account = this.conversation.getAccount(); Account account = this.conversation.getAccount();
Contact contact = account.getRoster().getContact(this.trueCounterpart); Contact contact = account.getRoster().getContact(
this.trueCounterpart);
if (contact.showInRoster()) { if (contact.showInRoster()) {
return contact; return contact;
} else { } else {
@ -136,20 +142,20 @@ public class Message extends AbstractEntity {
} }
} }
} }
public String getBody() { public String getBody() {
return body; return body;
} }
public String getReadableBody(Context context) { public String getReadableBody(Context context) {
if ((encryption == ENCRYPTION_PGP)&&(type == TYPE_TEXT)) { if ((encryption == ENCRYPTION_PGP) && (type == TYPE_TEXT)) {
return ""+context.getText(R.string.encrypted_message_received); return "" + context.getText(R.string.encrypted_message_received);
} else if ((encryption == ENCRYPTION_OTR)&&(type == TYPE_IMAGE)) { } else if ((encryption == ENCRYPTION_OTR) && (type == TYPE_IMAGE)) {
return ""+context.getText(R.string.encrypted_image_received); return "" + context.getText(R.string.encrypted_image_received);
} else if (encryption == ENCRYPTION_DECRYPTION_FAILED) { } else if (encryption == ENCRYPTION_DECRYPTION_FAILED) {
return ""+context.getText(R.string.decryption_failed); return "" + context.getText(R.string.decryption_failed);
} else if (type == TYPE_IMAGE) { } else if (type == TYPE_IMAGE) {
return ""+context.getText(R.string.image_file); return "" + context.getText(R.string.image_file);
} else { } else {
return body.trim(); return body.trim();
} }
@ -166,11 +172,11 @@ public class Message extends AbstractEntity {
public int getStatus() { public int getStatus() {
return status; return status;
} }
public String getRemoteMsgId() { public String getRemoteMsgId() {
return this.remoteMsgId; return this.remoteMsgId;
} }
public void setRemoteMsgId(String id) { public void setRemoteMsgId(String id) {
this.remoteMsgId = id; this.remoteMsgId = id;
} }
@ -199,11 +205,11 @@ public class Message extends AbstractEntity {
public boolean isRead() { public boolean isRead() {
return this.read; return this.read;
} }
public void markRead() { public void markRead() {
this.read = true; this.read = true;
} }
public void markUnread() { public void markUnread() {
this.read = false; this.read = false;
} }
@ -223,7 +229,7 @@ public class Message extends AbstractEntity {
public String getEncryptedBody() { public String getEncryptedBody() {
return this.encryptedBody; return this.encryptedBody;
} }
public void setEncryptedBody(String body) { public void setEncryptedBody(String body) {
this.encryptedBody = body; this.encryptedBody = body;
} }
@ -231,7 +237,7 @@ public class Message extends AbstractEntity {
public void setType(int type) { public void setType(int type) {
this.type = type; this.type = type;
} }
public int getType() { public int getType() {
return this.type; return this.type;
} }
@ -243,11 +249,11 @@ public class Message extends AbstractEntity {
this.counterpart = this.counterpart.split("/")[0] + "/" + presence; this.counterpart = this.counterpart.split("/")[0] + "/" + presence;
} }
} }
public void setTrueCounterpart(String trueCounterpart) { public void setTrueCounterpart(String trueCounterpart) {
this.trueCounterpart = trueCounterpart; this.trueCounterpart = trueCounterpart;
} }
public String getPresence() { public String getPresence() {
String[] counterparts = this.counterpart.split("/"); String[] counterparts = this.counterpart.split("/");
if (counterparts.length == 2) { if (counterparts.length == 2) {
@ -256,15 +262,15 @@ public class Message extends AbstractEntity {
return null; return null;
} }
} }
public void setJingleConnection(JingleConnection connection) { public void setJingleConnection(JingleConnection connection) {
this.jingleConnection = connection; this.jingleConnection = connection;
} }
public JingleConnection getJingleConnection() { public JingleConnection getJingleConnection() {
return this.jingleConnection; return this.jingleConnection;
} }
public static Message createStatusMessage(Conversation conversation) { public static Message createStatusMessage(Conversation conversation) {
Message message = new Message(); Message message = new Message();
message.setType(Message.TYPE_STATUS); message.setType(Message.TYPE_STATUS);
@ -275,12 +281,62 @@ public class Message extends AbstractEntity {
public void setCounterpart(String counterpart) { public void setCounterpart(String counterpart) {
this.counterpart = counterpart; this.counterpart = counterpart;
} }
public boolean equals(Message message) { public boolean equals(Message message) {
if ((this.remoteMsgId!=null) && (this.body != null) && (this.counterpart != null)) { if ((this.remoteMsgId != null) && (this.body != null)
return this.remoteMsgId.equals(message.getRemoteMsgId()) && this.body.equals(message.getBody()) && this.counterpart.equals(message.getCounterpart()); && (this.counterpart != null)) {
return this.remoteMsgId.equals(message.getRemoteMsgId())
&& this.body.equals(message.getBody())
&& this.counterpart.equals(message.getCounterpart());
} else { } else {
return false; return false;
} }
} }
public Message next() {
int index = this.conversation.getMessages().indexOf(this);
if (index < 0 || index >= this.conversation.getMessages().size() - 1) {
return null;
} else {
return this.conversation.getMessages().get(index + 1);
}
}
public Message prev() {
int index = this.conversation.getMessages().indexOf(this);
if (index <= 0 || index > this.conversation.getMessages().size()) {
return null;
} else {
return this.conversation.getMessages().get(index - 1);
}
}
public boolean mergable(Message message) {
if (message == null) {
return false;
}
return (this.getType() == Message.TYPE_TEXT
&& this.getType() == message.getType()
&& this.getCounterpart().equals(message.getCounterpart())
&& (message.getTimeSent() - this.getTimeSent()) <= 20000 && ((this
.getStatus() == message.getStatus()) || (this.getStatus() == Message.STATUS_SEND && message
.getStatus() == Message.STATUS_UNSEND)));
}
public String getMergedBody() {
Message next = this.next();
if (this.mergable(next)) {
return body.trim() + '\n' + next.getMergedBody();
}
return body.trim();
}
public boolean wasMergedIntoPrevious() {
Message prev = this.prev();
if (prev == null) {
return false;
} else {
return prev.mergable(this);
}
}
} }

View File

@ -36,6 +36,7 @@ public class MessageAdapter extends ArrayAdapter<Message> {
private static final int SENT = 0; private static final int SENT = 0;
private static final int RECIEVED = 1; private static final int RECIEVED = 1;
private static final int STATUS = 2; private static final int STATUS = 2;
private static final int NULL = 3;
private ConversationActivity activity; private ConversationActivity activity;
@ -74,12 +75,14 @@ public class MessageAdapter extends ArrayAdapter<Message> {
@Override @Override
public int getViewTypeCount() { public int getViewTypeCount() {
return 3; return 4;
} }
@Override @Override
public int getItemViewType(int position) { public int getItemViewType(int position) {
if (getItem(position).getType() == Message.TYPE_STATUS) { if (getItem(position).wasMergedIntoPrevious()) {
return NULL;
} else if (getItem(position).getType() == Message.TYPE_STATUS) {
return STATUS; return STATUS;
} else if (getItem(position).getStatus() <= Message.STATUS_RECEIVED) { } else if (getItem(position).getStatus() <= Message.STATUS_RECEIVED) {
return RECIEVED; return RECIEVED;
@ -212,7 +215,7 @@ public class MessageAdapter extends ArrayAdapter<Message> {
viewHolder.messageBody.setVisibility(View.VISIBLE); viewHolder.messageBody.setVisibility(View.VISIBLE);
if (message.getBody() != null) { if (message.getBody() != null) {
if (message.getType() != Message.TYPE_PRIVATE) { if (message.getType() != Message.TYPE_PRIVATE) {
viewHolder.messageBody.setText(message.getBody().trim()); viewHolder.messageBody.setText(message.getMergedBody());
} else { } else {
String privateMarker; String privateMarker;
if (message.getStatus() <= Message.STATUS_RECEIVED) { if (message.getStatus() <= Message.STATUS_RECEIVED) {
@ -301,6 +304,10 @@ public class MessageAdapter extends ArrayAdapter<Message> {
if (view == null) { if (view == null) {
viewHolder = new ViewHolder(); viewHolder = new ViewHolder();
switch (type) { switch (type) {
case NULL:
view = (View) activity.getLayoutInflater().inflate(
R.layout.message_null, parent, false);
break;
case SENT: case SENT:
view = (View) activity.getLayoutInflater().inflate( view = (View) activity.getLayoutInflater().inflate(
R.layout.message_sent, parent, false); R.layout.message_sent, parent, false);
@ -382,7 +389,7 @@ public class MessageAdapter extends ArrayAdapter<Message> {
viewHolder = (ViewHolder) view.getTag(); viewHolder = (ViewHolder) view.getTag();
} }
if (type == STATUS) { if (type == STATUS || type == NULL) {
return view; return view;
} }