diff --git a/public/language/en-GB/admin/manage/categories.json b/public/language/en-GB/admin/manage/categories.json new file mode 100644 index 0000000000..8bd5d5bfcf --- /dev/null +++ b/public/language/en-GB/admin/manage/categories.json @@ -0,0 +1,68 @@ +{ + "settings": "Category Settings", + "privileges": "Privileges", + + "name": "Category Name", + "description": "Category Description", + "bg-color": "Background Colour", + "text-color": "Text Colour", + "bg-image-size": "Background Image Size", + "custom-class": "Custom Class", + "num-recent-replies": "# of Recent Replies", + "ext-link": "External Link", + "upload-image": "Upload Image", + "delete-image": "Remove", + "category-image": "Category Image", + "parent-category": "Parent Category", + "optional-parent-category": "(Optional) Parent Category", + "parent-category-none": "(None)", + "copy-settings": "Copy Settings From", + "optional-clone-settings": "(Optional) Clone Settings From Category", + "purge": "Purge Category", + + "enable": "Enable", + "disable": "Disable", + "edit": "Edit", + + "select-category": "Select Category", + "set-parent-category": "Set Parent Category", + + "privileges.description": "You can configure the access control privileges for this category in this section. Privileges can be granted on a per-user or a per-group basis. You can add a new user to this table by searching for them in the form below.", + "privileges.warning": "Note: Privilege settings take effect immediately. It is not necessary to save the category after adjusting these settings.", + "privileges.section-viewing": "Viewing Privileges", + "privileges.section-posting": "Posting Privileges", + "privileges.section-moderation": "Moderation Privileges", + "privileges.section-user": "User", + "privileges.search-user": "Add User", + "privileges.no-users": "No user-specific privileges in this category.", + "privileges.section-group": "Group", + "privileges.group-private": "This group is private", + "privileges.search-group": "Add Group", + "privileges.copy-to-children": "Copy to Children", + "privileges.copy-from-category": "Copy from Category", + "privileges.inherit": "If the registered-users group is granted a specific privilege, all other groups receive an implicit privilege, even if they are not explicitly defined/checked. This implicit privilege is shown to you because all users are part of the registered-users user group, and so, privileges for additional groups need not be explicitly granted.", + + "analytics.back": "Back to Categories List", + "analytics.title": "Analytics for \"%1\" category", + "analytics.pageviews-hourly": "Figure 1 – Hourly page views for this category", + "analytics.pageviews-daily": "Figure 2 – Daily page views for this category", + "analytics.topics-daily": "Figure 3 – Daily topics created in this category", + "analytics.posts-daily": "Figure 4 – Daily posts made in this category", + + "alert.created": "Created", + "alert.create-success": "Category successfully created!", + "alert.none-active": "You have no active categories.", + "alert.create": "Create a Category", + "alert.confirm-moderate": "Are you sure you wish to grant the moderation privilege to this user group? This group is public, and any users can join at will.", + "alert.confirm-purge": "

Do you really want to purge this category \"%1\"?

Warning! All topics and posts in this category will be purged!

Purging a category will remove all topics and posts, and delete the category from the database. If you want to remove a category temporarily, you'll want to \"disable\" the category instead.

", + "alert.purge-success": "Category purged!", + "alert.copy-success": "Settings Copied!", + "alert.set-parent-category": "Set Parent Category", + "alert.updated": "Updated Categories", + "alert.updated-success": "Category IDs %1 was successfully updated.", + "alert.upload-image": "Upload category image", + "alert.find-user": "Find a User", + "alert.user-search": "Search for a user here...", + "alert.find-group": "Find a Group", + "alert.group-search": "Search for a group here..." +} \ No newline at end of file diff --git a/public/less/admin/manage/categories.less b/public/less/admin/manage/categories.less index 0f13af4deb..c0c0a544dc 100644 --- a/public/less/admin/manage/categories.less +++ b/public/less/admin/manage/categories.less @@ -103,6 +103,7 @@ div.categories { border-top: 0; text-transform: uppercase; font-size: 9px; + vertical-align: bottom; } .arrowed:after { diff --git a/public/src/admin/admin.js b/public/src/admin/admin.js index 217438e146..1c44a7e78d 100644 --- a/public/src/admin/admin.js +++ b/public/src/admin/admin.js @@ -87,7 +87,7 @@ url = url .replace(/\/\d+$/, '') .split('/').slice(0, 3).join('/') - .split('?')[0]; + .split('?')[0].replace(/(\/+$)|(^\/+)/, ''); // If index is requested, load the dashboard if (url === 'admin') { diff --git a/public/src/admin/manage/categories.js b/public/src/admin/manage/categories.js index 1042c723e6..acb55b82e9 100644 --- a/public/src/admin/manage/categories.js +++ b/public/src/admin/manage/categories.js @@ -40,31 +40,29 @@ define('admin/manage/categories', ['vendor/jquery/serializeObject/jquery.ba-seri templates.parse('admin/partials/categories/create', { categories: categories }, function (html) { - translator.translate(html, function (html) { - function submit() { - var formData = modal.find('form').serializeObject(); - formData.description = ''; - formData.icon = 'fa-comments'; - - Categories.create(formData); - modal.modal('hide'); - return false; - } + function submit() { + var formData = modal.find('form').serializeObject(); + formData.description = ''; + formData.icon = 'fa-comments'; + + Categories.create(formData); + modal.modal('hide'); + return false; + } - var modal = bootbox.dialog({ - title: 'Create a Category', - message: html, - buttons: { - save: { - label: 'Save', - className: 'btn-primary', - callback: submit - } + var modal = bootbox.dialog({ + title: '[[admin/manage/categories:alert.create]]', + message: html, + buttons: { + save: { + label: '[[global:save]]', + className: 'btn-primary', + callback: submit } - }); - - modal.find('form').on('submit', submit); + } }); + + modal.find('form').on('submit', submit); }); }); }; @@ -77,8 +75,8 @@ define('admin/manage/categories', ['vendor/jquery/serializeObject/jquery.ba-seri app.alert({ alert_id: 'category_created', - title: 'Created', - message: 'Category successfully created!', + title: '[[admin/manage/categories:alert.created]]', + message: '[[admin/manage/categories:alert.create-success]]', type: 'success', timeout: 2000 }); @@ -91,10 +89,12 @@ define('admin/manage/categories', ['vendor/jquery/serializeObject/jquery.ba-seri var container = $('.categories'); if (!categories || !categories.length) { - $('
') - .addClass('alert alert-info text-center') - .text('You have no active categories.') - .appendTo(container); + translator.translate('[[admin/manage/categories:alert.none-active]]', function (text) { + $('
') + .addClass('alert alert-info text-center') + .text(text) + .appendTo(container); + }); } else { sortables = {}; renderList(categories, container, 0); diff --git a/public/src/admin/manage/category.js b/public/src/admin/manage/category.js index 32da35b007..351c7a624b 100644 --- a/public/src/admin/manage/category.js +++ b/public/src/admin/manage/category.js @@ -12,6 +12,45 @@ define('admin/manage/category', [ var modified_categories = {}; 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, results) { + if (err) { + return app.alertError(err.message); + } + + if (results && results.length) { + app.flags._unsaved = false; + app.alert({ + title: '[[admin/manage/categories:alert.updated]]', + message: translator.compile( + 'admin/manage/categories:alert.updated-success', + results.join(', ') + ), + type: 'success', + timeout: 2000 + }); + } + }); + modified_categories = {}; + } + } $('.blockclass, form.category select').each(function () { var $this = $(this); @@ -76,7 +115,10 @@ define('admin/manage/category', [ $('.purge').on('click', function (e) { e.preventDefault(); - bootbox.confirm('

Do you really want to purge this category "' + $('form.category').find('input[data-name="name"]').val() + '"?

Warning! All topics and posts in this category will be purged!

Purging a category will remove all topics and posts, and delete the category from the database. If you want to remove a category temporarily, you\'ll want to "disable" the category instead.

', function (confirm) { + bootbox.confirm(translator.compile( + 'admin/manage/categories:alert.confirm-purge', + $('form.category').find('input[data-name="name"]').val() + ), function (confirm) { if (!confirm) { return; } @@ -84,7 +126,7 @@ define('admin/manage/category', [ if (err) { return app.alertError(err.message); } - app.alertSuccess('Category purged!'); + app.alertSuccess('[[admin/manage/categories:alert.purge-success]]'); ajaxify.go('admin/manage/categories'); }); }); @@ -96,7 +138,7 @@ define('admin/manage/category', [ if (err) { return app.alertError(err.message); } - app.alertSuccess('Settings Copied!'); + app.alertSuccess('[[admin/manage/categories:alert.copy-success]]'); ajaxify.refresh(); }); }); @@ -108,7 +150,7 @@ define('admin/manage/category', [ var cid = inputEl.attr('data-cid'); uploader.show({ - title: 'Upload category image', + title: '[[admin/manage/categories:alert.upload-image]]', route: config.relative_path + '/api/admin/category/uploadpicture', params: {cid: cid} }, function (imageUrlOnServer) { @@ -201,7 +243,7 @@ define('admin/manage/category', [ if (member) { if (isGroup && privilege === 'groups:moderate' && !isPrivate && state) { - bootbox.confirm('Are you sure you wish to grant the moderation privilege to this user group? This group is public, and any users can join at will.', function (confirm) { + bootbox.confirm('[[admin/manage/categories:alert.confirm-moderate]]', function (confirm) { if (confirm) { Category.setPrivilege(member, privilege, state, checkboxEl); } else { @@ -292,35 +334,33 @@ define('admin/manage/category', [ templates.parse('partials/category_list', { categories: categories }, function (html) { - translator.translate(html, function (html) { - var modal = bootbox.dialog({ - message: html, - title: 'Set Parent Category' - }); + var modal = bootbox.dialog({ + message: html, + title: '[[admin/manage/categories:alert.set-parent-category]]' + }); - modal.find('li[data-cid]').on('click', function () { - var parentCid = $(this).attr('data-cid'), - payload = {}; + modal.find('li[data-cid]').on('click', function () { + var parentCid = $(this).attr('data-cid'), + payload = {}; - payload[ajaxify.data.category.cid] = { - parentCid: parentCid - }; + 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 = ' ' + parent.name; - $('button[data-action="changeParent"]').html(buttonHtml).parent().removeClass('hide'); + 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 = ' ' + parent.name; + $('button[data-action="changeParent"]').html(buttonHtml).parent().removeClass('hide'); }); }); }); @@ -329,8 +369,8 @@ define('admin/manage/category', [ Category.addUserToPrivilegeTable = function () { var modal = bootbox.dialog({ - title: 'Find a User', - message: '', + title: '[[admin/manage/categories:alert.find-user]]', + message: '', show: true }); @@ -357,8 +397,8 @@ define('admin/manage/category', [ Category.addGroupToPrivilegeTable = function () { var modal = bootbox.dialog({ - title: 'Find a Group', - message: '', + title: '[[admin/manage/categories:alert.find-group]]', + message: '', show: true }); diff --git a/public/vendor/bootbox/wrapper.js b/public/vendor/bootbox/wrapper.js index bfc9457241..8e54e0e7db 100644 --- a/public/vendor/bootbox/wrapper.js +++ b/public/vendor/bootbox/wrapper.js @@ -30,15 +30,15 @@ require(['translator'], function (shim) { $elem = dialog.call(bootbox, options); - if (/\[\[[a-zA-Z0-9\-_.\/:]+\]\]/.test($elem[0].outerHTML)) { + if (/\[\[.+\]\]/.test($elem[0].outerHTML)) { nodes = descendantTextNodes($elem[0]); text = nodes.map(function (node) { return node.nodeValue; }).join(' || '); translator.translate(text).then(function (translated) { - translated.split(' || ').forEach(function (str, i) { - nodes[i].nodeValue = str; + translated.split(' || ').forEach(function (html, i) { + $(nodes[i]).replaceWith(html); }); if (show) { $elem.modal('show'); diff --git a/src/views/admin/manage/category-analytics.tpl b/src/views/admin/manage/category-analytics.tpl index eb02abd141..a582a3599e 100644 --- a/src/views/admin/manage/category-analytics.tpl +++ b/src/views/admin/manage/category-analytics.tpl @@ -1,6 +1,8 @@ - Back to Categories List + + [[admin/manage/categories:analytics.back]] + -

Analytics for "{name}" category

+

[[admin/manage/categories:analytics.title, {name}]]


@@ -12,7 +14,7 @@

- +
@@ -23,7 +25,7 @@

- + @@ -36,7 +38,7 @@

- +
@@ -47,7 +49,7 @@

- + \ No newline at end of file diff --git a/src/views/admin/manage/category.tpl b/src/views/admin/manage/category.tpl index 7b2638b584..4d006ad191 100644 --- a/src/views/admin/manage/category.tpl +++ b/src/views/admin/manage/category.tpl @@ -1,8 +1,10 @@

@@ -10,30 +12,40 @@
- -
+ +
- -
+ +
- +
- +
- - @@ -42,19 +54,25 @@

- +
- +
- +
@@ -81,38 +99,53 @@
- +
- +

- +
- +
- +
- +

- +
- +
@@ -120,12 +153,10 @@

- You can configure the access control privileges for this category in this section. Privileges can be granted on a per-user or - a per-group basis. You can add a new user to this table by searching for them in the form below. + [[admin/manage/categories:privileges.description]]

- Note: Privilege settings take effect immediately. It is not necessary to save the category after adjusting - these settings. + [[admin/manage/categories:privileges.warning]]


@@ -139,4 +170,3 @@ - diff --git a/src/views/admin/partials/categories/category-rows.tpl b/src/views/admin/partials/categories/category-rows.tpl index 13e902fb99..1ceeefb9d4 100644 --- a/src/views/admin/partials/categories/category-rows.tpl +++ b/src/views/admin/partials/categories/category-rows.tpl @@ -4,7 +4,13 @@
-
+
@@ -17,10 +23,18 @@
- - Edit + + + + + [[admin/manage/categories:edit]] +
diff --git a/src/views/admin/partials/categories/create.tpl b/src/views/admin/partials/categories/create.tpl index d4c551f1bf..436705b1d8 100644 --- a/src/views/admin/partials/categories/create.tpl +++ b/src/views/admin/partials/categories/create.tpl @@ -1,10 +1,10 @@
- +
- + diff --git a/src/views/admin/partials/categories/privileges.tpl b/src/views/admin/partials/categories/privileges.tpl index 28b5f0849f..f15d207d6c 100644 --- a/src/views/admin/partials/categories/privileges.tpl +++ b/src/views/admin/partials/categories/privileges.tpl @@ -1,12 +1,18 @@ - - - + + + - + @@ -27,14 +33,18 @@ @@ -43,12 +53,18 @@
Viewing PrivilegesPosting PrivilegesModeration Privileges + [[admin/manage/categories:privileges.section-viewing]] + + [[admin/manage/categories:privileges.section-posting]] + + [[admin/manage/categories:privileges.section-moderation]] +
User[[admin/manage/categories:privileges.section-user]] {privileges.labels.users.name}
- +
- - No user-specific privileges in this category. + [[admin/manage/categories:privileges.no-users]] +
- - - + + + - + @@ -57,7 +73,7 @@ @@ -68,16 +84,19 @@
Viewing PrivilegesPosting PrivilegesModeration Privileges + [[admin/manage/categories:privileges.section-viewing]] + + [[admin/manage/categories:privileges.section-posting]] + + [[admin/manage/categories:privileges.section-moderation]] +
Group[[admin/manage/categories:privileges.section-group]] {privileges.labels.groups.name}
- + {privileges.groups.name}
- - - + + +
- If the registered-users group is granted a specific privilege, all other groups receive an - implicit privilege, even if they are not explicitly defined/checked. This implicit - privilege is shown to you because all users are part of the registered-users user group, - and so, privileges for additional groups need not be explicitly granted. + [[admin/manage/categories:privileges.inherit]]
diff --git a/src/views/admin/partials/categories/select-category.tpl b/src/views/admin/partials/categories/select-category.tpl index 7e1f9f0d28..8c93286507 100644 --- a/src/views/admin/partials/categories/select-category.tpl +++ b/src/views/admin/partials/categories/select-category.tpl @@ -1,6 +1,6 @@
- +