mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 16:46:12 +01:00 
			
		
		
		
	refactor: announces
store number of announces on post hash, show announces like votes, with tooltip and a way to see all, remove them from topic.events so they dont load all tid:<tid>:posts everytime topic is loaded
This commit is contained in:
		| @@ -13,6 +13,6 @@ | ||||
| 	"help.federating": "Likewise, if users from outside of this forum start following <em>you</em>, then your posts will start appearing on those apps and websites as well.", | ||||
| 	"help.next-generation": "This is the next generation of social media, start contributing today!", | ||||
|  | ||||
| 	"topic-event-announce-ago": "%1 shared <a href=\"%2\">this post</a> %3", | ||||
| 	"topic-event-announce-on": "%1 shared <a href=\"%2\">this post</a> on %3" | ||||
| 	"announcers": "Announcers", | ||||
| 	"announcers-x": "Announcers (%1)" | ||||
| } | ||||
| @@ -186,6 +186,10 @@ paths: | ||||
|     $ref: 'write/posts/pid/voters.yaml' | ||||
|   /posts/{pid}/upvoters: | ||||
|     $ref: 'write/posts/pid/upvoters.yaml' | ||||
|   /posts/{pid}/announcers: | ||||
|     $ref: 'write/posts/pid/announcers.yaml' | ||||
|   /posts/{pid}/announcers/tooltip: | ||||
|     $ref: 'write/posts/pid/announcers-tooltip.yaml' | ||||
|   /posts/{pid}/bookmark: | ||||
|     $ref: 'write/posts/pid/bookmark.yaml' | ||||
|   /posts/{pid}/diffs: | ||||
|   | ||||
							
								
								
									
										33
									
								
								public/openapi/write/posts/pid/announcers-tooltip.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								public/openapi/write/posts/pid/announcers-tooltip.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| get: | ||||
|   tags: | ||||
|     - posts | ||||
|   summary: get announcers of a post | ||||
|   description: This is used for getting a list of usernames for the announcers tooltip | ||||
|   parameters: | ||||
|     - in: path | ||||
|       name: pid | ||||
|       schema: | ||||
|         type: string | ||||
|       required: true | ||||
|       description: a valid post id | ||||
|       example: 2 | ||||
|   responses: | ||||
|     '200': | ||||
|       description: Usernames of announcers of post | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             type: object | ||||
|             properties: | ||||
|               status: | ||||
|                 $ref: ../../../components/schemas/Status.yaml#/Status | ||||
|               response: | ||||
|                 type: object | ||||
|                 properties: | ||||
|                   otherCount: | ||||
|                     type: number | ||||
|                   usernames: | ||||
|                     type: array | ||||
|                   cutoff: | ||||
|                     type: number | ||||
|  | ||||
							
								
								
									
										32
									
								
								public/openapi/write/posts/pid/announcers.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								public/openapi/write/posts/pid/announcers.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| get: | ||||
