diff --git a/background.html b/background.html index 0fc48de2..ce4387f1 100644 --- a/background.html +++ b/background.html @@ -78,6 +78,7 @@

{{ message }}

{{ timestamp }} +
diff --git a/js/models/messages.js b/js/models/messages.js index 2d711a4c..18c61b80 100644 --- a/js/models/messages.js +++ b/js/models/messages.js @@ -139,9 +139,12 @@ }, send: function(promise) { + this.trigger('pending'); return promise.then(function() { + this.trigger('done'); this.save({sent: true}); }.bind(this)).catch(function(errors) { + this.trigger('done'); this.set({sent: true}); this.saveErrors(errors); }.bind(this)); diff --git a/js/views/message_view.js b/js/views/message_view.js index ad795159..3c0ca0a8 100644 --- a/js/views/message_view.js +++ b/js/views/message_view.js @@ -14,6 +14,8 @@ 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, 'pending', this.renderPending); + this.listenTo(this.model, 'done', this.renderDone); }, events: { 'click .timestamp': 'select', @@ -26,6 +28,12 @@ className: function() { return ["entry", this.model.get('type')].join(' '); }, + renderPending: function() { + this.$el.addClass('pending'); + }, + renderDone: function() { + this.$el.removeClass('pending'); + }, renderSent: function() { if (this.model.isOutgoing()) { this.$el.toggleClass('sent', !!this.model.get('sent')); diff --git a/stylesheets/_conversation.scss b/stylesheets/_conversation.scss index ee56482e..bbc01d49 100644 --- a/stylesheets/_conversation.scss +++ b/stylesheets/_conversation.scss @@ -189,6 +189,15 @@ margin-right: 3px; } +.sinewave { + background: url('/images/sinewave.png') repeat-x; + animation: backgroundScroll 0.5s linear infinite; +} + +@keyframes backgroundScroll { + from {background-position: 0 0;} + to {background-position: -18px 0;} +} .entry.sent .checkmark { display: inline-block; width: 18px; @@ -204,6 +213,14 @@ height: 1em; } } +.entry.pending { + .sinewave { + display: inline-block; + width: 18px; + height: 1em; + } + .double-checkmark, .checkmark { display: none; } +} .message-list { margin: 0; diff --git a/stylesheets/manifest.css b/stylesheets/manifest.css index 06e906e8..59bcd98f 100644 --- a/stylesheets/manifest.css +++ b/stylesheets/manifest.css @@ -673,6 +673,15 @@ input.search { font-size: smaller; margin-right: 3px; } +.sinewave { + background: url("/images/sinewave.png") repeat-x; + animation: backgroundScroll 0.5s linear infinite; } + +@keyframes backgroundScroll { + from { + background-position: 0 0; } + to { + background-position: -18px 0; } } .entry.sent .checkmark { display: inline-block; width: 18px; @@ -687,6 +696,13 @@ input.search { width: 18px; height: 1em; } +.entry.pending .sinewave { + display: inline-block; + width: 18px; + height: 1em; } +.entry.pending .double-checkmark, .entry.pending .checkmark { + display: none; } + .message-list { margin: 0; padding: 1em 0;