You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
421 lines
12 KiB
JavaScript
421 lines
12 KiB
JavaScript
"use strict";
|
|
/*global config, define, app, socket, ajaxify, bootbox, templates */
|
|
|
|
define('admin/manage/category', [
|
|
'uploader',
|
|
'iconSelect',
|
|
'admin/modules/colorpicker',
|
|
'autocomplete'
|
|
], function(uploader, iconSelect, colorpicker, autocomplete) {
|
|
var Category = {};
|
|
|
|
Category.init = function() {
|
|
var modified_categories = {};
|
|
|
|
function modified(el) {
|
|
var cid = $(el).parents('form').attr('data-cid');
|
|
|
|
if (cid) {
|
|
modified_categories[cid] = modified_categories[cid] || {};
|
|
modified_categories[cid][$(el).attr('data-name')] = $(el).val();
|
|
|
|
app.flags = app.flags || {};
|
|
app.flags._unsaved = true;
|
|
}
|
|
}
|
|
|
|
function save(e) {
|
|
e.preventDefault();
|
|
|
|
if(Object.keys(modified_categories).length) {
|
|
socket.emit('admin.categories.update', modified_categories, function(err, result) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
if (result && result.length) {
|
|
app.flags._unsaved = false;
|
|
app.alert({
|
|
title: 'Updated Categories',
|
|
message: 'Category IDs ' + result.join(', ') + ' was successfully updated.',
|
|
type: 'success',
|
|
timeout: 2000
|
|
});
|
|
}
|
|
});
|
|
modified_categories = {};
|
|
}
|
|
}
|
|
|
|
$('.blockclass, form.category select').each(function() {
|
|
var $this = $(this);
|
|
$this.val($this.attr('data-value'));
|
|
});
|
|
|
|
function enableColorPicker(idx, inputEl) {
|
|
var $inputEl = $(inputEl),
|
|
previewEl = $inputEl.parents('[data-cid]').find('.category-preview');
|
|
|
|
colorpicker.enable($inputEl, function(hsb, hex) {
|
|
if ($inputEl.attr('data-name') === 'bgColor') {
|
|
previewEl.css('background-color', '#' + hex);
|
|
} else if ($inputEl.attr('data-name') === 'color') {
|
|
previewEl.css('color', '#' + hex);
|
|
}
|
|
|
|
modified($inputEl[0]);
|
|
});
|
|
}
|
|
|
|
|
|
$('form.category input, form.category select')
|
|
.on('change', function(ev) {
|
|
modified(ev.target);
|
|
})
|
|
.on('keydown', function(ev) {
|
|
if (ev.which === 13) {
|
|
ev.preventDefault();
|
|
return false;
|
|
}
|
|
});
|
|
|
|
$('[data-name="imageClass"]').on('change', function() {
|
|
$('.category-preview').css('background-size', $(this).val());
|
|
});
|
|
|
|
$('[data-name="bgColor"], [data-name="color"]').each(enableColorPicker);
|
|
|
|
$('#save').on('click', save);
|
|
$('.purge').on('click', function(e) {
|
|
e.preventDefault();
|
|
|
|
bootbox.confirm('<p class="lead">Do you really want to purge this category "' + $('form.category').find('input[data-name="name"]').val() + '"?</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>', function(confirm) {
|
|
if (!confirm) {
|
|
return;
|
|
}
|
|
socket.emit('admin.categories.purge', ajaxify.data.category.cid, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
app.alertSuccess('Category purged!');
|
|
ajaxify.go('admin/manage/categories');
|
|
});
|
|
});
|
|
});
|
|
|
|
$('.copy-settings').on('click', function() {
|
|
selectCategoryModal(function(cid) {
|
|
socket.emit('admin.categories.copySettingsFrom', {fromCid: cid, toCid: ajaxify.data.category.cid}, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
app.alertSuccess('Settings Copied!');
|
|
ajaxify.refresh();
|
|
});
|
|
});
|
|
return false;
|
|
});
|
|
|
|
$('.upload-button').on('click', function() {
|
|
var inputEl = $(this);
|
|
var cid = inputEl.attr('data-cid');
|
|
|
|
uploader.show({
|
|
title: 'Upload category image',
|
|
route: config.relative_path + '/api/admin/category/uploadpicture',
|
|
params: {cid: cid}
|
|
}, function(imageUrlOnServer) {
|
|
$('#category-image').val(imageUrlOnServer);
|
|
var previewBox = inputEl.parent().parent().siblings('.category-preview');
|
|
previewBox.css('background', 'url(' + imageUrlOnServer + '?' + new Date().getTime() + ')');
|
|
|
|
modified($('#category-image'));
|
|
});
|
|
});
|
|
|
|
$('#category-image').on('change', function() {
|
|
$('.category-preview').css('background-image', $(this).val() ? ('url("' + $(this).val() + '")') : '');
|
|
});
|
|
|
|
$('.delete-image').on('click', function(e) {
|
|
e.preventDefault();
|
|
|
|
var inputEl = $('#category-image');
|
|
var previewBox = $('.category-preview');
|
|
|
|
inputEl.val('');
|
|
previewBox.css('background-image', '');
|
|
modified(inputEl[0]);
|
|
$(this).parent().addClass('hide').hide();
|
|
});
|
|
|
|
$('.category-preview').on('click', function() {
|
|
iconSelect.init($(this).find('i'), modified);
|
|
});
|
|
|
|
$('button[data-action="setParent"], button[data-action="changeParent"]').on('click', Category.launchParentSelector);
|
|
$('button[data-action="removeParent"]').on('click', function() {
|
|
var payload= {};
|
|
payload[ajaxify.data.category.cid] = {
|
|
parentCid: 0
|
|
};
|
|
|
|
socket.emit('admin.categories.update', payload, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
$('button[data-action="removeParent"]').parent().addClass('hide');
|
|
$('button[data-action="changeParent"]').parent().addClass('hide');
|
|
$('button[data-action="setParent"]').removeClass('hide');
|
|
});
|
|
});
|
|
|
|
Category.setupPrivilegeTable();
|
|
};
|
|
|
|
Category.setupPrivilegeTable = function() {
|
|
$('.privilege-table-container').on('change', 'input[type="checkbox"]', function() {
|
|
var checkboxEl = $(this),
|
|
privilege = checkboxEl.parent().attr('data-privilege'),
|
|
state = checkboxEl.prop('checked'),
|
|
rowEl = checkboxEl.parents('tr'),
|
|
member = rowEl.attr('data-group-name') || rowEl.attr('data-uid'),
|
|
isPrivate = parseInt(rowEl.attr('data-private') || 0, 10),
|
|
isGroup = rowEl.attr('data-group-name') !== undefined;
|
|
|
|
if (member) {
|
|
if (isGroup && privilege === 'groups:moderate' && !isPrivate && state) {
|
|
bootbox.confirm('<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.', function(confirm) {
|
|
if (confirm) {
|
|
Category.setPrivilege(member, privilege, state, checkboxEl);
|
|
} else {
|
|
checkboxEl.prop('checked', checkboxEl.prop('checked') ^ 1);
|
|
}
|
|
});
|
|
} else {
|
|
Category.setPrivilege(member, privilege, state, checkboxEl);
|
|
}
|
|
} else {
|
|
app.alertError('[[error:invalid-data]]');
|
|
}
|
|
});
|
|
|
|
$('.privilege-table-container').on('click', '[data-action="search.user"]', Category.addUserToPrivilegeTable);
|
|
$('.privilege-table-container').on('click', '[data-action="search.group"]', Category.addGroupToPrivilegeTable);
|
|
$('.privilege-table-container').on('click', '[data-action="copyToChildren"]', Category.copyPrivilegesToChildren);
|
|
$('.privilege-table-container').on('click', '[data-action="copyPrivilegesFrom"]', Category.copyPrivilegesFromCategory);
|
|
|
|
Category.exposeAssumedPrivileges();
|
|
};
|
|
|
|
Category.refreshPrivilegeTable = function() {
|
|
socket.emit('admin.categories.getPrivilegeSettings', ajaxify.data.category.cid, function(err, privileges) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
templates.parse('admin/partials/categories/privileges', {
|
|
privileges: privileges
|
|
}, function(html) {
|
|
$('.privilege-table-container').html(html);
|
|
Category.exposeAssumedPrivileges();
|
|
});
|
|
});
|
|
};
|
|
|
|
Category.exposeAssumedPrivileges = function() {
|
|
/*
|
|
If registered-users has a privilege enabled, then all users and groups of that privilege
|
|
should be assumed to have that privilege as well, even if not set in the db, so reflect
|
|
this arrangement in the table
|
|
*/
|
|
var privs = [];
|
|
$('.privilege-table tr[data-group-name="registered-users"] td input[type="checkbox"]').parent().each(function(idx, el) {
|
|
if ($(el).find('input').prop('checked')) {
|
|
privs.push(el.getAttribute('data-privilege'));
|
|
}
|
|
});
|
|
for(var x=0,numPrivs=privs.length;x<numPrivs;x++) {
|
|
var inputs = $('.privilege-table tr[data-group-name]:not([data-group-name="registered-users"],[data-group-name="guests"]) td[data-privilege="' + privs[x] + '"] input');
|
|
inputs.each(function(idx, el) {
|
|
if (!el.checked) {
|
|
el.indeterminate = true;
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
Category.setPrivilege = function(member, privilege, state, checkboxEl) {
|
|
socket.emit('admin.categories.setPrivilege', {
|
|
cid: ajaxify.data.category.cid,
|
|
privilege: privilege,
|
|
set: state,
|
|
member: member
|
|
}, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
checkboxEl.replaceWith('<i class="fa fa-spin fa-spinner"></i>');
|
|
Category.refreshPrivilegeTable();
|
|
});
|
|
};
|
|
|
|
Category.launchParentSelector = function() {
|
|
socket.emit('categories.get', function(err, categories) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
categories = categories.filter(function(category) {
|
|
return category && !category.disabled && parseInt(category.cid, 10) !== parseInt(ajaxify.data.category.cid, 10);
|
|
});
|
|
|
|
templates.parse('partials/category_list', {
|
|
categories: categories
|
|
}, function(html) {
|
|
var modal = bootbox.dialog({
|
|
message: html,
|
|
title: 'Set Parent Category'
|
|
});
|
|
|
|
modal.find('li[data-cid]').on('click', function() {
|
|
var parentCid = $(this).attr('data-cid'),
|
|
payload = {};
|
|
|
|
payload[ajaxify.data.category.cid] = {
|
|
parentCid: parentCid
|
|
};
|
|
|
|
socket.emit('admin.categories.update', payload, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
var parent = categories.filter(function(category) {
|
|
return category && parseInt(category.cid, 10) === parseInt(parentCid, 10);
|
|
});
|
|
parent = parent[0];
|
|
|
|
modal.modal('hide');
|
|
$('button[data-action="removeParent"]').parent().removeClass('hide');
|
|
$('button[data-action="setParent"]').addClass('hide');
|
|
var buttonHtml = '<i class="fa ' + parent.icon + '"></i> ' + parent.name;
|
|
$('button[data-action="changeParent"]').html(buttonHtml).parent().removeClass('hide');
|
|
});
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
Category.addUserToPrivilegeTable = function() {
|
|
var modal = bootbox.dialog({
|
|
title: 'Find a User',
|
|
message: '<input class="form-control input-lg" placeholder="Search for a user here..." />',
|
|
show: true
|
|
});
|
|
|
|
modal.on('shown.bs.modal', function() {
|
|
var inputEl = modal.find('input');
|
|
|
|
autocomplete.user(inputEl, function(ev, ui) {
|
|
socket.emit('admin.categories.setPrivilege', {
|
|
cid: ajaxify.data.category.cid,
|
|
privilege: ['find', 'read', 'topics:read'],
|
|
set: true,
|
|
member: ui.item.user.uid
|
|
}, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
Category.refreshPrivilegeTable();
|
|
modal.modal('hide');
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
Category.addGroupToPrivilegeTable = function() {
|
|
var modal = bootbox.dialog({
|
|
title: 'Find a Group',
|
|
message: '<input class="form-control input-lg" placeholder="Search for a group here..." />',
|
|
show: true
|
|
});
|
|
|
|
modal.on('shown.bs.modal', function() {
|
|
var inputEl = modal.find('input');
|
|
|
|
autocomplete.group(inputEl, function(ev, ui) {
|
|
socket.emit('admin.categories.setPrivilege', {
|
|
cid: ajaxify.data.category.cid,
|
|
privilege: ['groups:find', 'groups:read', 'groups:topics:read'],
|
|
set: true,
|
|
member: ui.item.group.name
|
|
}, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
Category.refreshPrivilegeTable();
|
|
modal.modal('hide');
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
Category.copyPrivilegesToChildren = function() {
|
|
socket.emit('admin.categories.copyPrivilegesToChildren', ajaxify.data.category.cid, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
app.alertSuccess('Privileges copied!');
|
|
});
|
|
};
|
|
|
|
Category.copyPrivilegesFromCategory = function() {
|
|
selectCategoryModal(function(cid) {
|
|
socket.emit('admin.categories.copyPrivilegesFrom', {toCid: ajaxify.data.category.cid, fromCid: cid}, function(err) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
ajaxify.refresh();
|
|
});
|
|
});
|
|
};
|
|
|
|
function selectCategoryModal(callback) {
|
|
socket.emit('admin.categories.getNames', function(err, categories) {
|
|
if (err) {
|
|
return app.alertError(err.message);
|
|
}
|
|
|
|
templates.parse('admin/partials/categories/select-category', {
|
|
categories: categories
|
|
}, function(html) {
|
|
function submit() {
|
|
var formData = modal.find('form').serializeObject();
|
|
callback(formData['select-cid']);
|
|
modal.modal('hide');
|
|
return false;
|
|
}
|
|
|
|
var modal = bootbox.dialog({
|
|
title: 'Select a Category',
|
|
message: html,
|
|
buttons: {
|
|
save: {
|
|
label: 'Copy',
|
|
className: 'btn-primary',
|
|
callback: submit
|
|
}
|
|
}
|
|
});
|
|
|
|
modal.find('form').on('submit', submit);
|
|
});
|
|
});
|
|
}
|
|
|
|
|
|
return Category;
|
|
}); |