feat: more discrete commit-on-save instead of commit-on-change w/ confirm modals (#8541)

* feat: privileges save button, #8537, WIP

* fix: disable firefox autocomplete on privilege form fields

* feat: closes #8537 privilege changes commit on save

- new language strings for confirmation and success modals/toasts
- indeterminate privilege handling (/cc @psychobunny)
- added new discard button
- both discard and save buttons now have confirmation dialogs

* fix(tests): remove duplicate template helper test

* fix(tests): broken template helper test

* feat: confirm dialogs for all privilege copy actions

Also, ability to add user to a privilege table without needing
to refresh the privilege table.

* feat: group row addition w/o table refresh

breaking: helpers.getUserPrivileges and helpers.getGroupPrivileges
no longer make socket calls to the following hooks:

- filter:privileges.list, filter:privileges.admin.list,
  filter:privileges.global.list, filter:privileges.groups.list,
  filter:privileges.admin.groups.list,
  filter:privileges.gloval.groups.list

The filters are still called, but done before the helper method
is called, and the results are passed in instead. This change
should only affect you if you directly call the helper methods,
otherwise the change is transparent.

* fix: stale ajaxify data on privilege category switch

* fix: implicit privileges not showing for user privs

* fix: groups, not group, also fix tests

* fix(tests): again

* fix: wrong tpl rendered when adding group to global priv table
This commit is contained in:
Julian Lam
2020-08-03 20:42:45 -04:00
committed by GitHub
parent 0f10e0836b
commit a716a5529c
15 changed files with 211 additions and 107 deletions

View File

@@ -27,6 +27,7 @@
"object-shorthand": "off", "object-shorthand": "off",
"prefer-arrow-callback": "off", "prefer-arrow-callback": "off",
"prefer-spread": "off", "prefer-spread": "off",
"prefer-object-spread": "off",
"prefer-reflect": "off", "prefer-reflect": "off",
"prefer-template": "off" "prefer-template": "off"
}, },

View File

@@ -66,7 +66,6 @@
"alert.create-success": "Category successfully created!", "alert.create-success": "Category successfully created!",
"alert.none-active": "You have no active categories.", "alert.none-active": "You have no active categories.",
"alert.create": "Create a Category", "alert.create": "Create a Category",
"alert.confirm-moderate": "<strong>Are you sure you wish to grant the moderation privilege to this user group?</strong> This group is public, and any users can join at will.",
"alert.confirm-purge": "<p class=\"lead\">Do you really want to purge this category \"%1\"?</p><h5><strong class=\"text-danger\">Warning!</strong> All topics and posts in this category will be purged!</h5> <p class=\"help-block\">Purging a category will remove all topics and posts, and delete the category from the database. If you want to remove a category <em>temporarily</em>, you'll want to \"disable\" the category instead.</p>", "alert.confirm-purge": "<p class=\"lead\">Do you really want to purge this category \"%1\"?</p><h5><strong class=\"text-danger\">Warning!</strong> All topics and posts in this category will be purged!</h5> <p class=\"help-block\">Purging a category will remove all topics and posts, and delete the category from the database. If you want to remove a category <em>temporarily</em>, you'll want to \"disable\" the category instead.</p>",
"alert.purge-success": "Category purged!", "alert.purge-success": "Category purged!",
"alert.copy-success": "Settings Copied!", "alert.copy-success": "Settings Copied!",

View File

