experimental message merging
This commit is contained in:
parent
9eafb10086
commit
8f8d4e320d
|
@ -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>
|
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue