feat: implement comprehensive leaderboard feature

Implements issue #55 with multiple leaderboard categories:
- Top Suggestions (by count and acceptance rate)
- Top Likes (by total likes given)
- Top Comments (by total comments posted)
- Overall Leaders (weighted by achievement points and engagement diversity)

Features:
- Tabbed UI with reactive state management
- Medal indicators for top 3 positions
- User avatars and badges display
- Current user highlighting
- Privacy controls via profilePublic setting
- Configurable result limits (max 100)
- Detailed statistics per category

Backend:
- Created LeaderboardService with aggregation logic
- Filters for public profiles and non-banned users
- Efficient sorting algorithms for each category
- Parallel data fetching for all leaderboards

Frontend:
- Standalone Angular component with signals
- Responsive card-based layout
- Integration with existing user profile system
- Navigation link in header dropdown

Technical notes:
- Uses Fastify AutoLoad with FastifyPluginAsync pattern
- Shared types across monorepo for type safety
- Leverages existing achievement system data
This commit is contained in:
2026-02-19 23:31:41 -08:00
committed by Naomi Carrigan
parent 8f95f57838
commit f839059dd2
9 changed files with 976 additions and 3 deletions
+85
View File
@@ -0,0 +1,85 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import { FastifyPluginAsync } from "fastify";
import type {
LeaderboardResponse,
SuggestionsLeaderboard,
LikesLeaderboard,
CommentsLeaderboard,
OverallLeaderboard,
} from "@library/shared-types";
import { LeaderboardService } from "../../services/leaderboard.service";
const leaderboardRoutes: FastifyPluginAsync = async (app) => {
const leaderboardService = new LeaderboardService();
/**
* Get all leaderboards at once.
*/
app.get<{
Querystring: { limit?: number };
Reply: LeaderboardResponse;
}>("/", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getAllLeaderboards(limit);
});
/**
* Get top users by suggestions.
*/
app.get<{
Querystring: { limit?: number };
Reply: SuggestionsLeaderboard[];
}>("/suggestions", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopSuggestions(limit);
});
/**
* Get top users by likes.
*/
app.get<{
Querystring: { limit?: number };
Reply: LikesLeaderboard[];
}>("/likes", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopLikes(limit);
});
/**
* Get top users by comments.
*/
app.get<{
Querystring: { limit?: number };
Reply: CommentsLeaderboard[];
}>("/comments", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getTopComments(limit);
});
/**
* Get overall leaderboard.
*/
app.get<{
Querystring: { limit?: number };
Reply: OverallLeaderboard[];
}>("/overall", async (request) => {
const limit = request.query.limit && request.query.limit > 0
? Math.min(request.query.limit, 100)
: 25;
return leaderboardService.getOverallLeaderboard(limit);
});
};
export default leaderboardRoutes;
+222
View File
@@ -0,0 +1,222 @@
/**
* @copyright 2026 NHCarrigan
* @license Naomi's Public License
* @author Naomi Carrigan
*/
import type {
LeaderboardResponse,
SuggestionsLeaderboard,
LikesLeaderboard,
CommentsLeaderboard,
OverallLeaderboard,
} from "@library/shared-types";
import { prisma } from "../lib/prisma";
export class LeaderboardService {
private prisma = prisma;
constructor() {}
/**
* Get top users by suggestions submitted and accepted.
*/
async getTopSuggestions(limit = 25): Promise<SuggestionsLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
suggestions: {
select: {
status: true,
},
},
},
});
const leaderboard = users
.map((user) => {
const totalSuggestions = user.suggestions.length;
const acceptedSuggestions = user.suggestions.filter(
(s) => s.status === "ACCEPTED"
).length;
const acceptanceRate =
totalSuggestions > 0
? Math.round((acceptedSuggestions / totalSuggestions) * 100)
: 0;
return {
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalSuggestions,
acceptedSuggestions,
acceptanceRate,
};
})
.filter((user) => user.totalSuggestions > 0)
.sort((a, b) => {
if (b.totalSuggestions !== a.totalSuggestions) {
return b.totalSuggestions - a.totalSuggestions;
}
return b.acceptanceRate - a.acceptanceRate;
})
.slice(0, limit);
return leaderboard;
}
/**
* Get top users by likes given.
*/
async getTopLikes(limit = 25): Promise<LikesLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
likes: true,
},
});
const leaderboard = users
.map((user) => ({
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalLikes: user.likes.length,
}))
.filter((user) => user.totalLikes > 0)
.sort((a, b) => b.totalLikes - a.totalLikes)
.slice(0, limit);
return leaderboard;
}
/**
* Get top users by comments posted.
*/
async getTopComments(limit = 25): Promise<CommentsLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
comments: true,
},
});
const leaderboard = users
.map((user) => ({
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalComments: user.comments.length,
}))
.filter((user) => user.totalComments > 0)
.sort((a, b) => b.totalComments - a.totalComments)
.slice(0, limit);
return leaderboard;
}
/**
* Get overall leaderboard based on combined engagement.
*/
async getOverallLeaderboard(limit = 25): Promise<OverallLeaderboard[]> {
const users = await this.prisma.user.findMany({
where: { profilePublic: true, isBanned: false },
include: {
suggestions: true,
likes: true,
comments: true,
userAchievements: true,
},
});
const leaderboard = users
.map((user) => {
const totalSuggestions = user.suggestions.length;
const totalLikes = user.likes.length;
const totalComments = user.comments.length;
const achievementCount = user.userAchievements.length;
const diversityScore =
(totalSuggestions > 0 ? 1 : 0) +
(totalLikes > 0 ? 1 : 0) +
(totalComments > 0 ? 1 : 0);
return {
id: user.id,
username: user.username,
slug: user.slug,
avatar: user.avatar,
primaryBadge: user.primaryBadge,
isVip: user.isVip,
isMod: user.isMod,
isStaff: user.isStaff,
createdAt: user.createdAt,
totalSuggestions,
totalLikes,
totalComments,
achievementCount,
achievementPoints: user.achievementPoints,
currentStreak: user.currentStreak,
diversityScore,
};
})
.filter(
(user) =>
user.totalSuggestions > 0 ||
user.totalLikes > 0 ||
user.totalComments > 0
)
.sort((a, b) => {
if (b.achievementPoints !== a.achievementPoints) {
return b.achievementPoints - a.achievementPoints;
}
if (b.diversityScore !== a.diversityScore) {
return b.diversityScore - a.diversityScore;
}
const totalA = a.totalSuggestions + a.totalLikes + a.totalComments;
const totalB = b.totalSuggestions + b.totalLikes + b.totalComments;
return totalB - totalA;
})
.slice(0, limit);
return leaderboard;
}
/**
* Get all leaderboards at once.
*/
async getAllLeaderboards(limit = 25): Promise<LeaderboardResponse> {
const [topSuggestions, topLikes, topComments, topOverall] =
await Promise.all([
this.getTopSuggestions(limit),
this.getTopLikes(limit),
this.getTopComments(limit),
this.getOverallLeaderboard(limit),
]);
return {
topSuggestions,
topLikes,
topComments,
topOverall,
};
}
}