mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 16:46:12 +01:00 
			
		
		
		
	feat: show best & latest posts on profile
This commit is contained in:
		| @@ -49,6 +49,7 @@ | ||||
| 	"account/following": "People %1 follows", | ||||
| 	"account/followers": "People who follow %1", | ||||
| 	"account/posts": "Posts made by %1", | ||||
| 	"account/latest-posts": "Latest posts made by %1", | ||||
| 	"account/topics": "Topics created by %1", | ||||
| 	"account/groups": "%1's Groups", | ||||
| 	"account/watched_categories": "%1's Watched Categories", | ||||
|   | ||||
| @@ -3,27 +3,19 @@ | ||||
|  | ||||
| define('forum/account/profile', [ | ||||
| 	'forum/account/header', | ||||
| 	'forum/infinitescroll', | ||||
| 	'components', | ||||
| ], function (header, infinitescroll) { | ||||
| ], function (header) { | ||||
| 	var Account = {}; | ||||
| 	var theirid; | ||||
|  | ||||
| 	Account.init = function () { | ||||
| 		header.init(); | ||||
|  | ||||
| 		theirid = ajaxify.data.theirid; | ||||
|  | ||||
| 		app.enterRoom('user/' + theirid); | ||||
| 		app.enterRoom('user/' + ajaxify.data.theirid); | ||||
|  | ||||
| 		processPage(); | ||||
|  | ||||
| 		socket.removeListener('event:user_status_change', onUserStatusChange); | ||||
| 		socket.on('event:user_status_change', onUserStatusChange); | ||||
|  | ||||
| 		if (!config.usePagination) { | ||||
| 			infinitescroll.init(loadMorePosts); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	function processPage() { | ||||
| @@ -38,43 +30,5 @@ define('forum/account/profile', [ | ||||
| 		app.updateUserStatus($('.account [data-uid="' + data.uid + '"] [component="user/status"]'), data.status); | ||||
| 	} | ||||
|  | ||||
| 	function loadMorePosts(direction) { | ||||
| 		if (direction < 0 || !$('[component="posts"]').length) { | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		$('[component="posts/loading"]').removeClass('hidden'); | ||||
|  | ||||
| 		infinitescroll.loadMore('posts.loadMoreUserPosts', { | ||||
| 			after: $('[component="posts"]').attr('data-nextstart'), | ||||
| 			uid: theirid, | ||||
| 		}, function (data, done) { | ||||
| 			if (data.posts && data.posts.length) { | ||||
| 				onPostsLoaded(data.posts, done); | ||||
| 			} else { | ||||
| 				done(); | ||||
| 			} | ||||
| 			$('[component="posts"]').attr('data-nextstart', data.nextStart); | ||||
| 			$('[component="posts/loading"]').addClass('hidden'); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	function onPostsLoaded(posts, callback) { | ||||
| 		posts = posts.filter(function (post) { | ||||
| 			return !$('[component="posts"] [data-pid=' + post.pid + ']').length; | ||||
| 		}); | ||||
|  | ||||
| 		if (!posts.length) { | ||||
| 			return callback(); | ||||
| 		} | ||||
|  | ||||
| 		app.parseAndTranslate('account/profile', 'posts', { posts: posts }, function (html) { | ||||
| 			$('[component="posts"]').append(html); | ||||
| 			html.find('.timeago').timeago(); | ||||
|  | ||||
| 			callback(); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	return Account; | ||||
| }); | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
| 		}; | ||||
|  | ||||
| 		process.profile = function (operation, start) { | ||||
| 			winston.log('%s took %d milliseconds', operation, process.elapsedTimeSince(start)); | ||||
| 			console.log('%s took %d milliseconds', operation, process.elapsedTimeSince(start)); | ||||
| 		}; | ||||
|  | ||||
| 		process.elapsedTimeSince = function (start) { | ||||
|   | ||||
| @@ -3,13 +3,14 @@ | ||||
| var nconf = require('nconf'); | ||||
| var async = require('async'); | ||||
|  | ||||
| const db = require('../../database'); | ||||
| const privileges = require('../../privileges'); | ||||
| var user = require('../../user'); | ||||
| var posts = require('../../posts'); | ||||
| var plugins = require('../../plugins'); | ||||
| var meta = require('../../meta'); | ||||
| var accountHelpers = require('./helpers'); | ||||
| var helpers = require('../helpers'); | ||||
| var pagination = require('../../pagination'); | ||||
| var messaging = require('../../messaging'); | ||||
| var translator = require('../../translator'); | ||||
| var utils = require('../../utils'); | ||||
| @@ -26,10 +27,7 @@ profileController.get = function (req, res, callback) { | ||||
| 			return res.redirect(nconf.get('relative_path') + '/user/' + lowercaseSlug); | ||||
| 		} | ||||
| 	} | ||||
| 	var page = Math.max(1, parseInt(req.query.page, 10) || 1); | ||||
| 	var itemsPerPage = 10; | ||||
| 	var start = (page - 1) * itemsPerPage; | ||||
| 	var stop = start + itemsPerPage - 1; | ||||
|  | ||||
| 	var userData; | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| @@ -54,8 +52,11 @@ profileController.get = function (req, res, callback) { | ||||
| 				hasPrivateChat: function (next) { | ||||
| 					messaging.hasPrivateChat(req.uid, userData.uid, next); | ||||
| 				}, | ||||
| 				posts: function (next) { | ||||
| 					posts.getPostSummariesFromSet('uid:' + userData.theirid + ':posts', req.uid, start, stop, next); | ||||
| 				latestPosts: function (next) { | ||||
| 					getLatestPosts(req.uid, userData, next); | ||||
| 				}, | ||||
| 				bestPosts: function (next) { | ||||
| 					getBestPosts(req.uid, userData, next); | ||||
| 				}, | ||||
| 				signature: function (next) { | ||||
| 					posts.parseSignature(userData, req.uid, next); | ||||
| @@ -74,55 +75,20 @@ profileController.get = function (req, res, callback) { | ||||
| 				delete userData.reputation; | ||||
| 			} | ||||
|  | ||||
| 			userData.posts = results.posts.posts.filter(p => p && !p.deleted); | ||||
| 			userData.posts = results.latestPosts; // for backwards compat. | ||||
| 			userData.latestPosts = results.latestPosts; | ||||
| 			userData.bestPosts = results.bestPosts; | ||||
| 			userData.hasPrivateChat = results.hasPrivateChat; | ||||
| 			userData.aboutme = translator.escape(results.aboutme); | ||||
| 			userData.nextStart = results.posts.nextStart; | ||||
| 			userData.breadcrumbs = helpers.buildBreadcrumbs([{ text: userData.username }]); | ||||
| 			userData.title = userData.username; | ||||
| 			userData.allowCoverPicture = !userData.isSelf || userData.reputation >= (meta.config['min:rep:cover-picture'] || 0); | ||||
| 			var pageCount = Math.ceil(userData.postcount / itemsPerPage); | ||||
| 			userData.pagination = pagination.create(page, pageCount, req.query); | ||||
|  | ||||
| 			if (!userData.profileviews) { | ||||
| 				userData.profileviews = 1; | ||||
| 			} | ||||
|  | ||||
| 			var plainAboutMe = userData.aboutme ? utils.stripHTMLTags(utils.decodeHTMLEntities(userData.aboutme)) : ''; | ||||
|  | ||||
| 			res.locals.metaTags = [ | ||||
| 				{ | ||||
| 					name: 'title', | ||||
| 					content: userData.fullname || userData.username, | ||||
| 				}, | ||||
| 				{ | ||||
| 					name: 'description', | ||||
| 					content: plainAboutMe, | ||||
| 				}, | ||||
| 				{ | ||||
| 					property: 'og:title', | ||||
| 					content: userData.fullname || userData.username, | ||||
| 				}, | ||||
| 				{ | ||||
| 					property: 'og:description', | ||||
| 					content: plainAboutMe, | ||||
| 				}, | ||||
| 			]; | ||||
|  | ||||
| 			if (userData.picture) { | ||||
| 				res.locals.metaTags.push( | ||||
| 					{ | ||||
| 						property: 'og:image', | ||||
| 						content: userData.picture, | ||||
| 						noEscape: true, | ||||
| 					}, | ||||
| 					{ | ||||
| 						property: 'og:image:url', | ||||
| 						content: userData.picture, | ||||
| 						noEscape: true, | ||||
| 					} | ||||
| 				); | ||||
| 			} | ||||
| 			addMetaTags(res, userData); | ||||
|  | ||||
| 			userData.selectedGroup = userData.groups.filter(function (group) { | ||||
| 				return group && userData.groupTitleArray.includes(group.name); | ||||
| @@ -135,3 +101,74 @@ profileController.get = function (req, res, callback) { | ||||
| 		}, | ||||
| 	], callback); | ||||
| }; | ||||
|  | ||||
| function getLatestPosts(callerUid, userData, callback) { | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			db.getSortedSetRevRange('uid:' + userData.uid + ':posts', 0, 99, next); | ||||
| 		}, | ||||
| 		function (pids, next) { | ||||
| 			getPosts(callerUid, pids, next); | ||||
| 		}, | ||||
| 	], callback); | ||||
| } | ||||
|  | ||||
| function getBestPosts(callerUid, userData, callback) { | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			db.getSortedSetRevRange('uid:' + userData.uid + ':posts:votes', 0, 99, next); | ||||
| 		}, | ||||
| 		function (pids, next) { | ||||
| 			getPosts(callerUid, pids, next); | ||||
| 		}, | ||||
| 	], callback); | ||||
| } | ||||
|  | ||||
| function getPosts(callerUid, pids, callback) { | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			privileges.posts.filter('topics:read', pids, callerUid, next); | ||||
| 		}, | ||||
| 		function (pids, next) { | ||||
| 			pids = pids.slice(0, 10); | ||||
| 			posts.getPostSummaryByPids(pids, callerUid, { stripTags: false }, next); | ||||
| 		}, | ||||
| 	], callback); | ||||
| } | ||||
|  | ||||
| function addMetaTags(res, userData) { | ||||
| 	var plainAboutMe = userData.aboutme ? utils.stripHTMLTags(utils.decodeHTMLEntities(userData.aboutme)) : ''; | ||||
| 	res.locals.metaTags = [ | ||||
| 		{ | ||||
| 			name: 'title', | ||||
| 			content: userData.fullname || userData.username, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: 'description', | ||||
| 			content: plainAboutMe, | ||||
| 		}, | ||||
| 		{ | ||||
| 			property: 'og:title', | ||||
| 			content: userData.fullname || userData.username, | ||||
| 		}, | ||||
| 		{ | ||||
| 			property: 'og:description', | ||||
| 			content: plainAboutMe, | ||||
| 		}, | ||||
| 	]; | ||||
|  | ||||
| 	if (userData.picture) { | ||||
| 		res.locals.metaTags.push( | ||||
| 			{ | ||||
| 				property: 'og:image', | ||||
| 				content: userData.picture, | ||||
| 				noEscape: true, | ||||
| 			}, | ||||
| 			{ | ||||
| 				property: 'og:image:url', | ||||
| 				content: userData.picture, | ||||
| 				noEscape: true, | ||||
| 			} | ||||
| 		); | ||||
| 	} | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user