- Added Directus hooks for automatic point awards: - Recording creation/publishing (50 points) - Recording featured status (100 points bonus) - Comments on recordings (5 points) - Created /leaderboard route with full UI - Server-side data loading with authentication guard - Responsive design with medal emojis for top 3 - User stats display (recordings, plays, achievements) - Pagination support - "How It Works" info section - Added comprehensive gamification translations - Time-weighted scoring displayed for rankings - Automatic achievement checking on point awards 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
146 lines
4.7 KiB
TypeScript
146 lines
4.7 KiB
TypeScript
import { createRequire } from "module";
|
|
global.require = createRequire(import.meta.url);
|
|
import { defineHook } from "@directus/extensions-sdk";
|
|
import slugify from "@sindresorhus/slugify";
|
|
import ffmpeg from "fluent-ffmpeg";
|
|
import { awardPoints, checkAchievements } from "../endpoint/gamification.js";
|
|
|
|
async function processVideo(
|
|
meta,
|
|
{ schema, accountability },
|
|
services,
|
|
logger,
|
|
) {
|
|
const { FilesService } = services;
|
|
const itemId = meta.key;
|
|
const videoPath = `/directus/uploads/${meta.payload.filename_disk}`; // Adjust path as needed
|
|
const videoService = new FilesService({ schema, accountability }); // Replace with your collection name
|
|
|
|
try {
|
|
const durationInSeconds = await new Promise((resolve, reject) => {
|
|
ffmpeg.ffprobe(videoPath, function (err, metadata) {
|
|
if (err) {
|
|
reject(err);
|
|
}
|
|
resolve(parseInt(metadata.format.duration));
|
|
});
|
|
});
|
|
// Update the item with the duration
|
|
await videoService.updateOne(itemId, { duration: durationInSeconds });
|
|
logger.info(`Video ${itemId} duration updated to ${durationInSeconds}`);
|
|
} catch (error) {
|
|
logger.error(`Error processing video ${itemId}:`, error);
|
|
}
|
|
}
|
|
|
|
export default defineHook(async ({ filter, action }, { services, logger, database, getSchema }) => {
|
|
action("files.upload", async (meta, context) => {
|
|
await processVideo(meta, context, services, logger);
|
|
});
|
|
|
|
filter(
|
|
"users.create",
|
|
(payload: {
|
|
first_name: string;
|
|
last_name: string;
|
|
artist_name: string;
|
|
slug: string;
|
|
}) => {
|
|
const artist_name = `${payload.first_name}-${new Date().getTime()}`;
|
|
const slug = slugify(artist_name);
|
|
const join_date = new Date();
|
|
return { ...payload, artist_name, slug, join_date };
|
|
},
|
|
);
|
|
|
|
filter(
|
|
"users.update",
|
|
(payload: {
|
|
first_name: string;
|
|
last_name: string;
|
|
artist_name: string;
|
|
slug: string;
|
|
}) => {
|
|
if (payload.artist_name) {
|
|
const slug = slugify(payload.artist_name);
|
|
return { ...payload, slug };
|
|
}
|
|
return payload;
|
|
},
|
|
);
|
|
|
|
// =========================================
|
|
// GAMIFICATION HOOKS
|
|
// =========================================
|
|
|
|
// Hook: Award points when recording is published
|
|
action("items.create", async (meta, { collection, accountability }) => {
|
|
if (collection === "sexy_recordings") {
|
|
const { payload, key } = meta;
|
|
|
|
// Award points if recording is published
|
|
if (payload.status === "published" && accountability?.user) {
|
|
try {
|
|
await awardPoints(database, accountability.user, "RECORDING_CREATE", key);
|
|
await checkAchievements(database, accountability.user, "recordings");
|
|
logger.info(`Awarded RECORDING_CREATE points to user ${accountability.user}`);
|
|
} catch (error) {
|
|
logger.error("Failed to award recording creation points:", error);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
// Hook: Award points when recording status changes to published or featured
|
|
action("items.update", async (meta, { collection, accountability, schema }) => {
|
|
if (collection === "sexy_recordings") {
|
|
const { payload, keys } = meta;
|
|
|
|
try {
|
|
const { ItemsService } = services;
|
|
const recordingsService = new ItemsService("sexy_recordings", {
|
|
schema: await getSchema(),
|
|
});
|
|
|
|
for (const key of keys) {
|
|
const recording = await recordingsService.readOne(key);
|
|
|
|
// Award points if status changed from non-published to published
|
|
if (payload.status === "published" && recording.status !== "published" && recording.user_created) {
|
|
await awardPoints(database, recording.user_created, "RECORDING_CREATE", key);
|
|
await checkAchievements(database, recording.user_created, "recordings");
|
|
logger.info(`Awarded RECORDING_CREATE points to user ${recording.user_created}`);
|
|
}
|
|
|
|
// Award bonus points if recording becomes featured
|
|
if (payload.featured === true && !recording.featured && recording.user_created) {
|
|
await awardPoints(database, recording.user_created, "RECORDING_FEATURED", key);
|
|
await checkAchievements(database, recording.user_created, "recordings");
|
|
logger.info(`Awarded RECORDING_FEATURED points to user ${recording.user_created}`);
|
|
}
|
|
}
|
|
} catch (error) {
|
|
logger.error("Failed to award recording update points:", error);
|
|
}
|
|
}
|
|
});
|
|
|
|
// Hook: Award points when user creates a comment on a recording
|
|
action("comments.create", async (meta, { accountability }) => {
|
|
if (!accountability?.user) return;
|
|
|
|
try {
|
|
const { payload } = meta;
|
|
|
|
// Check if comment is on a recording
|
|
if (payload.collection === "sexy_recordings") {
|
|
await awardPoints(database, accountability.user, "COMMENT_CREATE");
|
|
await checkAchievements(database, accountability.user, "social");
|
|
logger.info(`Awarded COMMENT_CREATE points to user ${accountability.user}`);
|
|
}
|
|
} catch (error) {
|
|
logger.error("Failed to award comment points:", error);
|
|
}
|
|
});
|
|
});
|