Update
This commit is contained in:
@@ -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.'
|
||||
});
|
||||
}
|
||||
},
|
||||
};
|
||||
@@ -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] });
|
||||
},
|
||||
};
|
||||
@@ -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] });
|
||||
},
|
||||
};
|
||||
@@ -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}` });
|
||||
}
|
||||
},
|
||||
};
|
||||
@@ -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}` });
|
||||
}
|
||||
},
|
||||
};
|
||||
Reference in New Issue
Block a user