Files
NodeBB/src/categories/index.js

429 lines
12 KiB
JavaScript
Raw Normal View History

2014-03-01 16:59:04 -05:00
'use strict';
2019-09-11 18:09:39 -04:00
const _ = require('lodash');
2014-11-08 23:54:21 -05:00
2019-09-11 18:09:39 -04:00
const db = require('../database');
const user = require('../user');
const topics = require('../topics');
2019-09-11 18:09:39 -04:00
const plugins = require('../plugins');
const privileges = require('../privileges');
const cache = require('../cache');
const meta = require('../meta');
2019-09-11 18:09:39 -04:00
const Categories = module.exports;
2017-05-26 16:56:26 -04:00
2018-10-23 15:03:32 -04:00
require('./data')(Categories);
require('./create')(Categories);
require('./delete')(Categories);
require('./topics')(Categories);
require('./unread')(Categories);
require('./activeusers')(Categories);
require('./recentreplies')(Categories);
require('./update')(Categories);
require('./watch')(Categories);
require('./search')(Categories);
2017-05-26 16:56:26 -04:00
Categories.exists = async function (cids) {
return await db.exists(
Array.isArray(cids) ? cids.map(cid => `category:${cid}`) : `category:${cids}`
);
2017-05-26 16:56:26 -04:00
};
2024-03-22 12:39:48 -04:00
Categories.existsByHandle = async function (handle) {
if (Array.isArray(handle)) {
return await db.isSortedSetMembers('categoryhandle:cid', handle);
}
return await db.isSortedSetMember('categoryhandle:cid', handle);
};
2019-07-16 00:41:42 -04:00
Categories.getCategoryById = async function (data) {
const categories = await Categories.getCategories([data.cid]);
2019-07-16 00:41:42 -04:00
if (!categories[0]) {
return null;
}
const category = categories[0];
data.category = category;
const promises = [
data.cid !== '-1' ? Categories.getCategoryTopics(data) : [],
2019-07-16 00:41:42 -04:00
Categories.getTopicCount(data),
Categories.getWatchState([data.cid], data.uid),
getChildrenTree(category, data.uid),
];
if (category.parentCid) {
promises.push(Categories.getCategoryData(category.parentCid));
}
const [topics, topicCount, watchState, , parent] = await Promise.all(promises);
category.topics = topics.topics;
category.nextStart = topics.nextStart;
category.topic_count = topicCount;
category.isWatched = watchState[0] === Categories.watchStates.watching;
category.isTracked = watchState[0] === Categories.watchStates.tracking;
2019-07-16 00:41:42 -04:00
category.isNotWatched = watchState[0] === Categories.watchStates.notwatching;
category.isIgnored = watchState[0] === Categories.watchStates.ignoring;
category.parent = parent;
calculateTopicPostCount(category);
const result = await plugins.hooks.fire('filter:category.get', {
category: category,
...data,
});
return { ...result.category };
2017-05-26 16:56:26 -04:00
};
2024-03-22 12:39:48 -04:00
Categories.getCidByHandle = async function (handle) {
return await db.sortedSetScore('categoryhandle:cid', handle);
};
2019-07-16 00:41:42 -04:00
Categories.getAllCidsFromSet = async function (key) {
let cids = cache.get(key);
if (cids) {
2019-07-16 00:41:42 -04:00
return cids.slice();
}
2019-07-16 00:41:42 -04:00
cids = await db.getSortedSetRange(key, 0, -1);
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
cids = cids.map(cid => parseInt(cid, 10));
2019-07-16 00:41:42 -04:00
cache.set(key, cids);
return cids.slice();
};
Categories.getAllCategories = async function () {
2019-07-16 00:41:42 -04:00
const cids = await Categories.getAllCidsFromSet('categories:cid');
return await Categories.getCategories(cids);
2017-05-26 16:56:26 -04:00
};
2019-07-16 00:41:42 -04:00
Categories.getCidsByPrivilege = async function (set, uid, privilege) {
const cids = await Categories.getAllCidsFromSet(set);
if (set === 'categories:cid') {
cids.unshift(-1);
}
2019-07-16 00:41:42 -04:00
return await privileges.categories.filterCids(privilege, cids, uid);
};
2019-07-16 00:41:42 -04:00
Categories.getCategoriesByPrivilege = async function (set, uid, privilege) {
const cids = await Categories.getCidsByPrivilege(set, uid, privilege);
return await Categories.getCategories(cids);
2017-05-26 16:56:26 -04:00
};
2019-07-16 00:41:42 -04:00
Categories.getModerators = async function (cid) {
const uids = await Categories.getModeratorUids([cid]);
return await user.getUsersFields(uids[0], ['uid', 'username', 'userslug', 'picture']);
2019-01-04 13:27:31 -05:00
};
2019-07-16 00:41:42 -04:00
Categories.getModeratorUids = async function (cids) {
return await privileges.categories.getUidsWithPrivilege(cids, 'moderate');
2017-05-26 16:56:26 -04:00
};
2013-07-03 12:59:45 -04:00
Categories.getCategories = async function (cids) {
2017-05-26 16:56:26 -04:00
if (!Array.isArray(cids)) {
2019-07-16 00:41:42 -04:00
throw new Error('[[error:invalid-cid]]');
2017-05-26 16:56:26 -04:00
}
2017-05-26 16:56:26 -04:00
if (!cids.length) {
2019-07-16 00:41:42 -04:00
return [];
2017-05-26 16:56:26 -04:00
}
2019-07-16 00:41:42 -04:00
const [categories, tagWhitelist] = await Promise.all([
2019-07-16 00:41:42 -04:00
Categories.getCategoriesData(cids),
Categories.getTagWhitelist(cids),
]);
2021-02-04 00:01:39 -07:00
categories.forEach((category, i) => {
2019-07-16 00:41:42 -04:00
if (category) {
category.tagWhitelist = tagWhitelist[i];
}
});
return categories;
2017-05-26 16:56:26 -04:00
};
Categories.setUnread = async function (tree, cids, uid) {
if (uid <= 0) {
return;
}
const { unreadCids } = await topics.getUnreadData({
uid: uid,
cid: cids,
});
if (!unreadCids.length) {
return;
}
function setCategoryUnread(category) {
if (category) {
category.unread = false;
if (unreadCids.includes(category.cid)) {
category.unread = category.topic_count > 0 && true;
} else if (category.children.length) {
category.children.forEach(setCategoryUnread);
category.unread = category.children.some(c => c && c.unread);
}
category['unread-class'] = category.unread ? 'unread' : '';
}
}
tree.forEach(setCategoryUnread);
};
2019-07-16 00:41:42 -04:00
Categories.getTagWhitelist = async function (cids) {
2018-12-09 16:03:41 -05:00
const cachedData = {};
const nonCachedCids = cids.filter((cid) => {
2021-02-03 23:59:08 -07:00
const data = cache.get(`cid:${cid}:tag:whitelist`);
2018-12-09 16:03:41 -05:00
const isInCache = data !== undefined;
if (isInCache) {
cachedData[cid] = data;
}
return !isInCache;
});
if (!nonCachedCids.length) {
return cids.map(cid => cachedData[cid]);
2018-12-09 16:03:41 -05:00
}
2021-02-03 23:59:08 -07:00
const keys = nonCachedCids.map(cid => `cid:${cid}:tag:whitelist`);
2019-07-16 00:41:42 -04:00
const data = await db.getSortedSetsMembers(keys);
nonCachedCids.forEach((cid, index) => {
cachedData[cid] = data[index];
2021-02-03 23:59:08 -07:00
cache.set(`cid:${cid}:tag:whitelist`, data[index]);
2018-12-09 16:03:41 -05:00
});
return cids.map(cid => cachedData[cid]);
2017-05-26 16:56:26 -04:00
};
// remove system tags from tag whitelist for non privileged user
Categories.filterTagWhitelist = function (tagWhitelist, isAdminOrMod) {
const systemTags = (meta.config.systemTags || '').split(',');
if (!isAdminOrMod && systemTags.length) {
return tagWhitelist.filter(tag => !systemTags.includes(tag));
}
return tagWhitelist;
};
2017-05-26 16:56:26 -04:00
function calculateTopicPostCount(category) {
if (!category) {
return;
}
2020-05-08 20:27:04 -04:00
let postCount = category.post_count;
let topicCount = category.topic_count;
if (Array.isArray(category.children)) {
2021-02-04 00:01:39 -07:00
category.children.forEach((child) => {
2020-05-08 20:27:04 -04:00
calculateTopicPostCount(child);
postCount += parseInt(child.totalPostCount, 10) || 0;
topicCount += parseInt(child.totalTopicCount, 10) || 0;
});
2015-04-19 15:18:55 -04:00
}
2017-05-26 16:56:26 -04:00
category.totalPostCount = postCount;
category.totalTopicCount = topicCount;
}
Categories.calculateTopicPostCount = calculateTopicPostCount;
2017-05-26 16:56:26 -04:00
2019-07-16 00:41:42 -04:00
Categories.getParents = async function (cids) {
const categoriesData = await Categories.getCategoriesFields(cids, ['parentCid']);
const parentCids = categoriesData.filter(c => c && c.parentCid).map(c => c.parentCid);
if (!parentCids.length) {
return cids.map(() => null);
}
const parentData = await Categories.getCategoriesData(parentCids);
const cidToParent = _.zipObject(parentCids, parentData);
return categoriesData.map(category => cidToParent[category.parentCid]);
2017-05-26 16:56:26 -04:00
};
2019-07-16 00:41:42 -04:00
Categories.getChildren = async function (cids, uid) {
const categoryData = await Categories.getCategoriesFields(cids, ['parentCid']);
const categories = categoryData.map((category, index) => ({ cid: cids[index], parentCid: category.parentCid }));
await Promise.all(categories.map(c => getChildrenTree(c, uid)));
return categories.map(c => c && c.children);
2017-05-26 16:56:26 -04:00
};
2019-07-16 00:41:42 -04:00
async function getChildrenTree(category, uid) {
let childrenCids = await Categories.getChildrenCids(category.cid);
childrenCids = await privileges.categories.filterCids('find', childrenCids, uid);
childrenCids = childrenCids.filter(cid => parseInt(category.cid, 10) !== parseInt(cid, 10));
if (!childrenCids.length) {
category.children = [];
return;
}
let childrenData = await Categories.getCategoriesData(childrenCids);
childrenData = childrenData.filter(Boolean);
childrenCids = childrenData.map(child => child.cid);
Categories.getTree([category].concat(childrenData), category.parentCid);
2017-05-26 16:56:26 -04:00
}
Categories.getChildrenTree = getChildrenTree;
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
Categories.getParentCids = async function (currentCid) {
let cid = currentCid;
const parents = [];
while (parseInt(cid, 10)) {
// eslint-disable-next-line
cid = await Categories.getCategoryField(cid, 'parentCid');
if (cid) {
parents.unshift(cid);
}
}
return parents;
};
2019-07-16 00:41:42 -04:00
Categories.getChildrenCids = async function (rootCid) {
let allCids = [];
2019-07-16 00:41:42 -04:00
async function recursive(keys) {
let childrenCids = await db.getSortedSetRange(keys, 0, -1);
childrenCids = childrenCids.filter(cid => !allCids.includes(parseInt(cid, 10)));
if (!childrenCids.length) {
return;
}
2021-02-03 23:59:08 -07:00
keys = childrenCids.map(cid => `cid:${cid}:children`);
2019-07-16 00:41:42 -04:00
childrenCids.forEach(cid => allCids.push(parseInt(cid, 10)));
2020-05-12 12:18:30 -04:00
await recursive(keys);
}
2021-02-03 23:59:08 -07:00
const key = `cid:${rootCid}:children`;
const cacheKey = `${key}:all`;
2021-02-05 16:29:56 -05:00
const childrenCids = cache.get(cacheKey);
if (childrenCids) {
2019-07-16 00:41:42 -04:00
return childrenCids.slice();
}
2018-11-06 13:35:55 -05:00
2019-07-16 00:41:42 -04:00
await recursive(key);
allCids = _.uniq(allCids);
2021-02-05 16:29:56 -05:00
cache.set(cacheKey, allCids);
2019-07-16 00:41:42 -04:00
return allCids.slice();
2018-11-06 13:35:55 -05:00
};
2017-05-26 16:56:26 -04:00
Categories.flattenCategories = function (allCategories, categoryData) {
2021-02-04 00:01:39 -07:00
categoryData.forEach((category) => {
2017-05-26 16:56:26 -04:00
if (category) {
allCategories.push(category);
2015-05-30 18:44:31 +03:00
2017-05-26 16:56:26 -04:00
if (Array.isArray(category.children) && category.children.length) {
Categories.flattenCategories(allCategories, category.children);
}
}
2017-05-26 16:56:26 -04:00
});
};
/**
2018-11-27 11:01:46 -05:00
* build tree from flat list of categories
2017-05-26 16:56:26 -04:00
*
* @param categories {array} flat list of categories
* @param parentCid {number} start from 0 to build full tree
*/
Categories.getTree = function (categories, parentCid) {
2018-11-27 11:01:46 -05:00
parentCid = parentCid || 0;
2018-11-29 07:55:56 -05:00
const cids = categories.map(category => category && category.cid);
2018-11-27 11:01:46 -05:00
const cidToCategory = {};
const parents = {};
cids.forEach((cid, index) => {
2018-11-29 07:55:56 -05:00
if (cid) {
categories[index].children = undefined;
2018-11-29 07:55:56 -05:00
cidToCategory[cid] = categories[index];
parents[cid] = { ...categories[index] };
2018-11-29 07:55:56 -05:00
}
2018-11-27 11:01:46 -05:00
});
2018-11-27 11:01:46 -05:00
const tree = [];
2015-05-30 18:44:31 +03:00
2021-02-04 00:01:39 -07:00
categories.forEach((category) => {
2018-11-27 11:01:46 -05:00
if (category) {
category.children = category.children || [];
if (!category.cid) {
return;
}
if (!category.hasOwnProperty('parentCid') || category.parentCid === null) {
category.parentCid = 0;
}
if (category.parentCid === parentCid) {
tree.push(category);
category.parent = parents[parentCid];
} else {
const parent = cidToCategory[category.parentCid];
if (parent && parent.cid !== category.cid) {
category.parent = parents[category.parentCid];
parent.children = parent.children || [];
parent.children.push(category);
}
}
2018-11-27 11:01:46 -05:00
}
});
function sortTree(tree) {
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
tree.sort((a, b) => {
if (a.order !== b.order) {
return a.order - b.order;
}
return a.cid - b.cid;
});
tree.forEach((category) => {
if (category && Array.isArray(category.children)) {
sortTree(category.children);
}
});
}
2018-11-27 11:01:46 -05:00
sortTree(tree);
2015-05-30 18:44:31 +03:00
categories.forEach(c => calculateTopicPostCount(c));
2017-05-26 16:56:26 -04:00
return tree;
};
2016-09-16 14:20:07 +03:00
2019-09-20 22:10:08 -04:00
Categories.buildForSelect = async function (uid, privilege, fields) {
const cids = await Categories.getCidsByPrivilege('categories:cid', uid, privilege);
return await getSelectData(cids, fields);
2017-05-30 14:10:12 -04:00
};
2019-09-20 22:10:08 -04:00
Categories.buildForSelectAll = async function (fields) {
const cids = await Categories.getAllCidsFromSet('categories:cid');
return await getSelectData(cids, fields);
};
2019-09-20 22:10:08 -04:00
async function getSelectData(cids, fields) {
const categoryData = await Categories.getCategoriesData(cids);
const tree = Categories.getTree(categoryData);
return Categories.buildForSelectCategories(tree, fields);
}
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
Categories.buildForSelectCategories = function (categories, fields, parentCid) {
function recursive({ ...category }, categoriesData, level, depth) {
const bullet = level ? '&bull; ' : '';
2017-05-26 16:56:26 -04:00
category.value = category.cid;
2017-05-30 17:21:30 -04:00
category.level = level;
2017-05-26 16:56:26 -04:00
category.text = level + bullet + category.name;
2018-01-05 14:44:18 -05:00
category.depth = depth;
2017-05-26 16:56:26 -04:00
categoriesData.push(category);
if (Array.isArray(category.children)) {
2021-02-03 23:59:08 -07:00
category.children.forEach(child => recursive(child, categoriesData, `&nbsp;&nbsp;&nbsp;&nbsp;${level}`, depth + 1));
}
2017-05-26 16:56:26 -04:00
}
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
parentCid = parentCid || 0;
const categoriesData = [];
2016-09-16 14:20:07 +03:00
Categories refactor (#9257) * feat: wip categories pagination * feat: add subCategoriesPerPage setting * feat: add load more sub categories button to category page * fix: openapi spec * feat: show sub categories left on category page hide button when no more categories left * breaking: rename categories to allCategories on /search categories contains the search results * fix: spec * refactor: remove cidsPerPage * fix: tests * feat: use component for subcategories * fix: prevent negative subCategoriesLeft * feat: new category filter/search WIP * feat: remove categories from /tag * fix: dont load all categories when showing move modal * feat: allow adding custom categories to list * breaking: dont load entire category tree on post queue removed unused code add hooks to filter/selector add options to filter/selector * feat: make selector modal work again * feat: replace old search module * fix: topic move selector * feat: dont load all categories on create category modal * fix: fix more categorySelectors * feat: dont load entire category tree on group details page * feat: dont load all categories on home page and user settings page * feat: add pagination to /user/:userslug/categories * fix: update schemas * fix: more tests * fix: test * feat: flags page, dont return entire category tree * fix: flag test * feat: categories manage page dont load all categories allow changing root category clear caches properly * fix: spec * feat: admins&mods page dont load all categories * fix: spec * fix: dont load all children when opening dropdown * fix: on search results dont return all children * refactor: pass all options, rename options.cids to options.selectedCids * fix: #9266 * fix: index 0 * fix: spec * feat: #9265, add setObjectBulk * refactor: shoter updateOrder * feat: selectors on categories/category * fix: tests and search filter * fix: category update test * feat: pagination on acp categories page show order in set order modal * fix: allow drag&drop on pages > 1 in /admin/manage/categories * fix: teasers for deep nested categories fix sub category display on /category page * fix: spec * refactor: use eslint-disable-next-line * refactor: shorter
2021-02-07 15:09:52 -05:00
const rootCategories = categories.filter(category => category && category.parentCid === parentCid);
2019-09-20 22:10:08 -04:00
Acp redesign (#11639) * acp sidebar * gap in nav * remove shadow * label fixes * color fixes * feat: settings page wip * feat: scroll spy :eyeglasses: move social into general, store social in meta.config like other settings write upgrade script * remove social * rermove openapi routes * cleanup, highlight selected nav item * more cleanup * advanced margin top * derp * match design * bring back version alert fix homepage js, since it moved to general settings * remove unused tpls these moved to general settings * remove more css * offcanvas for mobile fix search * add timeout * add new props * manage categories * small fixes * category-edit * feat category page fixes * add title to settings pages add user settings page * small fixes * some more settings pages * fix: plugin page titles * more settings pages * more padding * more pages, add acp paginator.tpl so it doesn't change when active theme changes * remove placeholder * dashboard table * fix: openapi * fix: controller tests * use fonts from core * some small fixes * fix rep page * refactor: fix name of upgrade script * create category modal group edit * group/groups pages * admins mods * privs * uploads * missing margin * more acp pages * more pages * plugins/rewards/widgets * wrap rewards * fix widgets * fix widget clone button * fix group acp edit link * update search dropdown * remove display block from tbody * use less css * remove some derp links * remove striped tables * remove p tags from lang files * update email settings * Update api.tpl * move tag-whitelist
2023-05-31 11:54:48 -04:00
rootCategories.sort((a, b) => {
if (a.order !== b.order) {
return a.order - b.order;
}
return a.cid - b.cid;
});
2019-09-20 22:10:08 -04:00
rootCategories.forEach(category => recursive(category, categoriesData, '', 0));
2017-05-30 14:10:12 -04:00
2019-09-20 19:04:47 -04:00
const pickFields = [
2021-11-18 16:42:18 -05:00
'cid', 'name', 'level', 'icon', 'parentCid',
2019-09-20 22:10:08 -04:00
'color', 'bgColor', 'backgroundImage', 'imageClass',
2019-09-20 19:04:47 -04:00
];
2019-09-20 22:10:08 -04:00
fields = fields || [];
if (fields.includes('text') && fields.includes('value')) {
return categoriesData.map(category => _.pick(category, fields));
}
if (fields.length) {
pickFields.push(...fields);
}
2019-09-20 19:04:47 -04:00
return categoriesData.map(category => _.pick(category, pickFields));
2017-05-26 16:56:26 -04:00
};
require('../promisify')(Categories);