@@ -1,6 +1,5 @@
{ {
"global": "Global", "global": "Global",
"global.no-users": "No user-specific global privileges.",
"admin": "Admin", "admin": "Admin",
"group-privileges": "Group Privileges", "group-privileges": "Group Privileges",
"user-privileges": "User Privileges", "user-privileges": "User Privileges",
@@ -38,5 +37,16 @@
"admin-categories": "Categories", "admin-categories": "Categories",
"admin-privileges": "Privileges", "admin-privileges": "Privileges",
"admin-users": "Users", "admin-users": "Users",
"admin-settings": "Settings" "admin-settings": "Settings",
"alert.confirm-moderate": "<strong>Are you sure you wish to grant the moderation privilege to this user group?</strong> This group is public, and any users can join at will.",
"alert.confirm-save": "Please confirm your intention to save these privileges",
"alert.saved": "Privilege changes saved and applied",
"alert.confirm-discard": "Are you sure you wish to discard your privilege changes?",
"alert.discarded": "Privilege changes discarded",
"alert.confirm-copyToAll": "Are you sure you wish to apply this privilege set to <strong>all categories</strong>?",
"alert.confirm-copyToAllGroup": "Are you sure you wish to apply this group's privilege set to <strong>all categories</strong>?",
"alert.confirm-copyToChildren": "Are you sure you wish to apply this privilege set to <strong>all descendant (child) categories</strong>?",
"alert.confirm-copyToChildrenGroup": "Are you sure you wish to apply this group's privilege set to <strong>all descendant (child) categories</strong>?",
"alert.no-undo": "<em>This action cannot be undone.</em>"
} }

View File

@@ -1,11 +1,3 @@
.page-manage-privileges {
.ui-autocomplete {
height: 500px;
overflow-y: auto;
overflow-x: hidden;
}
}
.page-admin-privileges { .page-admin-privileges {
@keyframes fadeOut { @keyframes fadeOut {
0% {background-color: @brand-primary;} 0% {background-color: @brand-primary;}
@@ -18,4 +10,23 @@
animation-fill-mode: both; animation-fill-mode: both;
animation-timing-function: ease-out; animation-timing-function: ease-out;
} }
.privilege-table {
td[data-delta="true"] > input {
&:after {
border-color: @state-success-text;
background-color: @state-success-text;
}
}
td[data-delta="false"] > input {
&:after {
border-color: @state-danger-bg;
}
&:indeterminate:after {
background-color: @state-danger-bg;
}
}
}
} }

View File

@@ -847,6 +847,17 @@ paths:
properties: properties:
name: name:
type: string type: string
keys:
type: object
properties:
users:
type: array
items:
type: string
groups:
type: array
items:
type: string
users: users:
type: array type: array
items: items:

View File

