mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-10-26 08:36:12 +01:00
feat: paginaton for admins-mods, closes #10610
ability to select categoriest to see children
This commit is contained in:
@@ -1,8 +1,10 @@
|
|||||||
{
|
{
|
||||||
"administrators": "Administrators",
|
"administrators": "Administrators",
|
||||||
"global-moderators": "Global Moderators",
|
"global-moderators": "Global Moderators",
|
||||||
|
"moderators": "Moderators",
|
||||||
"no-global-moderators": "No Global Moderators",
|
"no-global-moderators": "No Global Moderators",
|
||||||
"moderators-of-category": "%1 Moderators",
|
"no-sub-categories": "No subcategories",
|
||||||
|
"subcategories": "%1 subcategories",
|
||||||
"no-moderators": "No Moderators",
|
"no-moderators": "No Moderators",
|
||||||
"add-administrator": "Add Administrator",
|
"add-administrator": "Add Administrator",
|
||||||
"add-global-moderator": "Add Global Moderator",
|
"add-global-moderator": "Add Global Moderator",
|
||||||
|
|||||||
@@ -78,6 +78,7 @@ define('admin/manage/admins-mods', [
|
|||||||
|
|
||||||
|
|
||||||
categorySelector.init($('[component="category-selector"]'), {
|
categorySelector.init($('[component="category-selector"]'), {
|
||||||
|
parentCid: ajaxify.data.selectedCategory ? ajaxify.data.selectedCategory.cid : 0,
|
||||||
onSelect: function (selectedCategory) {
|
onSelect: function (selectedCategory) {
|
||||||
ajaxify.go('admin/manage/admins-mods' + (selectedCategory.cid ? '?cid=' + selectedCategory.cid : ''));
|
ajaxify.go('admin/manage/admins-mods' + (selectedCategory.cid ? '?cid=' + selectedCategory.cid : ''));
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -6,36 +6,56 @@ const db = require('../../database');
|
|||||||
const groups = require('../../groups');
|
const groups = require('../../groups');
|
||||||
const categories = require('../../categories');
|
const categories = require('../../categories');
|
||||||
const user = require('../../user');
|
const user = require('../../user');
|
||||||
|
const meta = require('../../meta');
|
||||||
|
const pagination = require('../../pagination');
|
||||||
|
const categoriesController = require('./categories');
|
||||||
|
|
||||||
const AdminsMods = module.exports;
|
const AdminsMods = module.exports;
|
||||||
|
|
||||||
AdminsMods.get = async function (req, res, next) {
|
AdminsMods.get = async function (req, res) {
|
||||||
let cid = parseInt(req.query.cid, 10) || 0;
|
const rootCid = parseInt(req.query.cid, 10) || 0;
|
||||||
if (!cid) {
|
|
||||||
cid = (await db.getSortedSetRange('cid:0:children', 0, 0))[0];
|
const cidsCount = await db.sortedSetCard(`cid:${rootCid}:children`);
|
||||||
}
|
|
||||||
const selectedCategory = await categories.getCategoryData(cid);
|
const pageCount = Math.max(1, Math.ceil(cidsCount / meta.config.categoriesPerPage));
|
||||||
if (!selectedCategory) {
|
const page = Math.min(parseInt(req.query.page, 10) || 1, pageCount);
|
||||||
return next();
|
const start = Math.max(0, (page - 1) * meta.config.categoriesPerPage);
|
||||||
}
|
const stop = start + meta.config.categoriesPerPage - 1;
|
||||||
const [admins, globalMods, moderators] = await Promise.all([
|
|
||||||
|
const cids = await db.getSortedSetRange(`cid:${rootCid}:children`, start, stop);
|
||||||
|
|
||||||
|
const selectedCategory = rootCid ? await categories.getCategoryData(rootCid) : null;
|
||||||
|
const pageCategories = await categories.getCategoriesData(cids);
|
||||||
|
|
||||||
|
const [admins, globalMods, moderators, crumbs] = await Promise.all([
|
||||||
groups.get('administrators', { uid: req.uid }),
|
groups.get('administrators', { uid: req.uid }),
|
||||||
groups.get('Global Moderators', { uid: req.uid }),
|
groups.get('Global Moderators', { uid: req.uid }),
|
||||||
getModeratorsOfCategories(selectedCategory),
|
getModeratorsOfCategories(pageCategories),
|
||||||
|
categoriesController.buildBreadCrumbs(selectedCategory, '/admin/manage/admins-mods'),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
res.render('admin/manage/admins-mods', {
|
res.render('admin/manage/admins-mods', {
|
||||||
admins: admins,
|
admins: admins,
|
||||||
globalMods: globalMods,
|
globalMods: globalMods,
|
||||||
categoryMods: [moderators],
|
categoryMods: moderators,
|
||||||
selectedCategory: selectedCategory,
|
selectedCategory: selectedCategory,
|
||||||
|
pagination: pagination.create(page, pageCount, req.query),
|
||||||
|
breadcrumbs: crumbs,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
async function getModeratorsOfCategories(categoryData) {
|
async function getModeratorsOfCategories(categoryData) {
|
||||||
const moderatorUids = await categories.getModeratorUids([categoryData.cid]);
|
const [moderatorUids, childrenCounts] = await Promise.all([
|
||||||
|
categories.getModeratorUids(categoryData.map(c => c.cid)),
|
||||||
|
db.sortedSetsCard(categoryData.map(c => `cid:${c.cid}:children`)),
|
||||||
|
]);
|
||||||
|
|
||||||
const uids = _.uniq(_.flatten(moderatorUids));
|
const uids = _.uniq(_.flatten(moderatorUids));
|
||||||
const moderatorData = await user.getUsersFields(uids, ['uid', 'username', 'userslug', 'picture']);
|
const moderatorData = await user.getUsersFields(uids, ['uid', 'username', 'userslug', 'picture']);
|
||||||
categoryData.moderators = moderatorData;
|
const moderatorMap = _.zipObject(uids, moderatorData);
|
||||||
|
categoryData.forEach((c, index) => {
|
||||||
|
c.moderators = moderatorUids[index].map(uid => moderatorMap[uid]);
|
||||||
|
c.subCategoryCount = childrenCounts[index];
|
||||||
|
});
|
||||||
return categoryData;
|
return categoryData;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -94,7 +94,7 @@ categoriesController.getAll = async function (req, res) {
|
|||||||
if (rootCid) {
|
if (rootCid) {
|
||||||
selectedCategory = await categories.getCategoryData(rootCid);
|
selectedCategory = await categories.getCategoryData(rootCid);
|
||||||
}
|
}
|
||||||
const crumbs = await buildBreadcrumbs(req, selectedCategory);
|
const crumbs = await buildBreadcrumbs(selectedCategory, '/admin/manage/categories');
|
||||||
res.render('admin/manage/categories', {
|
res.render('admin/manage/categories', {
|
||||||
categoriesTree: tree,
|
categoriesTree: tree,
|
||||||
selectedCategory: selectedCategory,
|
selectedCategory: selectedCategory,
|
||||||
@@ -104,14 +104,14 @@ categoriesController.getAll = async function (req, res) {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
async function buildBreadcrumbs(req, categoryData) {
|
async function buildBreadcrumbs(categoryData, url) {
|
||||||
if (!categoryData) {
|
if (!categoryData) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const breadcrumbs = [
|
const breadcrumbs = [
|
||||||
{
|
{
|
||||||
text: categoryData.name,
|
text: categoryData.name,
|
||||||
url: `${nconf.get('relative_path')}/admin/manage/categories?cid=${categoryData.cid}`,
|
url: `${nconf.get('relative_path')}${url}?cid=${categoryData.cid}`,
|
||||||
cid: categoryData.cid,
|
cid: categoryData.cid,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
@@ -119,16 +119,18 @@ async function buildBreadcrumbs(req, categoryData) {
|
|||||||
const crumbs = allCrumbs.filter(c => c.cid);
|
const crumbs = allCrumbs.filter(c => c.cid);
|
||||||
|
|
||||||
crumbs.forEach((c) => {
|
crumbs.forEach((c) => {
|
||||||
c.url = `/admin/manage/categories?cid=${c.cid}`;
|
c.url = `${url}?cid=${c.cid}`;
|
||||||
});
|
});
|
||||||
crumbs.unshift({
|
crumbs.unshift({
|
||||||
text: '[[admin/manage/categories:top-level]]',
|
text: '[[admin/manage/categories:top-level]]',
|
||||||
url: '/admin/manage/categories',
|
url: url,
|
||||||
});
|
});
|
||||||
|
|
||||||
return crumbs.concat(breadcrumbs);
|
return crumbs.concat(breadcrumbs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
categoriesController.buildBreadCrumbs = buildBreadcrumbs;
|
||||||
|
|
||||||
categoriesController.getAnalytics = async function (req, res) {
|
categoriesController.getAnalytics = async function (req, res) {
|
||||||
const [name, analyticsData] = await Promise.all([
|
const [name, analyticsData] = await Promise.all([
|
||||||
categories.getCategoryField(req.params.category_id, 'name'),
|
categories.getCategoryField(req.params.category_id, 'name'),
|
||||||
|
|||||||
@@ -38,11 +38,19 @@
|
|||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
|
|
||||||
|
<h4 id="moderators-title">[[admin/manage/admins-mods:moderators]]</h4>
|
||||||
|
|
||||||
|
<!-- IMPORT partials/breadcrumbs.tpl -->
|
||||||
|
|
||||||
<!-- IMPORT partials/category-selector.tpl -->
|
<!-- IMPORT partials/category-selector.tpl -->
|
||||||
|
|
||||||
|
{{{ if !categoryMods.length }}}
|
||||||
|
<div><p class="well">[[admin/manage/admins-mods:no-sub-categories]]</p></div>
|
||||||
|
{{{ end }}}
|
||||||
|
|
||||||
{{{ each categoryMods }}}
|
{{{ each categoryMods }}}
|
||||||
<div class="categories category-wrapper category-depth-{categoryMods.depth}">
|
<div class="categories category-wrapper category-depth-{categoryMods.depth}">
|
||||||
<h4>{{{ if categoryMods.icon }}}<i class="fa {categoryMods.icon}"></i> {{{ end }}}[[admin/manage/admins-mods:moderators-of-category, {categoryMods.name}]]{{{if categoryMods.disabled}}}<span class="badge badge-primary">[[admin/manage/admins-mods:disabled]]</span>{{{end}}}</h4>
|
<h4>{{{ if categoryMods.icon }}}<i class="fa {categoryMods.icon}"></i> {{{ end }}}{categoryMods.name} {{{ if categoryMods.subCategoryCount }}}<small><a href="{config.relative_path}/admin/manage/admins-mods?cid={categoryMods.cid}#moderators-title">[[admin/manage/admins-mods:subcategories, {categoryMods.subCategoryCount}]]</a></small>{{{ else }}}{{{ end }}}{{{if categoryMods.disabled}}}<span class="badge badge-primary">[[admin/manage/admins-mods:disabled]]</span>{{{end}}}</h4>
|
||||||
<div class="moderator-area" data-cid="{categoryMods.cid}">
|
<div class="moderator-area" data-cid="{categoryMods.cid}">
|
||||||
{{{ each categoryMods.moderators }}}
|
{{{ each categoryMods.moderators }}}
|
||||||
<div class="user-card pull-left" data-uid="{categoryMods.moderators.uid}">
|
<div class="user-card pull-left" data-uid="{categoryMods.moderators.uid}">
|
||||||
@@ -63,4 +71,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<br/>
|
<br/>
|
||||||
{{{ end }}}
|
{{{ end }}}
|
||||||
|
<div>
|
||||||
|
<!-- IMPORT partials/paginator.tpl -->
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user