|   tags: | ||||
|     - posts | ||||
|   summary: get announcers of a post | ||||
|   description: This returns the announcers of a post if the user has permission to view them | ||||
|   parameters: | ||||
|     - in: path | ||||
|       name: pid | ||||
|       schema: | ||||
|         type: string | ||||
|       required: true | ||||
|       description: a valid post id | ||||
|       example: 2 | ||||
|   responses: | ||||
|     '200': | ||||
|       description: Data about announcers of this post | ||||
|       content: | ||||
|         application/json: | ||||
|           schema: | ||||
|             type: object | ||||
|             properties: | ||||
|               status: | ||||
|                 $ref: ../../../components/schemas/Status.yaml#/Status | ||||
|               response: | ||||
|                 type: object | ||||
|                 properties: | ||||
|                   announceCount: | ||||
|                     type: number | ||||
|                   announcers: | ||||
|                     type: array | ||||
|  | ||||
|  | ||||
| @@ -141,6 +141,10 @@ define('forum/topic/postTools', [ | ||||
| 			votes.showVotes(getData($(this), 'data-pid')); | ||||
| 		}); | ||||
|  | ||||
| 		postContainer.on('click', '[component="post/announce-count"]', function () { | ||||
| 			votes.showAnnouncers(getData($(this), 'data-pid')); | ||||
| 		}); | ||||
|  | ||||
| 		postContainer.on('click', '[component="post/flag"]', function () { | ||||
| 			const pid = getData($(this), 'data-pid'); | ||||
| 			require(['flags'], function (flags) { | ||||
|   | ||||
| @@ -13,6 +13,9 @@ define('forum/topic/votes', [ | ||||
| 			components.get('topic').on('mouseenter', '[data-pid] [component="post/vote-count"]', loadDataAndCreateTooltip); | ||||
| 			components.get('topic').on('mouseleave', '[data-pid] [component="post/vote-count"]', destroyTooltip); | ||||
| 		} | ||||
|  | ||||
| 		components.get('topic').on('mouseenter', '[data-pid] [component="post/announce-count"]', loadDataAndCreateTooltip); | ||||
| 		components.get('topic').on('mouseleave', '[data-pid] [component="post/announce-count"]', destroyTooltip); | ||||
| 	}; | ||||
|  | ||||
| 	function canSeeVotes() { | ||||
| @@ -43,8 +46,11 @@ define('forum/topic/votes', [ | ||||
| 			tooltip.dispose(); | ||||
| 			$this.attr('title', ''); | ||||
| 		} | ||||
| 		const path = $this.attr('component') === 'post/vote-count' ? | ||||
| 			`/posts/${encodeURIComponent(pid)}/upvoters` : | ||||
| 			`/posts/${encodeURIComponent(pid)}/announcers/tooltip`; | ||||
|  | ||||
| 		api.get(`/posts/${encodeURIComponent(pid)}/upvoters`, {}, function (err, data) { | ||||
| 		api.get(path, {}, function (err, data) { | ||||
| 			if (err) { | ||||
| 				return alerts.error(err); | ||||
| 			} | ||||
| @@ -132,6 +138,24 @@ define('forum/topic/votes', [ | ||||
| 		}); | ||||
| 	}; | ||||
|  | ||||
| 	Votes.showAnnouncers = async function (pid) { | ||||
| 		const data = await api.get(`/posts/${encodeURIComponent(pid)}/announcers`, {}) | ||||
| 			.catch(err => alerts.error(err)); | ||||
|  | ||||
| 		const html = await app.parseAndTranslate('modals/announcers', data); | ||||
| 		const dialog = bootbox.dialog({ | ||||
| 			title: `[[activitypub:announcers-x, ${data.announceCount}]]`, | ||||
| 			message: html, | ||||
| 			className: 'announce-modal', | ||||
| 			show: true, | ||||
| 			onEscape: true, | ||||
| 			backdrop: true, | ||||
| 		}); | ||||
|  | ||||
| 		dialog.on('click', function () { | ||||
| 			dialog.modal('hide'); | ||||
| 		}); | ||||
| 	}; | ||||
|  | ||||
| 	return Votes; | ||||
| }); | ||||
|   | ||||
| @@ -365,14 +365,24 @@ Notes.announce.list = async ({ pid, tid }) => { | ||||
|  | ||||
| Notes.announce.add = async (pid, actor, timestamp = Date.now()) => { | ||||
| 	await db.sortedSetAdd(`pid:${pid}:announces`, timestamp, actor); | ||||
| 	await posts.setPostField(pid, 'announces', await db.sortedSetCard(`pid:${pid}:announces`)); | ||||
| }; | ||||
|  | ||||
| Notes.announce.remove = async (pid, actor) => { | ||||
| 	await db.sortedSetRemove(`pid:${pid}:announces`, actor); | ||||
| 	const count = await db.sortedSetCard(`pid:${pid}:announces`); | ||||
| 	if (count > 0) { | ||||
| 		await posts.setPostField(pid, 'announces', count); | ||||
| 	} else { | ||||
| 		await db.deleteObjectField(`post:${pid}`, 'announces'); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| Notes.announce.removeAll = async (pid) => { | ||||
| 	await db.delete(`pid:${pid}:announces`); | ||||
| 	await Promise.all([ | ||||
| 		db.delete(`pid:${pid}:announces`), | ||||
| 		db.deleteObjectField(`post:${pid}`, 'announces'), | ||||
| 	]); | ||||
| }; | ||||
|  | ||||
| Notes.delete = async (pids) => { | ||||
|   | ||||
| @@ -364,9 +364,13 @@ postsAPI.getUpvoters = async function (caller, data) { | ||||
| 		throw new Error('[[error:no-privileges]]'); | ||||
| 	} | ||||
|  | ||||
| 	let upvotedUids = (await posts.getUpvotedUidsByPids([pid]))[0]; | ||||
| 	const upvotedUids = (await posts.getUpvotedUidsByPids([pid]))[0]; | ||||
| 	return await getTooltipData(upvotedUids); | ||||
| }; | ||||
|  | ||||
| async function getTooltipData(uids) { | ||||
| 	const cutoff = 6; | ||||
| 	if (!upvotedUids.length) { | ||||
| 	if (!uids.length) { | ||||
| 		return { | ||||
| 			otherCount: 0, | ||||
| 			usernames: [], | ||||
| @@ -374,17 +378,41 @@ postsAPI.getUpvoters = async function (caller, data) { | ||||
| 		}; | ||||
| 	} | ||||
| 	let otherCount = 0; | ||||
| 	if (upvotedUids.length > cutoff) { | ||||
| 		otherCount = upvotedUids.length - (cutoff - 1); | ||||
| 		upvotedUids = upvotedUids.slice(0, cutoff - 1); | ||||
| 	if (uids.length > cutoff) { | ||||
| 		otherCount = uids.length - (cutoff - 1); | ||||
| 		uids = uids.slice(0, cutoff - 1); | ||||
| 	} | ||||
|  | ||||
| 	const usernames = await user.getUsernamesByUids(upvotedUids); | ||||
| 	const usernames = await user.getUsernamesByUids(uids); | ||||
| 	return { | ||||
| 		otherCount, | ||||
| 		usernames, | ||||
| 		cutoff, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| postsAPI.getAnnouncers = async (caller, data) => { | ||||
| 	if (!data.pid) { | ||||
| 		throw new Error('[[error:invalid-data]]'); | ||||
| 	} | ||||
| 	if (!meta.config.activitypubEnabled) { | ||||
| 		return []; | ||||
| 	} | ||||
| 	const { pid } = data; | ||||
| 	const cid = await posts.getCidByPid(pid); | ||||
| 	if (!await privileges.categories.isUserAllowedTo('topics:read', cid, caller.uid)) { | ||||
| 		throw new Error('[[error:no-privileges]]'); | ||||
| 	} | ||||
| 	const notes = require('../activitypub/notes'); | ||||
| 	const announcers = await notes.announce.list({ pid }); | ||||
| 	const uids = announcers.map(ann => ann.actor); | ||||
| 	if (data.tooltip) { | ||||
| 		return await getTooltipData(uids); | ||||
| 	} | ||||
| 	return { | ||||
| 		announceCount: uids.length, | ||||
| 		announcers: await user.getUsersFields(uids, ['username', 'userslug', 'picture']), | ||||
| 	}; | ||||
| }; | ||||
|  | ||||
| async function canSeeVotes(uid, cids) { | ||||
|   | ||||
| @@ -141,6 +141,16 @@ Posts.getUpvoters = async (req, res) => { | ||||
| 	helpers.formatApiResponse(200, res, data); | ||||
| }; | ||||
|  | ||||
| Posts.getAnnouncers = async (req, res) => { | ||||
| 	const data = await api.posts.getAnnouncers(req, { pid: req.params.pid, tooltip: 0 }); | ||||
| 	helpers.formatApiResponse(200, res, data); | ||||
| }; | ||||
|  | ||||
| Posts.getAnnouncersTooltip = async (req, res) => { | ||||
| 	const data = await api.posts.getAnnouncers(req, { pid: req.params.pid, tooltip: 1 }); | ||||
| 	helpers.formatApiResponse(200, res, data); | ||||
| }; | ||||
|  | ||||
| Posts.bookmark = async (req, res) => { | ||||
| 	const data = await mock(req); | ||||
| 	await api.posts.bookmark(req, data); | ||||
|   | ||||
| @@ -29,6 +29,8 @@ module.exports = function () { | ||||
| 	setupApiRoute(router, 'get', '/:pid/voters', [middleware.assert.post], controllers.write.posts.getVoters); | ||||
| 	setupApiRoute(router, 'get', '/:pid/upvoters', [middleware.assert.post], controllers.write.posts.getUpvoters); | ||||
|  | ||||
| 	setupApiRoute(router, 'get', '/:pid/announcers', [middleware.assert.post], controllers.write.posts.getAnnouncers); | ||||
| 	setupApiRoute(router, 'get', '/:pid/announcers/tooltip', [middleware.assert.post], controllers.write.posts.getAnnouncersTooltip); | ||||
| 	setupApiRoute(router, 'put', '/:pid/bookmark', middlewares, controllers.write.posts.bookmark); | ||||
| 	setupApiRoute(router, 'delete', '/:pid/bookmark', middlewares, controllers.write.posts.unbookmark); | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,6 @@ const categories = require('../categories'); | ||||
| const plugins = require('../plugins'); | ||||
| const translator = require('../translator'); | ||||
| const privileges = require('../privileges'); | ||||
| const activitypub = require('../activitypub'); | ||||
| const utils = require('../utils'); | ||||
| const helpers = require('../helpers'); | ||||
|  | ||||
| @@ -69,10 +68,6 @@ Events._types = { | ||||
| 		icon: 'fa-code-fork', | ||||
| 		translation: async (event, language) => translateEventArgs(event, language, 'topic:user-forked-topic', renderUser(event), `${relative_path}${event.href}`, renderTimeago(event)), | ||||
| 	}, | ||||
| 	announce: { | ||||
| 		icon: 'fa-share-alt', | ||||
| 		translation: async (event, language) => translateEventArgs(event, language, 'activitypub:topic-event-announce', renderUser(event), `${relative_path}${event.href}`, renderTimeago(event)), | ||||
| 	}, | ||||
| }; | ||||
|  | ||||
| Events.init = async () => { | ||||
| @@ -175,19 +170,6 @@ async function modifyEvent({ tid, uid, eventIds, timestamps, events }) { | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	// Add post announces | ||||
| 	const announces = await activitypub.notes.announce.list({ tid }); | ||||
| 	announces.forEach(({ actor, pid, timestamp }) => { | ||||
| 		events.push({ | ||||
| 			type: 'announce', | ||||
| 			uid: actor, | ||||
| 			href: `/post/${encodeURIComponent(pid)}`, | ||||
| 			pid, | ||||
| 			timestamp, | ||||
| 		}); | ||||
| 		timestamps.push(timestamp); | ||||
| 	}); | ||||
|  | ||||
| 	const [users, fromCategories, userSettings] = await Promise.all([ | ||||
| 		getUserInfo(events.map(event => event.uid).filter(Boolean)), | ||||
| 		getCategoryInfo(events.map(event => event.fromCid).filter(Boolean)), | ||||
|   | ||||
							
								
								
									
										5
									
								
								src/views/modals/announcers.tpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/views/modals/announcers.tpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| <div class="mb-3"> | ||||
| 	{{{ each announcers }}} | ||||
| 	<a class="text-decoration-none" href="{config.relative_path}/user/{./userslug}">{buildAvatar(@value, "24px", true)}</a> | ||||
| 	{{{ end }}} | ||||
| </div> | ||||
		Reference in New Issue
	
	Block a user