@@ -28,38 +28,72 @@ define('admin/manage/privileges', [
Privileges.setupPrivilegeTable = function () { Privileges.setupPrivilegeTable = function () {
$('.privilege-table-container').on('change', 'input[type="checkbox"]', function () { $('.privilege-table-container').on('change', 'input[type="checkbox"]', function () {
var checkboxEl = $(this); var checkboxEl = $(this);
var privilege = checkboxEl.parent().attr('data-privilege'); var wrapperEl = checkboxEl.parent();
var privilege = wrapperEl.attr('data-privilege');
var state = checkboxEl.prop('checked'); var state = checkboxEl.prop('checked');
var rowEl = checkboxEl.parents('tr'); var rowEl = checkboxEl.parents('tr');
var member = rowEl.attr('data-group-name') || rowEl.attr('data-uid'); var member = rowEl.attr('data-group-name') || rowEl.attr('data-uid');
var isPrivate = parseInt(rowEl.attr('data-private') || 0, 10); var isPrivate = parseInt(rowEl.attr('data-private') || 0, 10);
var isGroup = rowEl.attr('data-group-name') !== undefined; var isGroup = rowEl.attr('data-group-name') !== undefined;
var delta = checkboxEl.prop('checked') === (wrapperEl.attr('data-value') === 'true') ? null : state;
if (member) { if (member) {
if (isGroup && privilege === 'groups:moderate' && !isPrivate && state) { if (isGroup && privilege === 'groups:moderate' && !isPrivate && state) {
bootbox.confirm('[[admin/manage/categories:alert.confirm-moderate]]', function (confirm) { bootbox.confirm('[[admin/manage/privileges:alert.confirm-moderate]]', function (confirm) {
if (confirm) { if (confirm) {
Privileges.setPrivilege(member, privilege, state, checkboxEl); wrapperEl.attr('data-delta', delta);
Privileges.exposeAssumedPrivileges();
} else { } else {
checkboxEl.prop('checked', !checkboxEl.prop('checked')); checkboxEl.prop('checked', !checkboxEl.prop('checked'));
} }
}); });
} else { } else {
Privileges.setPrivilege(member, privilege, state, checkboxEl); wrapperEl.attr('data-delta', delta);
Privileges.exposeAssumedPrivileges();
} }
} else { } else {
app.alertError('[[error:invalid-data]]'); app.alertError('[[error:invalid-data]]');
} }
}); });
document.getElementById('save').addEventListener('click', function () {
bootbox.confirm('[[admin/manage/privileges:alert.confirm-save]]', function (ok) {
if (ok) {
var tableEl = document.querySelector('.privilege-table-container');
var requests = tableEl.querySelectorAll('td[data-delta]').forEach(function (el) {
var privilege = el.getAttribute('data-privilege');
var rowEl = el.parentNode;
var member = rowEl.getAttribute('data-group-name') || rowEl.getAttribute('data-uid');
var state = el.getAttribute('data-delta') === 'true' ? 1 : 0;
Privileges.setPrivilege(member, privilege, state);
});
$.when(requests).done(function () {
Privileges.refreshPrivilegeTable();
app.alertSuccess('[[admin/manage/privileges:alert.saved]]');
});
}
});
});
document.getElementById('discard').addEventListener('click', function () {
bootbox.confirm('[[admin/manage/privileges:alert.confirm-discard]]', function (ok) {
if (ok) {
Privileges.refreshPrivilegeTable();
app.alertSuccess('[[admin/manage/privileges:alert.discarded]]');
}
});
});
$('.privilege-table-container').on('click', '[data-action="search.user"]', Privileges.addUserToPrivilegeTable); $('.privilege-table-container').on('click', '[data-action="search.user"]', Privileges.addUserToPrivilegeTable);
$('.privilege-table-container').on('click', '[data-action="search.group"]', Privileges.addGroupToPrivilegeTable); $('.privilege-table-container').on('click', '[data-action="search.group"]', Privileges.addGroupToPrivilegeTable);
$('.privilege-table-container').on('click', '[data-action="copyToChildren"]', function () { $('.privilege-table-container').on('click', '[data-action="copyToChildren"]', function () {
Privileges.copyPrivilegesToChildren(cid, ''); throwConfirmModal('copyToChildren', Privileges.copyPrivilegesToChildren.bind(null, cid, ''));
}); });
$('.privilege-table-container').on('click', '[data-action="copyToChildrenGroup"]', function () { $('.privilege-table-container').on('click', '[data-action="copyToChildrenGroup"]', function () {
var groupName = $(this).parents('[data-group-name]').attr('data-group-name'); var groupName = $(this).parents('[data-group-name]').attr('data-group-name');
Privileges.copyPrivilegesToChildren(cid, groupName); throwConfirmModal('copyToChildrenGroup', Privileges.copyPrivilegesToChildren.bind(null, cid, groupName));
}); });
$('.privilege-table-container').on('click', '[data-action="copyPrivilegesFrom"]', function () { $('.privilege-table-container').on('click', '[data-action="copyPrivilegesFrom"]', function () {
@@ -71,13 +105,21 @@ define('admin/manage/privileges', [
}); });
$('.privilege-table-container').on('click', '[data-action="copyToAll"]', function () { $('.privilege-table-container').on('click', '[data-action="copyToAll"]', function () {
Privileges.copyPrivilegesToAllCategories(cid, ''); throwConfirmModal('copyToAll', Privileges.copyPrivilegesToAllCategories.bind(null, cid, ''));
}); });
$('.privilege-table-container').on('click', '[data-action="copyToAllGroup"]', function () { $('.privilege-table-container').on('click', '[data-action="copyToAllGroup"]', function () {
var groupName = $(this).parents('[data-group-name]').attr('data-group-name'); var groupName = $(this).parents('[data-group-name]').attr('data-group-name');
Privileges.copyPrivilegesToAllCategories(cid, groupName); throwConfirmModal('copyToAllGroup', Privileges.copyPrivilegesToAllCategories.bind(null, cid, groupName));
}); });
function throwConfirmModal(method, onConfirm) {
bootbox.confirm('[[admin/manage/privileges:alert.confirm-' + method + ']]<br /><br />[[admin/manage/privileges:alert.no-undo]]', function (ok) {
if (ok) {
onConfirm.call();
}
});
}
Privileges.exposeAssumedPrivileges(); Privileges.exposeAssumedPrivileges();
}; };
@@ -87,6 +129,7 @@ define('admin/manage/privileges', [
return app.alertError(err.message); return app.alertError(err.message);
} }
ajaxify.data.privileges = privileges;
var tpl = parseInt(cid, 10) ? 'admin/partials/privileges/category' : 'admin/partials/privileges/global'; var tpl = parseInt(cid, 10) ? 'admin/partials/privileges/category' : 'admin/partials/privileges/global';
Benchpress.parse(tpl, { Benchpress.parse(tpl, {
privileges: privileges, privileges: privileges,
@@ -113,8 +156,18 @@ define('admin/manage/privileges', [
privs.push(el.getAttribute('data-privilege')); privs.push(el.getAttribute('data-privilege'));
} }
}); });
// Also apply to non-group privileges
privs = privs.concat(privs.map(function (priv) {
if (priv.startsWith('groups:')) {
return priv.slice(7);
}
return false;
})).filter(Boolean);
for (var x = 0, numPrivs = privs.length; x < numPrivs; x += 1) { for (var x = 0, numPrivs = privs.length; x < numPrivs; x += 1) {
var inputs = $('.privilege-table tr[data-group-name]:not([data-group-name="registered-users"],[data-group-name="guests"],[data-group-name="spiders"]) td[data-privilege="' + privs[x] + '"] input'); var inputs = $('.privilege-table tr[data-group-name]:not([data-group-name="registered-users"],[data-group-name="guests"],[data-group-name="spiders"]) td[data-privilege="' + privs[x] + '"] input, .privilege-table tr[data-uid] td[data-privilege="' + privs[x] + '"] input');
inputs.each(function (idx, el) { inputs.each(function (idx, el) {
if (!el.checked) { if (!el.checked) {
el.indeterminate = true; el.indeterminate = true;
@@ -123,7 +176,9 @@ define('admin/manage/privileges', [
} }
}; };
Privileges.setPrivilege = function (member, privilege, state, checkboxEl) { Privileges.setPrivilege = function (member, privilege, state) {
var deferred = $.Deferred();
socket.emit('admin.categories.setPrivilege', { socket.emit('admin.categories.setPrivilege', {
cid: isNaN(cid) ? 0 : cid, cid: isNaN(cid) ? 0 : cid,
privilege: privilege, privilege: privilege,
@@ -131,12 +186,14 @@ define('admin/manage/privileges', [
member: member, member: member,
}, function (err) { }, function (err) {
if (err) { if (err) {
deferred.reject(err);
return app.alertError(err.message); return app.alertError(err.message);
} }
checkboxEl.replaceWith('<i class="fa fa-spin fa-spinner"></i>'); deferred.resolve();
Privileges.refreshPrivilegeTable();
}); });
return deferred.promise();
}; };
Privileges.addUserToPrivilegeTable = function () { Privileges.addUserToPrivilegeTable = function () {
@@ -151,23 +208,29 @@ define('admin/manage/privileges', [
inputEl.focus(); inputEl.focus();
autocomplete.user(inputEl, function (ev, ui) { autocomplete.user(inputEl, function (ev, ui) {
var defaultPrivileges; // Generate data for new row
if (ajaxify.data.url === '/admin/manage/privileges/admin') { var privilegeSet = ajaxify.data.privileges.keys.users.reduce(function (memo, cur) {
defaultPrivileges = ['admin:dashboard']; memo[cur] = false;
} else { return memo;
defaultPrivileges = cid ? ['find', 'read', 'topics:read'] : ['chat']; }, {});
}
socket.emit('admin.categories.setPrivilege', {
cid: isNaN(cid) ? 0 : cid,
privilege: defaultPrivileges,
set: true,
member: ui.item.user.uid,
}, function (err) {
if (err) {
return app.alertError(err.message);
}
Privileges.refreshPrivilegeTable(); app.parseAndTranslate('admin/partials/privileges/' + (isNaN(cid) ? 'global' : 'category'), 'privileges.users', {
privileges: {
users: [
{
picture: ui.item.user.picture,
username: ui.item.user.username,
uid: ui.item.user.uid,
'icon:text': ui.item.user['icon:text'],
'icon:bgColor': ui.item.user['icon:bgColor'],
privileges: privilegeSet,
},
],
},
}, function (html) {
var tableEl = document.querySelectorAll('.privilege-table');
var rows = tableEl[1].querySelectorAll('tbody tr');
html.insertBefore(rows[rows.length - 1]);
modal.modal('hide'); modal.modal('hide');
}); });
}); });
@@ -245,24 +308,28 @@ define('admin/manage/privileges', [
} }
function addGroupToCategory(group, cb) { function addGroupToCategory(group, cb) {
var defaultPrivileges; // Generate data for new row
if (ajaxify.data.url === '/admin/manage/privileges/admin') { var privilegeSet = ajaxify.data.privileges.keys.groups.reduce(function (memo, cur) {
defaultPrivileges = ['groups:admin:dashboard']; memo[cur] = false;
} else { return memo;
defaultPrivileges = cid ? ['groups:find', 'groups:read', 'groups:topics:read'] : ['groups:chat']; }, {});
}
socket.emit('admin.categories.setPrivilege', { app.parseAndTranslate('admin/partials/privileges/' + ((isNaN(cid) || cid === 0) ? 'global' : 'category'), 'privileges.groups', {
cid: isNaN(cid) ? 0 : cid, privileges: {
privilege: defaultPrivileges, groups: [
set: true, {
member: group, name: group,
}, function (err) { nameEscaped: translator.escape(group),
if (err) { privileges: privilegeSet,
return app.alertError(err.message); },
} ],
},
}, function (html) {
var tableEl = document.querySelector('.privilege-table');
var rows = tableEl.querySelectorAll('tbody tr');
html.insertBefore(rows[rows.length - 1]);
Privileges.exposeAssumedPrivileges();
Privileges.refreshPrivilegeTable(group);
if (typeof cb === 'function') { if (typeof cb === 'function') {
cb(); cb();
} }

View File

@@ -193,7 +193,7 @@
(member === 'spiders' && !spidersEnabled.includes(priv.name)) || (member === 'spiders' && !spidersEnabled.includes(priv.name)) ||
(member === 'Global Moderators' && globalModDisabled.includes(priv.name)); (member === 'Global Moderators' && globalModDisabled.includes(priv.name));
return '<td class="text-center" data-privilege="' + priv.name + '"><input type="checkbox"' + (priv.state ? ' checked' : '') + (disabled ? ' disabled="disabled"' : '') + ' /></td>'; return '<td class="text-center" data-privilege="' + priv.name + '" data-value="' + priv.state + '"><input autocomplete="off" type="checkbox"' + (priv.state ? ' checked' : '') + (disabled ? ' disabled="disabled"' : '') + ' /></td>';
}).join(''); }).join('');
} }

View File

@@ -117,11 +117,19 @@ module.exports = function (privileges) {
groups: plugins.fireHook('filter:privileges.admin.groups.list_human', privileges.admin.privilegeLabels.slice()), groups: plugins.fireHook('filter:privileges.admin.groups.list_human', privileges.admin.privilegeLabels.slice()),
}); });
} }
const keys = await utils.promiseParallel({
users: plugins.fireHook('filter:privileges.admin.list', privileges.admin.userPrivilegeList.slice()),
groups: plugins.fireHook('filter:privileges.admin.groups.list', privileges.admin.groupPrivilegeList.slice()),
});
const payload = await utils.promiseParallel({ const payload = await utils.promiseParallel({
labels: getLabels(), labels: getLabels(),
users: helpers.getUserPrivileges(0, 'filter:privileges.admin.list', privileges.admin.userPrivilegeList), users: helpers.getUserPrivileges(0, keys.users),
groups: helpers.getGroupPrivileges(0, 'filter:privileges.admin.groups.list', privileges.admin.groupPrivilegeList), groups: helpers.getGroupPrivileges(0, keys.groups),
}); });
payload.keys = keys;
// This is a hack because I can't do {labels.users.length} to echo the count in templates.js // This is a hack because I can't do {labels.users.length} to echo the count in templates.js
payload.columnCount = payload.labels.users.length + 2; payload.columnCount = payload.labels.users.length + 2;
return payload; return payload;

View File

@@ -22,11 +22,17 @@ module.exports = function (privileges) {
}); });
} }
const keys = await utils.promiseParallel({
users: plugins.fireHook('filter:privileges.list', privileges.userPrivilegeList.slice()),
groups: plugins.fireHook('filter:privileges.groups.list', privileges.groupPrivilegeList.slice()),
});
const payload = await utils.promiseParallel({ const payload = await utils.promiseParallel({
labels: getLabels(), labels: getLabels(),
users: helpers.getUserPrivileges(cid, 'filter:privileges.list', privileges.userPrivilegeList), users: helpers.getUserPrivileges(cid, keys.users),
groups: helpers.getGroupPrivileges(cid, 'filter:privileges.groups.list', privileges.groupPrivilegeList), groups: helpers.getGroupPrivileges(cid, keys.groups),
}); });
payload.keys = keys;
// This is a hack because I can't do {labels.users.length} to echo the count in templates.js // This is a hack because I can't do {labels.users.length} to echo the count in templates.js
payload.columnCountUser = payload.labels.users.length + 2; payload.columnCountUser = payload.labels.users.length + 2;

View File

@@ -55,11 +55,19 @@ module.exports = function (privileges) {
groups: plugins.fireHook('filter:privileges.global.groups.list_human', privileges.global.privilegeLabels.slice()), groups: plugins.fireHook('filter:privileges.global.groups.list_human', privileges.global.privilegeLabels.slice()),
}); });
} }
const keys = await utils.promiseParallel({
users: plugins.fireHook('filter:privileges.global.list', privileges.global.userPrivilegeList.slice()),
groups: plugins.fireHook('filter:privileges.global.groups.list', privileges.global.groupPrivilegeList.slice()),
});
const payload = await utils.promiseParallel({ const payload = await utils.promiseParallel({
labels: getLabels(), labels: getLabels(),
users: helpers.getUserPrivileges(0, 'filter:privileges.global.list', privileges.global.userPrivilegeList), users: helpers.getUserPrivileges(0, keys.users),
groups: helpers.getGroupPrivileges(0, 'filter:privileges.global.groups.list', privileges.global.groupPrivilegeList), groups: helpers.getGroupPrivileges(0, keys.groups),
}); });
payload.keys = keys;
// This is a hack because I can't do {labels.users.length} to echo the count in templates.js // This is a hack because I can't do {labels.users.length} to echo the count in templates.js
payload.columnCount = payload.labels.users.length + 2; payload.columnCount = payload.labels.users.length + 2;
return payload; return payload;

View File

@@ -88,8 +88,7 @@ async function isSystemGroupAllowedToPrivileges(privileges, uid, cid) {
return await groups.isMemberOfGroups(uidToSystemGroup[uid], groupKeys); return await groups.isMemberOfGroups(uidToSystemGroup[uid], groupKeys);
} }
helpers.getUserPrivileges = async function (cid, hookName, userPrivilegeList) { helpers.getUserPrivileges = async function (cid, userPrivileges) {
const userPrivileges = await plugins.fireHook(hookName, userPrivilegeList.slice());
let memberSets = await groups.getMembersOfGroups(userPrivileges.map(privilege => 'cid:' + cid + ':privileges:' + privilege)); let memberSets = await groups.getMembersOfGroups(userPrivileges.map(privilege => 'cid:' + cid + ':privileges:' + privilege));
memberSets = memberSets.map(function (set) { memberSets = memberSets.map(function (set) {
return set.map(uid => parseInt(uid, 10)); return set.map(uid => parseInt(uid, 10));
@@ -108,8 +107,7 @@ helpers.getUserPrivileges = async function (cid, hookName, userPrivilegeList) {
return memberData; return memberData;
}; };
helpers.getGroupPrivileges = async function (cid, hookName, groupPrivilegeList) { helpers.getGroupPrivileges = async function (cid, groupPrivileges) {
const groupPrivileges = await plugins.fireHook(hookName, groupPrivilegeList.slice());
const [memberSets, allGroupNames] = await Promise.all([ const [memberSets, allGroupNames] = await Promise.all([
groups.getMembersOfGroups(groupPrivileges.map(privilege => 'cid:' + cid + ':privileges:' + privilege)), groups.getMembersOfGroups(groupPrivileges.map(privilege => 'cid:' + cid + ':privileges:' + privilege)),
groups.getGroups('groups:createtime', 0, -1), groups.getGroups('groups:createtime', 0, -1),

View File

@@ -20,3 +20,13 @@
</div> </div>
</form> </form>
</div> </div>
<div class="floating-button">
<button id="discard" class="mdl-button mdl-js-button mdl-button--fab mdl-js-ripple-effect mdl-button--colored">
<i class="material-icons">delete</i>
</button>
<button id="save" class="mdl-button mdl-js-button mdl-button--fab mdl-js-ripple-effect mdl-button--colored primary">
<i class="material-icons">save</i>
</button>
</div>

View File

@@ -52,16 +52,20 @@
<tr> <tr>
<td colspan="{privileges.columnCountGroup}"> <td colspan="{privileges.columnCountGroup}">
<div class="btn-toolbar"> <div class="btn-toolbar">
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.group"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="search.group">
<i class="fa fa-users"></i>
[[admin/manage/categories:privileges.search-group]] [[admin/manage/categories:privileges.search-group]]
</button> </button>
<button type="button" class="btn btn-info pull-right" data-ajaxify="false" data-action="copyPrivilegesFrom"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="copyPrivilegesFrom">
<i class="fa fa-copy"></i>
[[admin/manage/categories:privileges.copy-from-category]] [[admin/manage/categories:privileges.copy-from-category]]
</button> </button>
<button type="button" class="btn btn-info pull-right" data-ajaxify="false" data-action="copyToChildren"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="copyToChildren">
<i class="fa fa-copy"></i>
[[admin/manage/categories:privileges.copy-to-children]] [[admin/manage/categories:privileges.copy-to-children]]
</button> </button>
<button type="button" class="btn btn-info pull-right" data-ajaxify="false" data-action="copyToAll"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="copyToAll">
<i class="fa fa-copy"></i>
[[admin/manage/categories:privileges.copy-privileges-to-all-categories]] [[admin/manage/categories:privileges.copy-privileges-to-all-categories]]
</button> </button>
</div> </div>
@@ -101,7 +105,6 @@
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
<!-- IF privileges.users.length -->
<!-- BEGIN privileges.users --> <!-- BEGIN privileges.users -->
<tr data-uid="{privileges.users.uid}"> <tr data-uid="{privileges.users.uid}">
<td> <td>
@@ -117,20 +120,11 @@
<!-- END privileges.users --> <!-- END privileges.users -->
<tr> <tr>
<td colspan="{privileges.columnCountUser}"> <td colspan="{privileges.columnCountUser}">
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.user"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="search.user">
<i class="fa fa-user"></i>
[[admin/manage/categories:privileges.search-user]] [[admin/manage/categories:privileges.search-user]]
</button> </button>
</td> </td>
</tr> </tr>
<!-- ELSE -->
<tr>
<td colspan="{privileges.columnCountUser}">
[[admin/manage/categories:privileges.no-users]]
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.user">
[[admin/manage/categories:privileges.search-user]]
</button>
</td>
</tr>
<!-- ENDIF privileges.users.length -->
</tbody> </tbody>
</table> </table>

View File

@@ -24,7 +24,8 @@
<tr> <tr>
<td colspan="{privileges.columnCount}"> <td colspan="{privileges.columnCount}">
<div class="btn-toolbar"> <div class="btn-toolbar">
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.group"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="search.group">
<i class="fa fa-users"></i>
[[admin/manage/categories:privileges.search-group]] [[admin/manage/categories:privileges.search-group]]
</button> </button>
</div> </div>
@@ -50,7 +51,6 @@
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
<!-- IF privileges.users.length -->
<!-- BEGIN privileges.users --> <!-- BEGIN privileges.users -->
<tr data-uid="{privileges.users.uid}"> <tr data-uid="{privileges.users.uid}">
<td> <td>
@@ -66,20 +66,11 @@
<!-- END privileges.users --> <!-- END privileges.users -->
<tr> <tr>
<td colspan="{privileges.columnCount}"> <td colspan="{privileges.columnCount}">
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.user"> <button type="button" class="btn btn-default pull-right" data-ajaxify="false" data-action="search.user">
<i class="fa fa-user"></i>
[[admin/manage/categories:privileges.search-user]] [[admin/manage/categories:privileges.search-user]]
</button> </button>
</td> </td>
</tr> </tr>
<!-- ELSE -->
<tr>
<td colspan="{privileges.columnCount}">
[[admin/manage/privileges:global.no-users]]
<button type="button" class="btn btn-primary pull-right" data-ajaxify="false" data-action="search.user">
[[admin/manage/categories:privileges.search-user]]
</button>
</td>
</tr>
<!-- ENDIF privileges.users.length -->
</tbody> </tbody>
</table> </table>

View File

@@ -148,17 +148,7 @@ describe('helpers', function () {
read: true, read: true,
}; };
var html = helpers.spawnPrivilegeStates('guests', privs); var html = helpers.spawnPrivilegeStates('guests', privs);
assert.equal(html, '<td class="text-center" data-privilege="find"><input type="checkbox" checked /></td><td class="text-center" data-privilege="read"><input type="checkbox" checked /></td>'); assert.equal(html, '<td class="text-center" data-privilege="find" data-value="true"><input autocomplete="off" type="checkbox" checked /></td><td class="text-center" data-privilege="read" data-value="true"><input autocomplete="off" type="checkbox" checked /></td>');
done();
});
it('should spawn privilege states', function (done) {
var privs = {
find: true,
read: true,
};
var html = helpers.spawnPrivilegeStates('guests', privs);
assert.equal(html, '<td class="text-center" data-privilege="find"><input type="checkbox" checked /></td><td class="text-center" data-privilege="read"><input type="checkbox" checked /></td>');
done(); done();
}); });