Files
amayo/src/components/modals/ldPointsModal.ts

185 lines
5.8 KiB
TypeScript
Raw Normal View History

import logger from "../../core/lib/logger";
import {
ModalSubmitInteraction,
MessageFlags,
PermissionFlagsBits,
EmbedBuilder
} from 'discord.js';
import { prisma } from '../../core/database/prisma';
export default {
customId: 'ld_points_modal',
run: async (interaction: ModalSubmitInteraction) => {
logger.info(`🔍 Modal ldPointsModal ejecutado. CustomId: ${interaction.customId}`);
if (!interaction.guild) {
return interaction.reply({
content: '❌ Solo disponible en servidores.',
flags: MessageFlags.Ephemeral
});
}
// Verificar permisos
const member = await interaction.guild.members.fetch(interaction.user.id);
if (!member.permissions.has(PermissionFlagsBits.ManageGuild)) {
return interaction.reply({
content: '❌ Solo los administradores pueden gestionar puntos.',
flags: MessageFlags.Ephemeral
});
}
try {
// Extraer el userId del customId (formato: ld_points_modal:userId)
const userId = interaction.customId.split(':')[1];
logger.info(`🔍 UserId extraído: ${userId}`);
if (!userId) {
return interaction.reply({
content: '❌ Error al identificar el usuario.',
flags: MessageFlags.Ephemeral
});
}
// Obtener valor del modal
// @ts-ignore
const totalInput = interaction.fields.getTextInputValue('total_points').trim();
logger.info(`🔍 Input recibido: ${totalInput}`);
if (!totalInput) {
return interaction.reply({
content: '❌ Debes ingresar un valor para modificar.',
flags: MessageFlags.Ephemeral
});
}
// Obtener o crear el registro de stats del usuario
let stats = await prisma.partnershipStats.findUnique({
where: {
userId_guildId: {
userId,
guildId: interaction.guild.id
}
}
});
if (!stats) {
logger.info(`🔍 Creando nuevo registro de stats para userId: ${userId}`);
// Crear nuevo registro si no existe
stats = await prisma.partnershipStats.create({
data: {
userId,
guildId: interaction.guild.id,
totalPoints: 0,
weeklyPoints: 0,
monthlyPoints: 0
}
});
}
logger.info(`🔍 Stats actuales - Total: ${stats.totalPoints}`);
// Función para parsear el input y calcular el nuevo valor
const calculateNewValue = (input: string, currentValue: number): number => {
const firstChar = input[0];
if (firstChar === '+') {
// Añadir puntos
const numValue = parseInt(input.substring(1)) || 0;
return Math.max(0, currentValue + numValue);
} else if (firstChar === '-') {
// Quitar puntos (los últimos N puntos añadidos)
const numValue = parseInt(input.substring(1)) || 0;
return Math.max(0, currentValue - numValue);
} else if (firstChar === '=') {
// Establecer valor absoluto
const numValue = parseInt(input.substring(1)) || 0;
return Math.max(0, numValue);
} else {
// Si no tiene símbolo, tratar como valor absoluto
const parsedValue = parseInt(input);
return isNaN(parsedValue) ? currentValue : Math.max(0, parsedValue);
}
};
// Calcular nuevo valor de puntos totales
const newTotalPoints = calculateNewValue(totalInput, stats.totalPoints);
logger.info(`🔍 Nuevo total calculado: ${newTotalPoints}`);
// Actualizar en base de datos (solo puntos totales)
await prisma.partnershipStats.update({
where: {
userId_guildId: {
userId,
guildId: interaction.guild.id
}
},
data: {
totalPoints: newTotalPoints
}
});
logger.info(`✅ Puntos actualizados exitosamente en la base de datos`);
// Obtener nombre del usuario
let userName = 'Usuario';
try {
const targetMember = await interaction.guild.members.fetch(userId);
userName = targetMember.displayName || targetMember.user.username;
} catch {
try {
const user = await interaction.client.users.fetch(userId);
userName = user.username;
} catch {
userName = userId;
}
}
// Calcular la diferencia
const difference = newTotalPoints - stats.totalPoints;
const diffText = difference > 0 ? `+${difference}` : `${difference}`;
// Crear embed de confirmación
const embed = new EmbedBuilder()
.setColor(difference >= 0 ? 0x00ff00 : 0xff9900)
.setTitle('✅ Puntos Actualizados')
.setDescription(`Se han actualizado los puntos de **${userName}**`)
.addFields(
{
name: '📊 Puntos Totales',
value: `${stats.totalPoints} → **${newTotalPoints}** (${diffText})`,
inline: false
},
{
name: '📝 Operación',
value: `\`${totalInput}\``,
inline: false
}
)
.setFooter({ text: `Modificado por ${interaction.user.username}` })
.setTimestamp();
await interaction.reply({
embeds: [embed],
flags: MessageFlags.Ephemeral
});
logger.info(`✅ Respuesta enviada al usuario`);
} catch (e) {
logger.error({ err: e }, 'Error en ldPointsModal');
// Intentar responder con el error
try {
if (!interaction.replied && !interaction.deferred) {
await interaction.reply({
content: '❌ Error al actualizar los puntos.',
flags: MessageFlags.Ephemeral
});
}
} catch (replyError) {
logger.error({ err: replyError }, 'Error al enviar respuesta de error');
}
}
}
};