This commit is contained in:
2026-03-15 12:22:42 +01:00
parent cd99275933
commit 311ba5e7f3
558 changed files with 55182 additions and 22981 deletions

View File

@@ -1,86 +1,79 @@
const { SlashCommandBuilder, EmbedBuilder } = require('discord.js');
const { SlashCommandBuilder, EmbedBuilder, MessageFlags } = require('discord.js');
const db = require('../../functions/database/db.js');
const { colors } = require('../../utils/constants.js');
const { colors, emojis } = require('../../utils/constants.js');
module.exports = {
category: 'xp',
data: new SlashCommandBuilder()
.setName('bumptime')
.setDescription('Vérifier le dernier bump et le temps restant avant le prochain'),
async execute(interaction) {
await interaction.deferReply({ ephemeral: true });
try {
// Récupérer le dernier bump pour ce serveur
const [bumps] = await db.query(
'SELECT * FROM bumps WHERE guildId = ? ORDER BY bumpTime DESC LIMIT 1',
[interaction.guild.id]
);
if (bumps.length === 0) {
return interaction.editReply({
content: '❌ Aucun bump enregistré pour ce serveur.'
});
}
const lastBump = bumps[0];
const bumpTime = lastBump.bumpTime;
const now = Date.now();
const timeSinceBump = now - bumpTime;
const twoHours = 2 * 60 * 60 * 1000;
const timeUntilNextBump = twoHours - timeSinceBump;
// Récupérer l'utilisateur qui a fait le bump
const user = await interaction.client.users.fetch(lastBump.userId).catch(() => null);
const userTag = user ? user.tag : `ID: ${lastBump.userId}`;
const { emojis } = require('../../utils/constants');
const embed = new EmbedBuilder()
.setAuthor({
name: `${interaction.guild.name}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
if (timeUntilNextBump > 0) {
// Le prochain bump n'est pas encore disponible
const hours = Math.floor(timeUntilNextBump / (60 * 60 * 1000));
const minutes = Math.floor((timeUntilNextBump % (60 * 60 * 1000)) / (60 * 1000));
const seconds = Math.floor((timeUntilNextBump % (60 * 1000)) / 1000);
embed.setTitle('⏳ Bump en Cooldown')
.setColor(colors.warning)
.setDescription(`Le dernier bump a été effectué il y a moins de 2 heures.`)
.addFields(
{ name: '👤 Dernier bump par', value: `\`${userTag}\``, inline: true },
{ name: '⏰ Il y a', value: `\`${Math.floor(timeSinceBump / (60 * 60 * 1000))}h ${Math.floor((timeSinceBump % (60 * 60 * 1000)) / (60 * 1000))}m\``, inline: true },
{ name: '🕐 Temps restant', value: `\`${hours}h ${minutes}m ${seconds}s\``, inline: true },
{ name: '📅 Prochain bump disponible', value: `<t:${Math.floor((bumpTime + twoHours) / 1000)}:R>`, inline: false }
)
.setFooter({ text: 'Utilise /bump (Disboard) pour bump le serveur' });
} else {
// Le prochain bump est disponible
embed.setTitle(`${emojis.success} Bump Disponible`)
.setColor(colors.success)
.setDescription(`**Le serveur peut être bumpé maintenant !**`)
.addFields(
{ name: '👤 Dernier bump par', value: `\`${userTag}\``, inline: true },
{ name: '⏰ Il y a', value: `\`${Math.floor(timeSinceBump / (60 * 60 * 1000))}h ${Math.floor((timeSinceBump % (60 * 60 * 1000)) / (60 * 1000))}m\``, inline: true },
{ name: '✅ Statut', value: '**Disponible**', inline: true },
{ name: '💡 Astuce', value: 'Utilise `/bump` (Disboard) pour bump le serveur !', inline: false }
)
.setFooter({ text: 'Utilise /bump (Disboard) pour bump le serveur' });
}
await interaction.editReply({ embeds: [embed] });
} catch (err) {
console.error('Erreur lors de la vérification du bump:', err);
await interaction.editReply({
content: '❌ Erreur lors de la vérification du bump.'
});
}
},
};
category: 'xp',
data: new SlashCommandBuilder()
.setName('bumptime')
.setDescription('Vérifier le dernier bump et le temps restant avant le prochain'),
async execute(interaction) {
await interaction.deferReply({ flags: MessageFlags.Ephemeral });
try {
const [bumps] = await db.query(
'SELECT * FROM bumps WHERE guildId = ? ORDER BY bumpTime DESC LIMIT 1',
[interaction.guild.id]
);
if (bumps.length === 0) {
return interaction.editReply({
content: '❌ Aucun bump enregistré pour ce serveur.'
});
}
const lastBump = bumps[0];
const bumpTime = parseInt(lastBump.bumpTime, 10);
const now = Date.now();
const timeSinceBump = now - bumpTime;
const twoHours = 2 * 60 * 60 * 1000;
const timeUntilNextBump = twoHours - timeSinceBump;
const user = await interaction.client.users.fetch(lastBump.userId).catch(() => null);
const userTag = user ? user.tag : `ID: ${lastBump.userId}`;
const embed = new EmbedBuilder()
.setAuthor({
name: interaction.guild.name,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
if (timeUntilNextBump > 0) {
const hours = Math.floor(timeUntilNextBump / (60 * 60 * 1000));
const minutes = Math.floor((timeUntilNextBump % (60 * 60 * 1000)) / (60 * 1000));
const seconds = Math.floor((timeUntilNextBump % (60 * 1000)) / 1000);
embed.setTitle('⏳ Bump en Cooldown')
.setColor(colors.warning)
.setDescription(`Le dernier bump a été effectué il y a moins de 2 heures.`)
.addFields(
{ name: '👤 Dernier bump par', value: `\`${userTag}\``, inline: true },
{ name: '⏰ Il y a', value: `\`${Math.floor(timeSinceBump / (60 * 60 * 1000))}h ${Math.floor((timeSinceBump % (60 * 60 * 1000)) / (60 * 1000))}m\``, inline: true },
{ name: '🕐 Temps restant', value: `\`${hours}h ${minutes}m ${seconds}s\``, inline: true },
{ name: '📅 Prochain bump disponible', value: `<t:${Math.floor((bumpTime + twoHours) / 1000)}:R>`, inline: false }
)
.setFooter({ text: 'Utilise /bump (Disboard) pour bump le serveur' });
} else {
embed.setTitle(`${emojis.success} Bump Disponible`)
.setColor(colors.success)
.setDescription(`**Le serveur peut être bumpé maintenant !**`)
.addFields(
{ name: '👤 Dernier bump par', value: `\`${userTag}\``, inline: true },
{ name: '⏰ Il y a', value: `\`${Math.floor(timeSinceBump / (60 * 60 * 1000))}h ${Math.floor((timeSinceBump % (60 * 60 * 1000)) / (60 * 1000))}m\``, inline: true },
{ name: '✅ Statut', value: '**Disponible**', inline: true },
{ name: '💡 Astuce', value: 'Utilise `/bump` (Disboard) pour bump le serveur !', inline: false }
)
.setFooter({ text: 'Utilise /bump (Disboard) pour bump le serveur' });
}
await interaction.editReply({ embeds: [embed] });
} catch (err) {
console.error('Erreur bumptime:', err);
await interaction.editReply({
content: '❌ Erreur lors de la vérification du bump.'
});
}
},
};

View File

@@ -1,72 +1,65 @@
const { SlashCommandBuilder, EmbedBuilder } = require('discord.js');
const { getLeaderboard, getXPProgress } = require('../../functions/xp/xp.js');
const { colors } = require('../../utils/constants');
const { colors, emojis } = require('../../utils/constants');
module.exports = {
category: 'xp',
data: new SlashCommandBuilder()
.setName('leaderboard')
.setDescription('Affiche le classement des niveaux')
.addIntegerOption(option =>
option.setName('limit')
.setDescription('Nombre d\'utilisateurs à afficher (par défaut: 10)')
.setRequired(false)
.setMinValue(1)
.setMaxValue(25)),
async execute(interaction) {
await interaction.deferReply();
const limit = interaction.options.getInteger('limit') || 10;
const leaderboard = await getLeaderboard(interaction.guild.id, limit);
if (leaderboard.length === 0) {
return interaction.editReply({
content: '❌ Aucun utilisateur trouvé dans le classement.'
});
}
// Récupérer les informations des utilisateurs
const leaderboardData = [];
for (const user of leaderboard) {
try {
const member = await interaction.guild.members.fetch(user.userId).catch(() => null);
if (member) {
leaderboardData.push({
user: member.user,
xp: user.xp,
level: user.level
});
}
} catch (err) {
// Ignorer les erreurs
}
}
const { emojis } = require('../../utils/constants');
// Créer la description du classement
const medals = ['🥇', '🥈', '🥉'];
const description = leaderboardData.map((data, index) => {
const medal = medals[index] || `\`${index + 1}.\``;
const progress = getXPProgress(data.xp, data.level);
return `${medal} **${data.user.displayName}** • ${emojis.level} Niveau \`${data.level}\`${emojis.xp} \`${data.xp.toLocaleString()} XP\`\`${progress.percentage}%\``;
}).join('\n');
const embed = new EmbedBuilder()
.setAuthor({
name: `${interaction.guild.name}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTitle(`${emojis.leaderboard} Classement des Niveaux`)
.setColor(colors.xp)
.setDescription(description || 'Aucun utilisateur dans le classement')
.setFooter({
text: `Top ${limit}${interaction.guild.name}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
},
};
category: 'xp',
data: new SlashCommandBuilder()
.setName('leaderboard')
.setDescription('Affiche le classement des niveaux')
.addIntegerOption(option =>
option.setName('limit')
.setDescription('Nombre d\'utilisateurs à afficher (par défaut: 10)')
.setRequired(false)
.setMinValue(1)
.setMaxValue(25)),
async execute(interaction) {
await interaction.deferReply();
const limit = interaction.options.getInteger('limit') || 10;
const leaderboard = await getLeaderboard(interaction.guild.id, limit);
if (leaderboard.length === 0) {
return interaction.editReply({
content: '❌ Aucun utilisateur trouvé dans le classement.'
});
}
const leaderboardData = [];
for (const user of leaderboard) {
try {
const member = await interaction.guild.members.fetch(user.userId).catch(() => null);
if (member) {
leaderboardData.push({
user: member.user,
xp: user.xp,
level: user.level
});
}
} catch (err) { /* Ignorer */ }
}
const medals = ['🥇', '🥈', '🥉'];
const description = leaderboardData.map((data, index) => {
const medal = medals[index] || `\`${index + 1}.\``;
const progress = getXPProgress(data.xp, data.level);
return `${medal} **${data.user.displayName}** • ${emojis.level} Niveau \`${data.level}\`${emojis.xp} \`${data.xp.toLocaleString()} XP\`\`${progress.percentage}%\``;
}).join('\n');
const embed = new EmbedBuilder()
.setAuthor({
name: interaction.guild.name,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTitle(`${emojis.leaderboard} Classement des Niveaux`)
.setColor(colors.xp)
.setDescription(description || 'Aucun utilisateur dans le classement')
.setFooter({
text: `Top ${limit}${interaction.guild.name}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
},
};

View File

@@ -1,96 +1,97 @@
const { SlashCommandBuilder, EmbedBuilder } = require('discord.js');
const { getUserXP, getXPProgress, getXPForNextLevel, getUserRank } = require('../../functions/xp/xp.js');
const { colors } = require('../../utils/constants');
const { colors, emojis } = require('../../utils/constants');
module.exports = {
category: 'xp',
data: new SlashCommandBuilder()
.setName('level')
.setDescription('Affiche ton niveau et ton XP')
.addUserOption(option =>
option.setName('user')
.setDescription('L\'utilisateur dont tu veux voir le niveau')
.setRequired(false)),
async execute(interaction) {
await interaction.deferReply();
const target = interaction.options.getUser('user') || interaction.user;
const userXP = await getUserXP(target.id, interaction.guild.id);
if (!userXP) {
return interaction.editReply({
content: '❌ Erreur lors de la récupération des données XP.'
});
}
const progress = getXPProgress(userXP.xp, userXP.level);
const xpForNextLevel = getXPForNextLevel(userXP.level);
const rank = await getUserRank(target.id, interaction.guild.id);
// Créer une barre de progression visuelle
const barLength = 20;
let filled = 0;
let empty = barLength;
// Calculer la barre de progression seulement si progress.needed > 0
if (progress.needed > 0 && progress.current >= 0) {
const percentage = Math.min(100, Math.max(0, (progress.current / progress.needed) * 100));
filled = Math.max(0, Math.min(barLength, Math.floor((percentage / 100) * barLength)));
empty = Math.max(0, barLength - filled);
}
const progressBar = '█'.repeat(filled) + '░'.repeat(empty);
const { emojis } = require('../../utils/constants');
const embed = new EmbedBuilder()
.setAuthor({
name: `${target.displayName}`,
iconURL: target.displayAvatarURL({ dynamic: true })
})
.setTitle(`${emojis.level} Niveau ${userXP.level}`)
.setColor(colors.xp)
.setThumbnail(target.displayAvatarURL({ dynamic: true, size: 256 }))
.setDescription(`**${emojis.rank} Rang #${rank}** sur ${interaction.guild.name}`)
.addFields(
{
name: `${emojis.xp} XP Total`,
value: `\`${userXP.xp.toLocaleString()} XP\``,
inline: true
},
{
name: '📈 Progression',
value: `\`${progress.current}/${progress.needed} XP\` (${progress.percentage}%)`,
inline: true
},
{
name: '🎯 Prochain Niveau',
value: `\`${xpForNextLevel} XP\` requis`,
inline: true
},
{
name: '📊 Barre de Progression',
value: `\`\`\`${progressBar}\`\`\``,
inline: false
},
{
name: '💬 Messages',
value: `\`${(userXP.totalMessages || 0).toLocaleString()}\``,
inline: true
},
{
name: '🎤 Temps Vocal',
value: `\`${Math.floor((userXP.totalVoiceTime || 0) / 60)}h\``,
inline: true
}
)
.setFooter({
text: `${interaction.guild.name} • Niveau ${userXP.level}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
},
};
category: 'xp',
data: new SlashCommandBuilder()
.setName('level')
.setDescription('Affiche ton niveau et ton XP')
.addUserOption(option =>
option.setName('user')
.setDescription('L\'utilisateur dont tu veux voir le niveau')
.setRequired(false)),
async execute(interaction) {
await interaction.deferReply();
const target = interaction.options.getUser('user') || interaction.user;
const userXP = await getUserXP(target.id, interaction.guild.id);
if (!userXP) {
return interaction.editReply({
content: '❌ Erreur lors de la récupération des données XP.'
});
}
const progress = getXPProgress(userXP.xp, userXP.level);
const xpForNextLevel = getXPForNextLevel(userXP.level);
const rank = await getUserRank(target.id, interaction.guild.id);
const barLength = 20;
let filled = 0;
let empty = barLength;
if (progress.needed > 0 && progress.current >= 0) {
const percentage = Math.min(100, Math.max(0, (progress.current / progress.needed) * 100));
filled = Math.max(0, Math.min(barLength, Math.floor((percentage / 100) * barLength)));
empty = Math.max(0, barLength - filled);
}
const progressBar = '█'.repeat(filled) + '░'.repeat(empty);
// Formatage du temps vocal
const totalMinutes = userXP.totalVoiceTime || 0;
const hours = Math.floor(totalMinutes / 60);
const minutes = totalMinutes % 60;
const voiceTimeStr = `${hours}h ${minutes}m`;
const embed = new EmbedBuilder()
.setAuthor({
name: target.displayName,
iconURL: target.displayAvatarURL({ dynamic: true })
})
.setTitle(`${emojis.level} Niveau ${userXP.level}`)
.setColor(colors.xp)
.setThumbnail(target.displayAvatarURL({ dynamic: true, size: 256 }))
.setDescription(`**${emojis.rank} Rang #${rank}** sur ${interaction.guild.name}`)
.addFields(
{
name: `${emojis.xp} XP Total`,
value: `\`${userXP.xp.toLocaleString()} XP\``,
inline: true
},
{
name: '📈 Progression',
value: `\`${progress.current}/${progress.needed} XP\` (${progress.percentage}%)`,
inline: true
},
{
name: '🎯 Prochain Niveau',
value: `\`${xpForNextLevel} XP\` requis`,
inline: true
},
{
name: '📊 Barre de Progression',
value: `\`\`\`${progressBar}\`\`\``,
inline: false
},
{
name: '💬 Messages',
value: `\`${(userXP.totalMessages || 0).toLocaleString()}\``,
inline: true
},
{
name: '🎤 Temps Vocal',
value: `\`${voiceTimeStr}\``,
inline: true
}
)
.setFooter({
text: `${interaction.guild.name} • Niveau ${userXP.level}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
},
};

View File

@@ -1,165 +1,116 @@
const { SlashCommandBuilder, EmbedBuilder, PermissionFlagsBits } = require('discord.js');
const { SlashCommandBuilder, EmbedBuilder, PermissionFlagsBits, MessageFlags } = require('discord.js');
const db = require('../../functions/database/db.js');
const { getUserXP } = require('../../functions/xp/xp.js');
const { colors } = require('../../utils/constants');
const { colors, emojis } = require('../../utils/constants');
const { sendLog } = require('../../utils/helpers');
module.exports = {
category: 'xp',
data: new SlashCommandBuilder()
.setName('resetlevel')
.setDescription('Réinitialiser le niveau d\'un utilisateur (Admin/Modérateur uniquement)')
.addUserOption(option =>
option.setName('user')
.setDescription('L\'utilisateur dont tu veux réinitialiser le niveau')
.setRequired(true))
.addStringOption(option =>
option.setName('reason')
.setDescription('Raison du reset (optionnel)')
.setRequired(false))
.setDefaultMemberPermissions(PermissionFlagsBits.ManageMessages),
async execute(interaction) {
await interaction.deferReply({ ephemeral: true });
// Vérifier les permissions (Modérateur ou Admin)
if (!interaction.member.permissions.has(PermissionFlagsBits.ManageMessages) &&
!interaction.member.permissions.has(PermissionFlagsBits.Administrator)) {
return interaction.editReply({
content: '❌ Tu n\'as pas la permission d\'utiliser cette commande. (Modérateur/Admin requis)'
});
}
const target = interaction.options.getUser('user');
const reason = interaction.options.getString('reason') || 'Aucune raison fournie';
if (!target) {
return interaction.editReply({
content: '❌ Aucun utilisateur spécifié !'
});
}
// Vérifier que l'utilisateur ne se reset pas lui-même
if (target.id === interaction.user.id) {
return interaction.editReply({
content: '❌ Tu ne peux pas te reset toi-même.'
});
}
// Vérifier que l'utilisateur ne reset pas le propriétaire du serveur
if (target.id === interaction.guild.ownerId) {
return interaction.editReply({
content: '❌ Tu ne peux pas reset le niveau du propriétaire du serveur.'
});
}
// Vérifier les rôles (ne pas permettre de reset quelqu'un avec un rôle supérieur)
const targetMember = await interaction.guild.members.fetch(target.id).catch(() => null);
if (targetMember) {
if (targetMember.roles.highest.position >= interaction.member.roles.highest.position &&
interaction.guild.ownerId !== interaction.user.id) {
return interaction.editReply({
content: '❌ Tu ne peux pas reset le niveau d\'un utilisateur avec un rôle supérieur ou égal au tien.'
});
}
}
try {
// Récupérer les données XP de l'utilisateur avant le reset
const userXP = await getUserXP(target.id, interaction.guild.id);
if (!userXP || userXP.xp === 0) {
return interaction.editReply({
content: `${target.tag} n'a aucun niveau/XP à réinitialiser.`
});
}
const oldLevel = userXP.level;
const oldXP = userXP.xp;
const oldMessages = userXP.totalMessages || 0;
const oldVoiceTime = userXP.totalVoiceTime || 0;
// Supprimer complètement l'utilisateur de la table user_xp
await db.query(
'DELETE FROM user_xp WHERE userId = ? AND guildId = ?',
[target.id, interaction.guild.id]
);
// Supprimer aussi les logs XP de l'utilisateur (optionnel, pour un reset complet)
await db.query(
'DELETE FROM xp_logs WHERE userId = ? AND guildId = ?',
[target.id, interaction.guild.id]
);
// Logger l'action dans la table logs
await db.query(
`INSERT INTO logs (userId, userTag, modId, modTag, action, reason, type, guildId, timestamp)
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,
[
target.id,
target.tag,
interaction.user.id,
interaction.user.tag,
'Reset de niveau',
`Niveau ${oldLevel} (${oldXP} XP) réinitialisé. Raison: ${reason}`,
'Permanent',
interaction.guild.id,
Date.now()
]
);
const { emojis } = require('../../utils/constants');
// Envoyer un log dans le canal de logs
const logEmbed = new EmbedBuilder()
.setAuthor({
name: `${interaction.user.displayName}`,
iconURL: interaction.user.displayAvatarURL({ dynamic: true })
})
.setTitle(`${emojis.warning} Reset de Niveau`)
.setColor(colors.warning)
.setDescription(`Le niveau de ${target.toString()} a été réinitialisé.`)
.setThumbnail(target.displayAvatarURL({ dynamic: true }))
.addFields(
{ name: '👤 Utilisateur', value: `${target.toString()}\n\`${target.tag}\``, inline: true },
{ name: '👮 Modérateur', value: `${interaction.user.toString()}\n\`${interaction.user.tag}\``, inline: true },
{ name: '📊 Niveau avant', value: `\`${oldLevel}\``, inline: true },
{ name: '📈 XP avant', value: `\`${oldXP.toLocaleString()} XP\``, inline: true },
{ name: '💬 Messages', value: `\`${oldMessages.toLocaleString()}\``, inline: true },
{ name: '🎤 Temps vocal', value: `\`${Math.floor(oldVoiceTime / 60)}h\``, inline: true },
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: `ID: ${target.id}${interaction.guild.name}` })
.setTimestamp();
await sendLog(interaction.guild, { embeds: [logEmbed] });
// Réponse à l'utilisateur
const embed = new EmbedBuilder()
.setAuthor({
name: `${target.displayName}`,
iconURL: target.displayAvatarURL({ dynamic: true })
})
.setTitle(`${emojis.success} Niveau Réinitialisé`)
.setColor(colors.success)
.setDescription(`Le niveau de ${target.toString()} a été complètement réinitialisé.`)
.setThumbnail(target.displayAvatarURL({ dynamic: true }))
.addFields(
{ name: '👤 Utilisateur', value: `\`${target.tag}\``, inline: true },
{ name: '📊 Niveau avant', value: `\`${oldLevel}\``, inline: true },
{ name: '📈 XP avant', value: `\`${oldXP.toLocaleString()} XP\``, inline: true },
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: 'Toutes les données XP ont été supprimées de la base de données.' })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} catch (err) {
console.error('Erreur lors du reset de niveau:', err);
await interaction.editReply({
content: `❌ Erreur lors du reset de niveau: ${err.message}`
});
}
},
};
category: 'xp',
data: new SlashCommandBuilder()
.setName('resetlevel')
.setDescription('Réinitialiser le niveau d\'un utilisateur (Admin/Modérateur uniquement)')
.addUserOption(option =>
option.setName('user')
.setDescription('L\'utilisateur dont tu veux réinitialiser le niveau')
.setRequired(true))
.addStringOption(option =>
option.setName('reason')
.setDescription('Raison du reset (optionnel)')
.setRequired(false))
.setDefaultMemberPermissions(PermissionFlagsBits.ManageMessages),
async execute(interaction) {
await interaction.deferReply({ flags: MessageFlags.Ephemeral });
const target = interaction.options.getUser('user');
const reason = interaction.options.getString('reason') || 'Aucune raison fournie';
if (!target) return interaction.editReply({ content: '❌ Aucun utilisateur spécifié !' });
if (target.id === interaction.user.id) return interaction.editReply({ content: '❌ Tu ne peux pas te reset toi-même.' });
if (target.id === interaction.guild.ownerId) return interaction.editReply({ content: '❌ Tu ne peux pas reset le niveau du propriétaire du serveur.' });
const targetMember = await interaction.guild.members.fetch(target.id).catch(() => null);
if (targetMember) {
if (targetMember.roles.highest.position >= interaction.member.roles.highest.position &&
interaction.guild.ownerId !== interaction.user.id) {
return interaction.editReply({
content: '❌ Tu ne peux pas reset le niveau d\'un utilisateur avec un rôle supérieur ou égal au tien.'
});
}
}
try {
const userXP = await getUserXP(target.id, interaction.guild.id);
if (!userXP || userXP.xp === 0) {
return interaction.editReply({
content: `${target.tag} n'a aucun niveau/XP à réinitialiser.`
});
}
const oldLevel = userXP.level;
const oldXP = userXP.xp;
const oldMessages = userXP.totalMessages || 0;
const oldVoiceTime = userXP.totalVoiceTime || 0;
await db.query('DELETE FROM user_xp WHERE userId = ? AND guildId = ?', [target.id, interaction.guild.id]);
await db.query('DELETE FROM xp_logs WHERE userId = ? AND guildId = ?', [target.id, interaction.guild.id]);
await db.query(
`INSERT INTO logs (userId, userTag, modId, modTag, action, reason, type, guildId, timestamp)
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,
[
target.id,
target.tag,
interaction.user.id,
interaction.user.tag,
'Reset de niveau',
`Niveau ${oldLevel} (${oldXP} XP) réinitialisé. Raison: ${reason}`,
'Permanent',
interaction.guild.id,
Date.now()
]
);
const logEmbed = new EmbedBuilder()
.setAuthor({ name: interaction.user.displayName, iconURL: interaction.user.displayAvatarURL({ dynamic: true }) })
.setTitle(`${emojis.warning} Reset de Niveau`)
.setColor(colors.warning)
.setDescription(`Le niveau de ${target.toString()} a été réinitialisé.`)
.setThumbnail(target.displayAvatarURL({ dynamic: true }))
.addFields(
{ name: '👤 Utilisateur', value: `${target.toString()}\n\`${target.tag}\``, inline: true },
{ name: '👮 Modérateur', value: `${interaction.user.toString()}\n\`${interaction.user.tag}\``, inline: true },
{ name: '📊 Niveau avant', value: `\`${oldLevel}\``, inline: true },
{ name: '📈 XP avant', value: `\`${oldXP.toLocaleString()} XP\``, inline: true },
{ name: '💬 Messages', value: `\`${oldMessages.toLocaleString()}\``, inline: true },
{ name: '🎤 Temps vocal', value: `\`${Math.floor(oldVoiceTime / 60)}h\``, inline: true },
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: `ID: ${target.id}${interaction.guild.name}` })
.setTimestamp();
await sendLog(interaction.guild, { embeds: [logEmbed] });
const embed = new EmbedBuilder()
.setAuthor({ name: target.displayName, iconURL: target.displayAvatarURL({ dynamic: true }) })
.setTitle(`${emojis.success} Niveau Réinitialisé`)
.setColor(colors.success)
.setDescription(`Le niveau de ${target.toString()} a été complètement réinitialisé.`)
.setThumbnail(target.displayAvatarURL({ dynamic: true }))
.addFields(
{ name: '👤 Utilisateur', value: `\`${target.tag}\``, inline: true },
{ name: '📊 Niveau avant', value: `\`${oldLevel}\``, inline: true },
{ name: '📈 XP avant', value: `\`${oldXP.toLocaleString()} XP\``, inline: true },
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: 'Toutes les données XP ont été supprimées de la base de données.' })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} catch (err) {
console.error('Erreur resetlevel:', err);
await interaction.editReply({ content: `❌ Erreur lors du reset de niveau: ${err.message}` });
}
},
};

