Provide user feedback when number is invalid
While typing a number, the new contact element is faded out. When the number becomes valid it is opaque. If the element is clicked while invalid, it displays 'Invalid number' and waits for the input to change again. A new conversation is only opened if the number is valid. // FREEBIE
This commit is contained in:
parent
01593363eb
commit
61a57a753b
4 changed files with 49 additions and 13 deletions
|
@ -132,6 +132,12 @@
|
||||||
{{> avatar }}
|
{{> avatar }}
|
||||||
<div class='contact-details'> {{> contact_name_and_number }} </div>
|
<div class='contact-details'> {{> contact_name_and_number }} </div>
|
||||||
</script>
|
</script>
|
||||||
|
<script type='text/x-tmpl-mustache' id='new-contact'>
|
||||||
|
{{> avatar }}
|
||||||
|
<div class='contact-details'>
|
||||||
|
{{> contact_name_and_number }}
|
||||||
|
</div>
|
||||||
|
</script>
|
||||||
<script type='text/x-tmpl-mustache' id='conversation-preview'>
|
<script type='text/x-tmpl-mustache' id='conversation-preview'>
|
||||||
{{> avatar }}
|
{{> avatar }}
|
||||||
<div class='contact-details'>
|
<div class='contact-details'>
|
||||||
|
|
|
@ -5,6 +5,31 @@
|
||||||
'use strict';
|
'use strict';
|
||||||
window.Whisper = window.Whisper || {};
|
window.Whisper = window.Whisper || {};
|
||||||
|
|
||||||
|
Whisper.NewContactView = Whisper.View.extend({
|
||||||
|
templateName: 'new-contact',
|
||||||
|
className: 'conversation-list-item contact',
|
||||||
|
events: {
|
||||||
|
'click': 'validate'
|
||||||
|
},
|
||||||
|
initialize: function() {
|
||||||
|
this.listenTo(this.model, 'change', this.render); // auto update
|
||||||
|
},
|
||||||
|
render_attributes: function() {
|
||||||
|
return {
|
||||||
|
number: 'Click to create new contact',
|
||||||
|
title: this.model.getNumber(),
|
||||||
|
avatar: this.model.getAvatar(),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
validate: function() {
|
||||||
|
if (this.model.isValid()) {
|
||||||
|
this.$el.addClass('valid');
|
||||||
|
} else {
|
||||||
|
this.$el.removeClass('valid');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
Whisper.ConversationSearchView = Whisper.View.extend({
|
Whisper.ConversationSearchView = Whisper.View.extend({
|
||||||
className: 'conversation-search',
|
className: 'conversation-search',
|
||||||
initialize: function(options) {
|
initialize: function(options) {
|
||||||
|
@ -25,7 +50,7 @@
|
||||||
},
|
},
|
||||||
|
|
||||||
events: {
|
events: {
|
||||||
'select .new-contact': 'createConversation',
|
'click .new-contact': 'createConversation',
|
||||||
},
|
},
|
||||||
|
|
||||||
filterContacts: function(e) {
|
filterContacts: function(e) {
|
||||||
|
@ -34,7 +59,9 @@
|
||||||
if (this.maybeNumber(query)) {
|
if (this.maybeNumber(query)) {
|
||||||
this.new_contact_view.model.set('id', query);
|
this.new_contact_view.model.set('id', query);
|
||||||
this.new_contact_view.render().$el.show();
|
this.new_contact_view.render().$el.show();
|
||||||
|
this.new_contact_view.validate();
|
||||||
this.hideHints();
|
this.hideHints();
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
this.new_contact_view.$el.hide();
|
this.new_contact_view.$el.hide();
|
||||||
}
|
}
|
||||||
|
@ -55,7 +82,7 @@
|
||||||
this.new_contact_view.$el.hide();
|
this.new_contact_view.$el.hide();
|
||||||
}
|
}
|
||||||
// Creates a view to display a new contact
|
// Creates a view to display a new contact
|
||||||
this.new_contact_view = new Whisper.ConversationListItemView({
|
this.new_contact_view = new Whisper.NewContactView({
|
||||||
el: this.$new_contact,
|
el: this.$new_contact,
|
||||||
model: ConversationController.create({
|
model: ConversationController.create({
|
||||||
type: 'private',
|
type: 'private',
|
||||||
|
@ -66,8 +93,7 @@
|
||||||
|
|
||||||
createConversation: function() {
|
createConversation: function() {
|
||||||
var conversation = this.new_contact_view.model;
|
var conversation = this.new_contact_view.model;
|
||||||
var error = conversation.validate(conversation.attributes);
|
if (this.new_contact_view.model.isValid()) {
|
||||||
if (!error) {
|
|
||||||
ConversationController.findOrCreatePrivateById(
|
ConversationController.findOrCreatePrivateById(
|
||||||
this.new_contact_view.model.id
|
this.new_contact_view.model.id
|
||||||
).then(function(conversation) {
|
).then(function(conversation) {
|
||||||
|
@ -75,6 +101,9 @@
|
||||||
this.initNewContact();
|
this.initNewContact();
|
||||||
this.resetTypeahead();
|
this.resetTypeahead();
|
||||||
}.bind(this));
|
}.bind(this));
|
||||||
|
} else {
|
||||||
|
this.new_contact_view.$('.number').text("Invalid number");
|
||||||
|
this.$input.focus();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -147,15 +147,17 @@ input.search {
|
||||||
.new-contact {
|
.new-contact {
|
||||||
display: none;
|
display: none;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
opacity: 0.7;
|
||||||
background: $grey_l;
|
background: $grey_l;
|
||||||
.name, .last-timestamp, .avatar { display: none; }
|
.contact-details .number {
|
||||||
.contact-details::before {
|
|
||||||
content: 'Create new contact';
|
|
||||||
display: block;
|
display: block;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
opacity: 0.7;
|
|
||||||
padding-right: 8px;
|
padding-right: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.valid {
|
||||||
|
opacity: 1.0
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.index {
|
.index {
|
||||||
|
|
|
@ -487,15 +487,14 @@ input.search {
|
||||||
.new-contact {
|
.new-contact {
|
||||||
display: none;
|
display: none;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
opacity: 0.7;
|
||||||
background: #f3f3f3; }
|
background: #f3f3f3; }
|
||||||
.new-contact .name, .new-contact .last-timestamp, .new-contact .avatar {
|
.new-contact .contact-details .number {
|
||||||
display: none; }
|
|
||||||
.new-contact .contact-details::before {
|
|
||||||
content: 'Create new contact';
|
|
||||||
display: block;
|
display: block;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
opacity: 0.7;
|
|
||||||
padding-right: 8px; }
|
padding-right: 8px; }
|
||||||
|
.new-contact.valid {
|
||||||
|
opacity: 1; }
|
||||||
|
|
||||||
.index {
|
.index {
|
||||||
max-width: 1300px;
|
max-width: 1300px;
|
||||||
|
|
Loading…
Reference in a new issue