Support for group-member verifications via second-level panel

Also:
- All the necessary wire-up to update things in real time. If you have
a safety number page up via a group member view as well as via a 1:1
conversation with that contact, they'll both be updated as the
underlying model changes. Similarly, the overall group will update
in real-time as members change.
- A bit of special-casing for yourself in a group conversation - you're
shown as 'me' and are not clickable, where normally that would take you
to the Safety Number screen for that contact. You are also not included
in the trust calculations for a given group.

FREEBIE
This commit is contained in:
Scott Nonnenberg 2017-06-10 12:18:24 -07:00
parent ae3587f05e
commit bedf10056b
12 changed files with 281 additions and 46 deletions

View file

@ -1,4 +1,8 @@
{ {
"me": {
"message": "Me",
"description": "The label for yourself when shown in a group member list"
},
"youLeftTheGroup": { "youLeftTheGroup": {
"message": "You left the group", "message": "You left the group",
"description": "Displayed when a user can't send a message because they have left the group" "description": "Displayed when a user can't send a message because they have left the group"
@ -172,15 +176,40 @@
"message": "Send a message", "message": "Send a message",
"description": "Placeholder text in the message entry field" "description": "Placeholder text in the message entry field"
}, },
"members": { "groupMembers": {
"message": "Members" "message": "Group members"
},
"showMembers": {
"message": "Show members"
}, },
"resetSession": { "resetSession": {
"message": "Reset session", "message": "Reset session",
"description": "This is a menu item for resetting the session, using the imperative case, as in a command." "description": "This is a menu item for resetting the session, using the imperative case, as in a command."
}, },
"showSafetyNumber": { "showSafetyNumber": {
"message": "Show safety number" "message": "Show safety number"
},
"markAsNotVerified": {
"message": "Mark as not verified"
},
"verifyHelp": {
"message": "If you wish to verify the security of your end-to-end encryption with $name$, compare the numbers above with the numbers on their device.",
"placeholders": {
"name": {
"content": "$1",
"example": "John"
}
}
},
"isVerified": {
"message": "$name$ is verified",
"description": "If the user has manually marked a contact's safety number as verified, this string is shown on the 'Show Safety Number' screen",
"placeholders": {
"name": {
"content": "$1",
"example": "John"
}
}
}, },
"theirIdentityUnknown": { "theirIdentityUnknown": {
"message": "You haven't exchanged any messages with this contact yet. Your safety number with them will be available after the first message." "message": "You haven't exchanged any messages with this contact yet. Your safety number with them will be available after the first message."

View file

@ -66,6 +66,17 @@
<script type='text/x-tmpl-mustache' id='hint'> <script type='text/x-tmpl-mustache' id='hint'>
<p> {{ content }}</p> <p> {{ content }}</p>
</script> </script>
<script type='text/x-tmpl-mustache' id='conversation-title'>
{{ #name }}
<span class='conversation-name' dir='auto'>{{ name }}</span>
{{ /name }}
{{ #number }}
<span class='conversation-number'>{{ number }}</span>
{{ /number }}
{{ #verified }}
<span class='verified'>✓ Verified</span>
{{ /verified }}
</script>
<script type='text/x-tmpl-mustache' id='conversation'> <script type='text/x-tmpl-mustache' id='conversation'>
<div class='conversation-header {{ avatar.color }}'> <div class='conversation-header {{ avatar.color }}'>
<div class='header-buttons left'> <div class='header-buttons left'>
@ -78,16 +89,16 @@
<div class='conversation-menu menu'> <div class='conversation-menu menu'>
<button class='hamburger' alt='conversation menu'></button> <button class='hamburger' alt='conversation menu'></button>
<ul class='menu-list'> <ul class='menu-list'>
<li class='disappearing-messages'>{{ disappearing-messages }}</li>
{{#group}} {{#group}}
<li class='view-members'>{{ view-members }}</li> <li class='show-members'>{{ show-members }}</li>
<!-- <li class='update-group'>Update group</li> --> <!-- <li class='update-group'>Update group</li> -->
<!-- <li class='leave-group'>Leave group</li> --> <!-- <li class='leave-group'>Leave group</li> -->
{{/group}} {{/group}}
{{^group}} {{^group}}
<li class='end-session'>{{ end-session }}</li>
<li class='show-identity'>{{ show-identity }}</li> <li class='show-identity'>{{ show-identity }}</li>
<li class='end-session'>{{ end-session }}</li>
{{/group}} {{/group}}
<li class='disappearing-messages'>{{ disappearing-messages }}</li>
<li class='destroy'>{{ destroy }}</li> <li class='destroy'>{{ destroy }}</li>
</ul> </ul>
</div> </div>
@ -101,14 +112,7 @@
</div> </div>
</div> </div>
</div> </div>
<span class='conversation-title'> <span class='conversation-title'></span>
{{ #name }}
<span class='conversation-name' dir='auto'>{{ name }}</span>
{{ /name }}
{{ #number }}
<span class='conversation-number'>{{ number }}</span>
{{ /number }}
</span>
{{> avatar }} {{> avatar }}
</div> </div>
<div class='main panel'> <div class='main panel'>
@ -239,11 +243,11 @@
</script> </script>
<script type='text/x-tmpl-mustache' id='contact_name_and_number'> <script type='text/x-tmpl-mustache' id='contact_name_and_number'>
<h3 class='name' dir='auto'> {{ title }} </h3> <h3 class='name' dir='auto'> {{ title }} </h3>
<div class='number'>{{ number }}</div> <div class='number'>{{ #verified }}✓ Verified &middot;{{ /verified }} {{ number }}</div>
</script> </script>
<script type='text/x-tmpl-mustache' id='contact'> <script type='text/x-tmpl-mustache' id='contact'>
{{> avatar }} {{> avatar }}
<div class='contact-details'> {{> contact_name_and_number }} </div> <div class='contact-details {{ class }}'> {{> contact_name_and_number }} </div>
</script> </script>
<script type='text/x-tmpl-mustache' id='new-contact'> <script type='text/x-tmpl-mustache' id='new-contact'>
{{> avatar }} {{> avatar }}
@ -321,7 +325,13 @@
{{ #chunks }} <span>{{ . }}</span> {{ /chunks }} {{ #chunks }} <span>{{ . }}</span> {{ /chunks }}
</div> </div>
{{ /has_their_key }} {{ /has_their_key }}
{{ verifyHelp }}
<p> {{> link_to_support }} </p> <p> {{> link_to_support }} </p>
<div class='verify'>
<button class='verify'>
{{ verifyButton }}
</button>
</div>
</div> </div>
</script> </script>
<!-- index --> <!-- index -->

View file

@ -37,6 +37,8 @@
}, },
initialize: function() { initialize: function() {
this.ourNumber = textsecure.storage.user.getNumber();
this.contactCollection = new Backbone.Collection(); this.contactCollection = new Backbone.Collection();
this.messageCollection = new Whisper.MessageCollection([], { this.messageCollection = new Whisper.MessageCollection([], {
conversation: this conversation: this
@ -48,6 +50,84 @@
this.on('destroy', this.revokeAvatarUrl); this.on('destroy', this.revokeAvatarUrl);
}, },
updateVerified: function() {
// TODO: replace this with the real call
function checkTrustStore() {
return Promise.resolve('default');
}
if (this.isPrivate()) {
return Promise.all([
checkTrustStore(this.id),
this.fetch()
]).then(function(results) {
var trust = results[0];
return this.save({verified: trust});
});
} else {
return this.fetchContacts().then(function() {
return Promise.all(this.contactCollection.map(function(contact) {
if (contact.id !== this.myNumber) {
return contact.updateVerified();
}
}.bind(this)));
}.bind(this));
}
},
isVerified: function() {
if (this.isPrivate()) {
return this.get('verified') === 'verified';
} else {
return this.contactCollection.every(function(contact) {
if (contact.id === this.myNumber) {
return true;
} else {
return contact.isVerified();
}
}.bind(this));
}
},
isConflict: function() {
if (this.isPrivate()) {
var verified = this.get('verified');
return verified !== 'verified' && verified !== 'default';
} else {
return Boolean(this.getConflicts().length);
}
},
getConflicts: function() {
if (this.isPrivate()) {
return this.isConflict() ? [this] : [];
} else {
return this.contactCollection.filter(function(contact) {
if (contact.id === this.myNumber) {
return false;
} else {
return contact.isConflict();
}
}.bind(this));
}
},
onMemberVerifiedChange: function() {
// If the verified state of a member changes, our aggregate state changes.
// We trigger both events to replicate the behavior of Backbone.Model.set()
this.trigger('change:verified');
this.trigger('change');
},
toggleVerified: function() {
if (!this.isPrivate()) {
throw new Error('You cannot verify a group conversation. ' +
'You must verify individual contacts.');
}
if (this.isVerified()) {
this.save({verified: 'default'});
} else {
this.save({verified: 'verified'});
}
},
addKeyChange: function(id) { addKeyChange: function(id) {
console.log('adding key change advisory for', this.id, this.get('timestamp')); console.log('adding key change advisory for', this.id, this.get('timestamp'));
var timestamp = Date.now(); var timestamp = Date.now();
@ -373,12 +453,14 @@
} else { } else {
var promises = []; var promises = [];
var members = this.get('members') || []; var members = this.get('members') || [];
this.contactCollection.reset( this.contactCollection.reset(
members.map(function(number) { members.map(function(number) {
var c = ConversationController.create({ var c = ConversationController.create({
id : number, id : number,
type : 'private' type : 'private'
}); });
this.listenTo(c, 'change:verified', this.onMemberVerifiedChange);
promises.push(new Promise(function(resolve) { promises.push(new Promise(function(resolve) {
c.fetch().always(resolve); c.fetch().always(resolve);
})); }));

View file

@ -12,12 +12,41 @@
tagName: 'div', tagName: 'div',
className: 'contact', className: 'contact',
templateName: 'contact', templateName: 'contact',
events: {
'click': 'showIdentity'
},
initialize: function(options) {
this.ourNumber = textsecure.storage.user.getNumber();
this.listenBack = options.listenBack;
this.listenTo(this.model, 'change', this.render);
},
render_attributes: function() { render_attributes: function() {
if (this.model.id === this.ourNumber) {
return {
class: 'not-clickable',
title: i18n('me'),
number: this.model.getNumber(),
avatar: this.model.getAvatar()
};
}
return { return {
class: '',
title: this.model.getTitle(), title: this.model.getTitle(),
number: this.model.getNumber(), number: this.model.getNumber(),
avatar: this.model.getAvatar() avatar: this.model.getAvatar(),
verified: this.model.isVerified()
}; };
},
showIdentity: function() {
if (this.model.id === this.ourNumber) {
return;
}
var view = new Whisper.KeyVerificationPanelView({
model: this.model
});
this.listenBack(view);
} }
}) })
}); });

View file

@ -57,6 +57,17 @@
} }
}); });
Whisper.ConversationTitleView = Whisper.View.extend({
templateName: 'conversation-title',
render_attributes: function() {
return {
verified: this.model.isVerified(),
name: this.model.getName(),
number: this.model.getNumber(),
};
}
});
Whisper.ConversationView = Whisper.View.extend({ Whisper.ConversationView = Whisper.View.extend({
className: function() { className: function() {
return [ 'conversation', this.model.get('type') ].join(' '); return [ 'conversation', this.model.get('type') ].join(' ');
@ -68,13 +79,11 @@
render_attributes: function() { render_attributes: function() {
return { return {
group: this.model.get('type') === 'group', group: this.model.get('type') === 'group',
name: this.model.getName(),
number: this.model.getNumber(),
avatar: this.model.getAvatar(), avatar: this.model.getAvatar(),
expireTimer: this.model.get('expireTimer'), expireTimer: this.model.get('expireTimer'),
'view-members' : i18n('members'), 'show-members' : i18n('showMembers'),
'end-session' : i18n('resetSession'), 'end-session' : i18n('resetSession'),
'show-identity' : i18n('showSafetyNumber'), 'show-identity' : i18n('showSafetyNumber'),
'destroy' : i18n('deleteMessages'), 'destroy' : i18n('deleteMessages'),
'send-message' : i18n('sendMessage'), 'send-message' : i18n('sendMessage'),
'disappearing-messages': i18n('disappearingMessages'), 'disappearing-messages': i18n('disappearingMessages'),
@ -84,6 +93,7 @@
}, },
initialize: function(options) { initialize: function(options) {
this.listenTo(this.model, 'destroy', this.stopListening); this.listenTo(this.model, 'destroy', this.stopListening);
this.listenTo(this.model, 'change', this.updateTitle);
this.listenTo(this.model, 'change:color', this.updateColor); this.listenTo(this.model, 'change:color', this.updateColor);
this.listenTo(this.model, 'change:name', this.updateTitle); this.listenTo(this.model, 'change:name', this.updateTitle);
this.listenTo(this.model, 'newmessage', this.addMessage); this.listenTo(this.model, 'newmessage', this.addMessage);
@ -103,6 +113,13 @@
window: this.window window: this.window
}); });
this.titleView = new Whisper.ConversationTitleView({
el: this.$('.conversation-title'),
model: this.model
});
this.titleView.render();
this.titleView.render();
this.view = new Whisper.MessageListView({ this.view = new Whisper.MessageListView({
collection: this.model.messageCollection, collection: this.model.messageCollection,
window: this.window window: this.window
@ -145,7 +162,7 @@
'click .leave-group': 'leaveGroup', 'click .leave-group': 'leaveGroup',
'click .update-group': 'newGroupUpdate', 'click .update-group': 'newGroupUpdate',
'click .show-identity': 'showIdentity', 'click .show-identity': 'showIdentity',
'click .view-members': 'viewMembers', 'click .show-members': 'showMembers',
'click .conversation-menu .hamburger': 'toggleMenu', 'click .conversation-menu .hamburger': 'toggleMenu',
'click .openInbox' : 'openInbox', 'click .openInbox' : 'openInbox',
'click' : 'onClick', 'click' : 'onClick',
@ -167,6 +184,10 @@
'force-resize': 'forceUpdateMessageFieldSize', 'force-resize': 'forceUpdateMessageFieldSize',
'show-identity': 'showIdentity' 'show-identity': 'showIdentity'
}, },
updateTitle: function() {
this.titleView.render();
},
enableDisappearingMessages: function() { enableDisappearingMessages: function() {
if (!this.model.get('expireTimer')) { if (!this.model.get('expireTimer')) {
this.model.updateExpirationTimer( this.model.updateExpirationTimer(
@ -416,9 +437,12 @@
this.model.messageCollection.add(message, {merge: true}); this.model.messageCollection.add(message, {merge: true});
}, },
viewMembers: function() { showMembers: function() {
return this.model.fetchContacts().then(function() { return this.model.fetchContacts().then(function() {
var view = new Whisper.GroupMemberList({ model: this.model }); var view = new Whisper.GroupMemberList({
model: this.model,
listenBack: this.listenBack.bind(this)
});
this.listenBack(view); this.listenBack(view);
}.bind(this)); }.bind(this));
}, },
@ -515,16 +539,25 @@
}, },
listenBack: function(view) { listenBack: function(view) {
this.panel = view; this.panels = this.panels || [];
this.$('.main.panel, .header-buttons.right').hide(); this.panels.unshift(view);
this.$('.back').show();
view.$el.insertBefore(this.$('.panel')); if (this.panels.length === 1) {
this.$('.main.panel, .header-buttons.right').hide();
this.$('.back').show();
}
view.$el.insertBefore(this.$('.panel').first());
}, },
resetPanel: function() { resetPanel: function() {
this.panel.remove(); var view = this.panels.shift();
this.$('.main.panel, .header-buttons.right').show(); view.remove();
this.$('.back').hide();
this.$el.trigger('force-resize'); if (this.panels.length === 0) {
this.$('.main.panel, .header-buttons.right').show();
this.$('.back').hide();
this.$el.trigger('force-resize');
}
}, },
closeMenu: function(e) { closeMenu: function(e) {
@ -614,10 +647,6 @@
}); });
}, },
updateTitle: function() {
this.$('.conversation-title').text(this.model.getTitle());
},
updateColor: function(model, color) { updateColor: function(model, color) {
var header = this.$('.conversation-header'); var header = this.$('.conversation-header');
header.removeClass(Whisper.Conversation.COLORS); header.removeClass(Whisper.Conversation.COLORS);

View file

@ -13,17 +13,22 @@
Whisper.GroupMemberList = Whisper.View.extend({ Whisper.GroupMemberList = Whisper.View.extend({
className: 'group-member-list panel', className: 'group-member-list panel',
templateName: 'group-member-list', templateName: 'group-member-list',
initialize: function() { initialize: function(options) {
this.render(); this.render();
console.log('GroupMemberList', options);
this.member_list_view = new Whisper.ContactListView({ this.member_list_view = new Whisper.ContactListView({
collection: this.model.contactCollection, collection: this.model.contactCollection,
className: 'members' className: 'members',
toInclude: {
listenBack: options.listenBack
}
}); });
this.member_list_view.render(); this.member_list_view.render();
this.$('.container').append(this.member_list_view.el); this.$('.container').append(this.member_list_view.el);
}, },
render_attributes: { render_attributes: {
members: i18n('members') members: i18n('groupMembers')
} }
}); });
})(); })();

View file

@ -5,25 +5,30 @@
'use strict'; 'use strict';
window.Whisper = window.Whisper || {}; window.Whisper = window.Whisper || {};
// TODO; find all uses of that removed panel
// Add the Verify functionality to this view
Whisper.KeyVerificationPanelView = Whisper.View.extend({ Whisper.KeyVerificationPanelView = Whisper.View.extend({
className: 'key-verification panel', className: 'key-verification panel',
templateName: 'key-verification', templateName: 'key-verification',
events: {
'click button.verify': 'toggleVerified',
},
initialize: function(options) { initialize: function(options) {
this.our_number = textsecure.storage.user.getNumber(); this.our_number = textsecure.storage.user.getNumber();
if (options.newKey) { if (options.newKey) {
this.their_key = options.newKey; this.their_key = options.newKey;
} }
Promise.all([ Promise.all([
this.loadTheirKey(), this.loadTheirKey(),
this.loadOurKey(), this.loadOurKey(),
]).then(this.generateSecurityNumber.bind(this)) ]).then(this.generateSecurityNumber.bind(this))
.then(function() {
this.listenTo(this.model, 'change', this.render);
}.bind(this))
.then(this.render.bind(this)); .then(this.render.bind(this));
//.then(this.makeQRCode.bind(this)); //.then(this.makeQRCode.bind(this));
}, },
makeQRCode: function() { makeQRCode: function() {
// Per Lilia: We can't turn this on until it geneates a Latin1 string, as is // Per Lilia: We can't turn this on until it generates a Latin1 string, as is
// required by the mobile clients. // required by the mobile clients.
new QRCode(this.$('.qr')[0]).makeCode( new QRCode(this.$('.qr')[0]).makeCode(
dcodeIO.ByteBuffer.wrap(this.our_key).toString('base64') dcodeIO.ByteBuffer.wrap(this.our_key).toString('base64')
@ -58,6 +63,9 @@
this.securityNumber = securityNumber; this.securityNumber = securityNumber;
}.bind(this)); }.bind(this));
}, },
toggleVerified: function() {
this.model.toggleVerified();
},
render_attributes: function() { render_attributes: function() {
var s = this.securityNumber; var s = this.securityNumber;
var chunks = []; var chunks = [];
@ -67,10 +75,15 @@
var yourSafetyNumberWith = i18n( var yourSafetyNumberWith = i18n(
'yourSafetyNumberWith', this.model.getTitle() 'yourSafetyNumberWith', this.model.getTitle()
); );
console.log('this.model',this.model);
var verifyButton = this.model.isVerified() ? i18n('markAsNotVerified') : i18n('verify');
return { return {
learnMore : i18n('learnMore'), learnMore : i18n('learnMore'),
their_key_unknown : i18n('theirIdentityUnknown'), their_key_unknown : i18n('theirIdentityUnknown'),
yourSafetyNumberWith : i18n('yourSafetyNumberWith', this.model.getTitle()), yourSafetyNumberWith : i18n('yourSafetyNumberWith', this.model.getTitle()),
verifyHelp : i18n('verifyHelp', this.model.getTitle()),
verifyButton : verifyButton,
has_their_key : this.their_key !== undefined, has_their_key : this.their_key !== undefined,
chunks : chunks, chunks : chunks,
}; };

View file

@ -13,13 +13,15 @@
tagName: 'ul', tagName: 'ul',
itemView: Backbone.View, itemView: Backbone.View,
initialize: function(options) { initialize: function(options) {
this.options = options || {};
this.listenTo(this.collection, 'add', this.addOne); this.listenTo(this.collection, 'add', this.addOne);
this.listenTo(this.collection, 'reset', this.addAll); this.listenTo(this.collection, 'reset', this.addAll);
}, },
addOne: function(model) { addOne: function(model) {
if (this.itemView) { if (this.itemView) {
var view = new this.itemView({model: model}); var options = Object.assign({}, this.options.toInclude, {model: model});
var view = new this.itemView(options);
this.$el.append(view.render().el); this.$el.append(view.render().el);
this.$el.trigger('add'); this.$el.trigger('add');
} }

View file

@ -14,6 +14,13 @@
padding: 0 5px 0 4px; padding: 0 5px 0 4px;
} }
} }
.conversation-title .verified {
&:before {
content:"\00b7"; // &middot
font-weight: bold;
padding: 0 5px 0 4px;
}
}
.conversation { .conversation {
background-color: white; background-color: white;
@ -101,6 +108,15 @@
max-width: 100%; max-width: 100%;
} }
} }
div.verify {
text-align: center;
}
button.verify {
border-radius: 5px;
font-weight: bold;
padding: 10px;
}
} }
.message-detail { .message-detail {

View file

@ -342,6 +342,7 @@ $avatar-size: 44px;
margin: 0 0 0 $left-margin; margin: 0 0 0 $left-margin;
width: calc(100% - #{$avatar-size} - #{$left-margin} - #{(4/14) + em}); width: calc(100% - #{$avatar-size} - #{$left-margin} - #{(4/14) + em});
text-align: left; text-align: left;
cursor: pointer;
p { p {
overflow-x: hidden; overflow-x: hidden;
@ -361,6 +362,10 @@ $avatar-size: 44px;
color: $grey; color: $grey;
font-size: $font-size-small; font-size: $font-size-small;
} }
&.not-clickable {
cursor: default;
}
} }

View file

@ -332,7 +332,8 @@ button.hamburger {
display: inline-block; display: inline-block;
margin: 0 0 0 8px; margin: 0 0 0 8px;
width: calc(100% - 44px - 8px - 0.28571em); width: calc(100% - 44px - 8px - 0.28571em);
text-align: left; } text-align: left;
cursor: pointer; }
.contact-details p { .contact-details p {
overflow-x: hidden; overflow-x: hidden;
text-overflow: ellipsis; } text-overflow: ellipsis; }
@ -346,6 +347,8 @@ button.hamburger {
.contact-details .number { .contact-details .number {
color: #616161; color: #616161;
font-size: 0.92857em; } font-size: 0.92857em; }
.contact-details.not-clickable {
cursor: default; }
.recipients-input { .recipients-input {
position: relative; } position: relative; }
@ -988,6 +991,11 @@ input.search {
font-weight: bold; font-weight: bold;
padding: 0 5px 0 4px; } padding: 0 5px 0 4px; }
.conversation-title .verified:before {
content: "\00b7";
font-weight: bold;
padding: 0 5px 0 4px; }
.conversation { .conversation {
background-color: white; background-color: white;
height: 100%; } height: 100%; }
@ -1052,6 +1060,12 @@ input.search {
.key-verification .qr img { .key-verification .qr img {
display: inline-block; display: inline-block;
max-width: 100%; } max-width: 100%; }
.key-verification div.verify {
text-align: center; }
.key-verification button.verify {
border-radius: 5px;
font-weight: bold;
padding: 10px; }
.message-detail { .message-detail {
background-color: #eee; } background-color: #eee; }

View file

@ -92,14 +92,15 @@
<div class='conversation-menu menu'> <div class='conversation-menu menu'>
<button class='hamburger' alt='conversation menu'></button> <button class='hamburger' alt='conversation menu'></button>
<ul class='menu-list'> <ul class='menu-list'>
<li class='disappearing-messages'>{{ disappearing-messages }}</li>
{{#group}} {{#group}}
<li class='view-members'>{{ view-members }}</li> <li class='show-members'>{{ show-members }}</li>
<!-- <li class='update-group'>Update group</li> --> <!-- <li class='update-group'>Update group</li> -->
<!-- <li class='leave-group'>Leave group</li> --> <!-- <li class='leave-group'>Leave group</li> -->
{{/group}} {{/group}}
{{^group}} {{^group}}
<li class='end-session'>{{ end-session }}</li>
<li class='show-identity'>{{ show-identity }}</li> <li class='show-identity'>{{ show-identity }}</li>
<li class='end-session'>{{ end-session }}</li>
{{/group}} {{/group}}
<li class='destroy'>{{ destroy }}</li> <li class='destroy'>{{ destroy }}</li>
</ul> </ul>