diff --git a/background.html b/background.html
index d394b45939d5..00adb58e2214 100644
--- a/background.html
+++ b/background.html
@@ -480,6 +480,7 @@
+
diff --git a/js/background.js b/js/background.js
index 2eb48c0466ab..c8f34aebe8ae 100644
--- a/js/background.js
+++ b/js/background.js
@@ -169,7 +169,8 @@
received_at : now,
conversationId : data.destination,
type : 'outgoing',
- sent : true
+ sent : true,
+ expirationStartTimestamp: data.expirationStartTimestamp,
});
message.handleDataMessage(data.message);
diff --git a/js/expiring_messages.js b/js/expiring_messages.js
new file mode 100644
index 000000000000..40fa55650353
--- /dev/null
+++ b/js/expiring_messages.js
@@ -0,0 +1,14 @@
+
+/*
+ * vim: ts=4:sw=4:expandtab
+ */
+;(function() {
+ 'use strict';
+ window.Whisper = window.Whisper || {};
+ Whisper.ExpiringMessages = new (Whisper.MessageCollection.extend({
+ initialize: function() {
+ this.on('expired', this.remove);
+ this.fetchExpiring();
+ }
+ }))();
+})();
diff --git a/js/models/conversations.js b/js/models/conversations.js
index 909320d86454..69e586c71232 100644
--- a/js/models/conversations.js
+++ b/js/models/conversations.js
@@ -230,17 +230,20 @@
this.getUnread().then(function(unreadMessages) {
var read = unreadMessages.map(function(m) {
+ if (this.messageCollection.get(m.id)) {
+ m = this.messageCollection.get(m.id);
+ }
m.markRead();
return {
sender : m.get('source'),
timestamp : m.get('sent_at')
};
- });
+ }.bind(this));
if (read.length > 0) {
console.log('Sending', read.length, 'read receipts');
textsecure.messaging.syncReadMessages(read);
}
- });
+ }.bind(this));
}
},
diff --git a/js/models/messages.js b/js/models/messages.js
index ade716c46da8..aaa83e12d172 100644
--- a/js/models/messages.js
+++ b/js/models/messages.js
@@ -11,6 +11,9 @@
initialize: function() {
this.on('change:attachments', this.updateImageUrl);
this.on('destroy', this.revokeImageUrl);
+ this.on('change:expirationStartTimestamp', this.setToExpire);
+ this.on('change:expireTimer', this.setToExpire);
+ this.setToExpire();
},
defaults : function() {
return {
@@ -344,6 +347,10 @@
errors : []
});
+ if (dataMessage.expireTimer) {
+ message.set({expireTimer: dataMessage.expireTimer});
+ }
+
var conversation_timestamp = conversation.get('timestamp');
if (!conversation_timestamp || message.get('sent_at') > conversation_timestamp) {
conversation.set({
@@ -367,12 +374,35 @@
});
});
},
- markRead: function(sync) {
+ markRead: function() {
this.unset('unread');
+ if (this.get('expireTimer') && !this.get('expirationStartTimestamp')) {
+ this.set('expirationStartTimestamp', Date.now());
+ }
Whisper.Notifications.remove(Whisper.Notifications.where({
messageId: this.id
}));
return this.save();
+ },
+ markExpired: function() {
+ console.log('message', this.get('sent_at'), 'expired');
+ clearInterval(this.expirationTimeout);
+ this.expirationTimeout = null;
+ this.trigger('expired', this);
+ this.destroy();
+ },
+ setToExpire: function() {
+ if (this.get('expireTimer') && this.get('expirationStartTimestamp') && !this.expireTimer) {
+ var now = Date.now();
+ var start = this.get('expirationStartTimestamp');
+ var delta = this.get('expireTimer') * 1000;
+ var ms_from_now = start + delta - now;
+ if (ms_from_now < 0) {
+ ms_from_now = 0;
+ }
+ console.log('message', this.get('sent_at'), 'expires in', ms_from_now, 'ms');
+ this.expirationTimeout = setTimeout(this.markExpired.bind(this), ms_from_now);
+ }
}
});
@@ -434,6 +464,10 @@
}.bind(this));
},
+ fetchExpiring: function() {
+ this.fetch({conditions: {expireTimer: {$gte: 0}}});
+ },
+
hasKeyConflicts: function() {
return this.any(function(m) { return m.hasKeyConflicts(); });
}
diff --git a/js/views/conversation_view.js b/js/views/conversation_view.js
index 1b276630ad6b..72eeaf09f694 100644
--- a/js/views/conversation_view.js
+++ b/js/views/conversation_view.js
@@ -43,6 +43,7 @@
this.listenTo(this.model, 'change:name', this.updateTitle);
this.listenTo(this.model, 'newmessage', this.addMessage);
this.listenTo(this.model, 'opened', this.onOpened);
+ this.listenTo(this.model.messageCollection, 'expired', this.onExpired);
this.render();
@@ -166,8 +167,13 @@
// TODO catch?
},
+ onExpired: function(message) {
+ this.model.messageCollection.remove(message.id);
+ },
+
addMessage: function(message) {
this.model.messageCollection.add(message, {merge: true});
+ message.setToExpire();
if (!this.isHidden() && window.isFocused()) {
this.markRead();
diff --git a/js/views/message_view.js b/js/views/message_view.js
index 20dea9cd5cfe..3a206fb12e64 100644
--- a/js/views/message_view.js
+++ b/js/views/message_view.js
@@ -35,7 +35,8 @@
this.listenTo(this.model, 'change:delivered', this.renderDelivered);
this.listenTo(this.model, 'change', this.renderSent);
this.listenTo(this.model, 'change:flags change:group_update', this.renderControl);
- this.listenTo(this.model, 'destroy', this.remove);
+ this.listenTo(this.model, 'destroy', this.onDestroy);
+ this.listenTo(this.model, 'expired', this.onExpired);
this.listenTo(this.model, 'pending', this.renderPending);
this.listenTo(this.model, 'done', this.renderDone);
this.timeStampView = new Whisper.ExtendedTimestampView();
@@ -62,6 +63,17 @@
this.model.resend(number);
}.bind(this));
},
+ onExpired: function() {
+ this.$el.addClass('expired');
+ this.$el.find('.bubble').one('webkitAnimationEnd animationend',
+ this.remove.bind(this));
+ },
+ onDestroy: function() {
+ if (this.$el.hasClass('expired')) {
+ return;
+ }
+ this.remove();
+ },
select: function(e) {
this.$el.trigger('select', {message: this.model});
e.stopPropagation();
diff --git a/stylesheets/_conversation.scss b/stylesheets/_conversation.scss
index 24e32c6f0fca..a13c1cd55bdf 100644
--- a/stylesheets/_conversation.scss
+++ b/stylesheets/_conversation.scss
@@ -383,6 +383,18 @@ li.entry .error-icon-container {
}
}
+ @keyframes shake {
+ 0% { transform: translateX(0px); }
+ 25% { transform: translateX(-5px); }
+ 50% { transform: translateX(0px); }
+ 75% { transform: translateX(5px); }
+ 100% { transform: translateX(0px); }
+ }
+
+ .expired .bubble {
+ animation: shake 0.2s linear 3;
+ }
+
.control {
.bubble {
.content {
diff --git a/stylesheets/manifest.css b/stylesheets/manifest.css
index 50628b4470f0..61dea07ffc01 100644
--- a/stylesheets/manifest.css
+++ b/stylesheets/manifest.css
@@ -1207,6 +1207,20 @@ li.entry .error-icon-container {
.message-container .outgoing .bubble,
.message-list .outgoing .bubble {
clear: left; }
+@keyframes shake {
+ 0% {
+ transform: translateX(0px); }
+ 25% {
+ transform: translateX(-5px); }
+ 50% {
+ transform: translateX(0px); }
+ 75% {
+ transform: translateX(5px); }
+ 100% {
+ transform: translateX(0px); } }
+ .message-container .expired .bubble,
+ .message-list .expired .bubble {
+ animation: shake 0.2s linear 3; }
.message-container .control .bubble .content,
.message-list .control .bubble .content {
font-style: italic; }