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.
211 lines
7.0 KiB
JavaScript
211 lines
7.0 KiB
JavaScript
'use strict';
|
|
|
|
const db = require('../database');
|
|
const topics = require('../topics');
|
|
const plugins = require('../plugins');
|
|
const meta = require('../meta');
|
|
const privileges = require('../privileges');
|
|
const user = require('../user');
|
|
|
|
module.exports = function (Categories) {
|
|
Categories.getCategoryTopics = async function (data) {
|
|
let results = await plugins.hooks.fire('filter:category.topics.prepare', data);
|
|
const tids = await Categories.getTopicIds(results);
|
|
let topicsData = await topics.getTopicsByTids(tids, data.uid);
|
|
topicsData = await user.blocks.filter(data.uid, topicsData);
|
|
|
|
if (!topicsData.length) {
|
|
return { topics: [], uid: data.uid };
|
|
}
|
|
topics.calculateTopicIndices(topicsData, data.start);
|
|
|
|
results = await plugins.hooks.fire('filter:category.topics.get', { cid: data.cid, topics: topicsData, uid: data.uid });
|
|
return { topics: results.topics, nextStart: data.stop + 1 };
|
|
};
|
|
|
|
Categories.getTopicIds = async function (data) {
|
|
const dataForPinned = { ...data };
|
|
dataForPinned.start = 0;
|
|
dataForPinned.stop = -1;
|
|
|
|
const [pinnedTids, set, direction] = await Promise.all([
|
|
Categories.getPinnedTids(dataForPinned),
|
|
Categories.buildTopicsSortedSet(data),
|
|
Categories.getSortedSetRangeDirection(data.sort),
|
|
]);
|
|
|
|
const totalPinnedCount = pinnedTids.length;
|
|
const pinnedTidsOnPage = pinnedTids.slice(data.start, data.stop !== -1 ? data.stop + 1 : undefined);
|
|
const pinnedCountOnPage = pinnedTidsOnPage.length;
|
|
const topicsPerPage = data.stop - data.start + 1;
|
|
const normalTidsToGet = Math.max(0, topicsPerPage - pinnedCountOnPage);
|
|
|
|
if (!normalTidsToGet && data.stop !== -1) {
|
|
return pinnedTidsOnPage;
|
|
}
|
|
|
|
if (plugins.hooks.hasListeners('filter:categories.getTopicIds')) {
|
|
const result = await plugins.hooks.fire('filter:categories.getTopicIds', {
|
|
tids: [],
|
|
data: data,
|
|
pinnedTids: pinnedTidsOnPage,
|
|
allPinnedTids: pinnedTids,
|
|
totalPinnedCount: totalPinnedCount,
|
|
normalTidsToGet: normalTidsToGet,
|
|
});
|
|
return result && result.tids;
|
|
}
|
|
|
|
let { start } = data;
|
|
if (start > 0 && totalPinnedCount) {
|
|
start -= totalPinnedCount - pinnedCountOnPage;
|
|
}
|
|
|
|
const stop = data.stop === -1 ? data.stop : start + normalTidsToGet - 1;
|
|
let normalTids;
|
|
const reverse = direction === 'highest-to-lowest';
|
|
if (Array.isArray(set)) {
|
|
const weights = set.map((s, index) => (index ? 0 : 1));
|
|
normalTids = await db[reverse ? 'getSortedSetRevIntersect' : 'getSortedSetIntersect']({ sets: set, start: start, stop: stop, weights: weights });
|
|
} else {
|
|
normalTids = await db[reverse ? 'getSortedSetRevRange' : 'getSortedSetRange'](set, start, stop);
|
|
}
|
|
normalTids = normalTids.filter(tid => !pinnedTids.includes(tid));
|
|
return pinnedTidsOnPage.concat(normalTids);
|
|
};
|
|
|
|
Categories.getTopicCount = async function (data) {
|
|
if (plugins.hooks.hasListeners('filter:categories.getTopicCount')) {
|
|
const result = await plugins.hooks.fire('filter:categories.getTopicCount', {
|
|
topicCount: data.category.topic_count,
|
|
data: data,
|
|
});
|
|
return result && result.topicCount;
|
|
}
|
|
const set = await Categories.buildTopicsSortedSet(data);
|
|
if (Array.isArray(set)) {
|
|
return await db.sortedSetIntersectCard(set);
|
|
} else if (data.targetUid && set) {
|
|
return await db.sortedSetCard(set);
|
|
}
|
|
return data.category.topic_count;
|
|
};
|
|
|
|
Categories.buildTopicsSortedSet = async function (data) {
|
|
const { cid } = data;
|
|
let set = `cid:${cid}:tids`;
|
|
const sort = data.sort || (data.settings && data.settings.categoryTopicSort) || meta.config.categoryTopicSort || 'newest_to_oldest';
|
|
|
|
if (sort === 'most_posts') {
|
|
set = `cid:${cid}:tids:posts`;
|
|
} else if (sort === 'most_votes') {
|
|
set = `cid:${cid}:tids:votes`;
|
|
} else if (sort === 'most_views') {
|
|
set = `cid:${cid}:tids:views`;
|
|
}
|
|
|
|
if (data.tag) {
|
|
if (Array.isArray(data.tag)) {
|
|
set = [set].concat(data.tag.map(tag => `tag:${tag}:topics`));
|
|
} else {
|
|
set = [set, `tag:${data.tag}:topics`];
|
|
}
|
|
}
|
|
|
|
if (data.targetUid) {
|
|
set = (Array.isArray(set) ? set : [set]).concat([`cid:${cid}:uid:${data.targetUid}:tids`]);
|
|
}
|
|
|
|
const result = await plugins.hooks.fire('filter:categories.buildTopicsSortedSet', {
|
|
set: set,
|
|
data: data,
|
|
});
|
|
return result && result.set;
|
|
};
|
|
|
|
Categories.getSortedSetRangeDirection = async function (sort) {
|
|
sort = sort || 'newest_to_oldest';
|
|
const direction = ['newest_to_oldest', 'most_posts', 'most_votes', 'most_views'].includes(sort) ? 'highest-to-lowest' : 'lowest-to-highest';
|
|
const result = await plugins.hooks.fire('filter:categories.getSortedSetRangeDirection', {
|
|
sort: sort,
|
|
direction: direction,
|
|
});
|
|
return result && result.direction;
|
|
};
|
|
|
|
Categories.getAllTopicIds = async function (cid, start, stop) {
|
|
return await db.getSortedSetRange([`cid:${cid}:tids:pinned`, `cid:${cid}:tids`], start, stop);
|
|
};
|
|
|
|
Categories.getPinnedTids = async function (data) {
|
|
if (plugins.hooks.hasListeners('filter:categories.getPinnedTids')) {
|
|
const result = await plugins.hooks.fire('filter:categories.getPinnedTids', {
|
|
pinnedTids: [],
|
|
data: data,
|
|
});
|
|
return result && result.pinnedTids;
|
|
}
|
|
const [allPinnedTids, canSchedule] = await Promise.all([
|
|
db.getSortedSetRevRange(`cid:${data.cid}:tids:pinned`, data.start, data.stop),
|
|
privileges.categories.can('topics:schedule', data.cid, data.uid),
|
|
]);
|
|
const pinnedTids = canSchedule ? allPinnedTids : await filterScheduledTids(allPinnedTids);
|
|
|
|
return await topics.tools.checkPinExpiry(pinnedTids);
|
|
};
|
|
|
|
Categories.modifyTopicsByPrivilege = function (topics, privileges) {
|
|
if (!Array.isArray(topics) || !topics.length || privileges.view_deleted) {
|
|
return;
|
|
}
|
|
|
|
topics.forEach((topic) => {
|
|
if (!topic.scheduled && topic.deleted && !topic.isOwner) {
|
|
topic.title = '[[topic:topic_is_deleted]]';
|
|
if (topic.hasOwnProperty('titleRaw')) {
|
|
topic.titleRaw = '[[topic:topic_is_deleted]]';
|
|
}
|
|
topic.slug = topic.tid;
|
|
topic.teaser = null;
|
|
topic.noAnchor = true;
|
|
topic.tags = [];
|
|
}
|
|
});
|
|
};
|
|
|
|
Categories.onNewPostMade = async function (cid, pinned, postData) {
|
|
if (!cid || !postData) {
|
|
return;
|
|
}
|
|
const promises = [
|
|
db.sortedSetAdd(`cid:${cid}:pids`, postData.timestamp, postData.pid),
|
|
db.incrObjectField(`category:${cid}`, 'post_count'),
|
|
];
|
|
if (!pinned) {
|
|
promises.push(db.sortedSetIncrBy(`cid:${cid}:tids:posts`, 1, postData.tid));
|
|
}
|
|
await Promise.all(promises);
|
|
await Categories.updateRecentTidForCid(cid);
|
|
};
|
|
|
|
Categories.onTopicsMoved = async (cids) => {
|
|
await Promise.all(cids.map(async (cid) => {
|
|
await Promise.all([
|
|
Categories.setCategoryField(
|
|
cid, 'topic_count', await db.sortedSetCard(`cid:${cid}:tids:lastposttime`)
|
|
),
|
|
Categories.setCategoryField(
|
|
cid, 'post_count', await db.sortedSetCard(`cid:${cid}:pids`)
|
|
),
|
|
Categories.updateRecentTidForCid(cid),
|
|
]);
|
|
}));
|
|
};
|
|
|
|
async function filterScheduledTids(tids) {
|
|
const scores = await db.sortedSetScores('topics:scheduled', tids);
|
|
const now = Date.now();
|
|
return tids.filter((tid, index) => tid && (!scores[index] || scores[index] <= now));
|
|
}
|
|
};
|