View File

@@ -1,154 +1,140 @@
const { SlashCommandBuilder, EmbedBuilder, PermissionFlagsBits } = require('discord.js');
const { SlashCommandBuilder, EmbedBuilder, PermissionFlagsBits, MessageFlags } = require('discord.js');
const db = require('../../functions/database/db.js');
const { isChannelExcluded } = require('../../functions/xp/xp.js');
const { colors } = require('../../utils/constants');
const { colors, emojis } = require('../../utils/constants');
module.exports = {
category: 'xp',
data: new SlashCommandBuilder()
.setName('xpexclude')
.setDescription('Gérer les salons exclus de l\'XP (Admin uniquement)')
.addSubcommand(subcommand =>
subcommand
.setName('add')
.setDescription('Exclure un salon de l\'XP')
.addChannelOption(option =>
option.setName('channel')
.setDescription('Le salon à exclure')
.setRequired(true))
.addStringOption(option =>
option.setName('reason')
.setDescription('Raison de l\'exclusion')
.setRequired(false)))
.addSubcommand(subcommand =>
subcommand
.setName('remove')
.setDescription('Retirer un salon de l\'exclusion')
.addChannelOption(option =>
option.setName('channel')
.setDescription('Le salon à retirer')
.setRequired(true)))
.addSubcommand(subcommand =>
subcommand
.setName('list')
.setDescription('Liste des salons exclus')),
async execute(interaction) {
await interaction.deferReply({ ephemeral: true });
// Vérifier les permissions (Admin uniquement)
if (!interaction.member.permissions.has(PermissionFlagsBits.Administrator)) {
return interaction.editReply({
content: '❌ Tu n\'as pas la permission d\'utiliser cette commande. (Administrateur requis)'
});
}
const subcommand = interaction.options.getSubcommand();
try {
if (subcommand === 'add') {
const channel = interaction.options.getChannel('channel');
const reason = interaction.options.getString('reason') || 'Aucune raison fournie';
if (!channel.isTextBased()) {
return interaction.editReply({
content: '❌ Le salon doit être un canal texte.'
});
}
// Vérifier si déjà exclus
const excluded = await isChannelExcluded(channel.id, interaction.guild.id);
if (excluded) {
return interaction.editReply({
content: `❌ Le salon ${channel.toString()} est déjà exclu de l'XP.`
});
}
// Ajouter l'exclusion
await db.query(
'INSERT INTO xp_excluded_channels (channelId, guildId, reason, addedBy, addedAt) VALUES (?, ?, ?, ?, ?)',
[channel.id, interaction.guild.id, reason, interaction.user.id, Date.now()]
);
const { emojis } = require('../../utils/constants');
const embed = new EmbedBuilder()
.setTitle(`${emojis.success} Salon Exclu`)
.setColor(colors.success)
.setDescription(`Le salon ${channel.toString()} a été exclu de l'XP.`)
.addFields(
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: `${interaction.guild.name} • Système XP` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} else if (subcommand === 'remove') {
const channel = interaction.options.getChannel('channel');
// Vérifier si exclu
const excluded = await isChannelExcluded(channel.id, interaction.guild.id);
if (!excluded) {
return interaction.editReply({
content: `❌ Le salon ${channel.toString()} n'est pas exclu de l'XP.`
});
}
// Retirer l'exclusion
await db.query(
'DELETE FROM xp_excluded_channels WHERE channelId = ? AND guildId = ?',
[channel.id, interaction.guild.id]
);
const { emojis } = require('../../utils/constants');
const embed = new EmbedBuilder()
.setTitle(`${emojis.success} Exclusion Retirée`)
.setColor(colors.success)
.setDescription(`Le salon ${channel.toString()} n'est plus exclu de l'XP.`)
.setFooter({ text: `${interaction.guild.name} • Système XP` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} else if (subcommand === 'list') {
const [excludedChannels] = await db.query(
'SELECT * FROM xp_excluded_channels WHERE guildId = ?',
[interaction.guild.id]
);
if (excludedChannels.length === 0) {
return interaction.editReply({
content: '✅ Aucun salon n\'est exclu de l\'XP.'
});
}
const description = excludedChannels.map(ch => {
const channel = interaction.guild.channels.cache.get(ch.channelId);
return `${channel ? channel.toString() : `ID: ${ch.channelId}`} - ${ch.reason || 'Aucune raison'}`;
}).join('\n');
const { emojis } = require('../../utils/constants');
const embed = new EmbedBuilder()
.setAuthor({
name: `${interaction.guild.name}`,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTitle('📋 Salons Exclus de l\'XP')
.setColor(colors.xp)
.setDescription(description || 'Aucun salon exclu')
.setFooter({ text: `${excludedChannels.length} salon(s) exclu(s)` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
}
} catch (err) {
console.error('Erreur lors de la gestion des exclusions XP:', err);
await interaction.editReply({
content: `❌ Erreur: ${err.message}`
});
}
},
};
category: 'xp',
data: new SlashCommandBuilder()
.setName('xpexclude')
.setDescription('Gérer les salons exclus de l\'XP (Admin uniquement)')
.addSubcommand(subcommand =>
subcommand
.setName('add')
.setDescription('Exclure un salon de l\'XP')
.addChannelOption(option =>
option.setName('channel')
.setDescription('Le salon à exclure')
.setRequired(true))
.addStringOption(option =>
option.setName('reason')
.setDescription('Raison de l\'exclusion')
.setRequired(false)))
.addSubcommand(subcommand =>
subcommand
.setName('remove')
.setDescription('Retirer un salon de l\'exclusion')
.addChannelOption(option =>
option.setName('channel')
.setDescription('Le salon à retirer')
.setRequired(true)))
.addSubcommand(subcommand =>
subcommand
.setName('list')
.setDescription('Liste des salons exclus')),
async execute(interaction) {
await interaction.deferReply({ flags: MessageFlags.Ephemeral });
if (!interaction.member.permissions.has(PermissionFlagsBits.Administrator)) {
return interaction.editReply({
content: '❌ Tu n\'as pas la permission d\'utiliser cette commande. (Administrateur requis)'
});
}
const subcommand = interaction.options.getSubcommand();
try {
if (subcommand === 'add') {
const channel = interaction.options.getChannel('channel');
const reason = interaction.options.getString('reason') || 'Aucune raison fournie';
if (!channel.isTextBased()) {
return interaction.editReply({
content: '❌ Le salon doit être un canal texte.'
});
}
const excluded = await isChannelExcluded(channel.id, interaction.guild.id);
if (excluded) {
return interaction.editReply({
content: `❌ Le salon ${channel.toString()} est déjà exclu de l'XP.`
});
}
await db.query(
'INSERT INTO xp_excluded_channels (channelId, guildId, reason, addedBy, addedAt) VALUES (?, ?, ?, ?, ?)',
[channel.id, interaction.guild.id, reason, interaction.user.id, Date.now()]
);
const embed = new EmbedBuilder()
.setTitle(`${emojis.success} Salon Exclu`)
.setColor(colors.success)
.setDescription(`Le salon ${channel.toString()} a été exclu de l'XP.`)
.addFields(
{ name: '📝 Raison', value: reason, inline: false }
)
.setFooter({ text: `${interaction.guild.name} • Système XP` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} else if (subcommand === 'remove') {
const channel = interaction.options.getChannel('channel');
const excluded = await isChannelExcluded(channel.id, interaction.guild.id);
if (!excluded) {
return interaction.editReply({
content: `❌ Le salon ${channel.toString()} n'est pas exclu de l'XP.`
});
}
await db.query(
'DELETE FROM xp_excluded_channels WHERE channelId = ? AND guildId = ?',
[channel.id, interaction.guild.id]
);
const embed = new EmbedBuilder()
.setTitle(`${emojis.success} Exclusion Retirée`)
.setColor(colors.success)
.setDescription(`Le salon ${channel.toString()} n'est plus exclu de l'XP.`)
.setFooter({ text: `${interaction.guild.name} • Système XP` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
} else if (subcommand === 'list') {
const [excludedChannels] = await db.query(
'SELECT * FROM xp_excluded_channels WHERE guildId = ?',
[interaction.guild.id]
);
if (excludedChannels.length === 0) {
return interaction.editReply({
content: '✅ Aucun salon n\'est exclu de l\'XP.'
});
}
const description = excludedChannels.map(ch => {
const channel = interaction.guild.channels.cache.get(ch.channelId);
return `${channel ? channel.toString() : `ID: ${ch.channelId}`} - ${ch.reason || 'Aucune raison'}`;
}).join('\n');
const embed = new EmbedBuilder()
.setAuthor({
name: interaction.guild.name,
iconURL: interaction.guild.iconURL({ dynamic: true }) || undefined
})
.setTitle('📋 Salons Exclus de l\'XP')
.setColor(colors.xp)
.setDescription(description || 'Aucun salon exclu')
.setFooter({ text: `${excludedChannels.length} salon(s) exclu(s)` })
.setTimestamp();
await interaction.editReply({ embeds: [embed] });
}
} catch (err) {
console.error('Erreur xpexclude:', err);
await interaction.editReply({ content: `❌ Erreur: ${err.message}` });
}
},
};