mirror of
https://github.com/aydenjahola/discord-multipurpose-bot.git
synced 2024-11-21 16:25:55 +00:00
economy: add sell command, fix logic of the economy system
This commit is contained in:
parent
e4e6135a21
commit
81544d3f65
5 changed files with 332 additions and 21 deletions
|
@ -4,32 +4,28 @@ const UserEconomy = require("../../models/UserEconomy");
|
||||||
module.exports = {
|
module.exports = {
|
||||||
data: new SlashCommandBuilder()
|
data: new SlashCommandBuilder()
|
||||||
.setName("balance")
|
.setName("balance")
|
||||||
.setDescription("Check your balance and claim a bonus for checking daily!"),
|
.setDescription("Check your balance."),
|
||||||
|
|
||||||
async execute(interaction) {
|
async execute(interaction) {
|
||||||
const { user, guild } = interaction;
|
const { user, guild } = interaction;
|
||||||
|
|
||||||
let userEconomy = await UserEconomy.findOne({
|
let userEconomy = await UserEconomy.findOne({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
guildId: guild.id,
|
guildId: guild.id,
|
||||||
});
|
});
|
||||||
const checkInBonus = 10;
|
|
||||||
|
|
||||||
if (!userEconomy) {
|
if (!userEconomy) {
|
||||||
userEconomy = await UserEconomy.create({
|
userEconomy = await UserEconomy.create({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
guildId: guild.id,
|
guildId: guild.id,
|
||||||
|
balance: 0,
|
||||||
});
|
});
|
||||||
} else {
|
|
||||||
userEconomy.balance += checkInBonus;
|
|
||||||
await userEconomy.save();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const embed = new EmbedBuilder()
|
const embed = new EmbedBuilder()
|
||||||
.setColor("#0099ff")
|
.setColor("#0099ff")
|
||||||
.setTitle(`${user.username}'s Balance`)
|
.setTitle(`${user.username}'s Balance`)
|
||||||
.setDescription(
|
.setDescription(`Your balance: **${userEconomy.balance}** coins.`)
|
||||||
`Your balance: **${userEconomy.balance}** coins. (Check-in Bonus: +${checkInBonus} coins)`
|
|
||||||
)
|
|
||||||
.setFooter({
|
.setFooter({
|
||||||
text: `Requested by ${user.username}`,
|
text: `Requested by ${user.username}`,
|
||||||
iconURL: user.displayAvatarURL(),
|
iconURL: user.displayAvatarURL(),
|
||||||
|
|
|
@ -22,7 +22,8 @@ module.exports = {
|
||||||
userEconomy = await UserEconomy.create({
|
userEconomy = await UserEconomy.create({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
guildId: guild.id,
|
guildId: guild.id,
|
||||||
streak: 0, // Initialize streak
|
streak: 0,
|
||||||
|
balance: 0,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,17 +57,14 @@ module.exports = {
|
||||||
}
|
}
|
||||||
|
|
||||||
userEconomy.streak += 1;
|
userEconomy.streak += 1;
|
||||||
|
const reward = dailyBaseReward + userEconomy.streak * streakBonus;
|
||||||
let reward = dailyBaseReward + userEconomy.streak * streakBonus;
|
|
||||||
userEconomy.lastDaily = now;
|
userEconomy.lastDaily = now;
|
||||||
userEconomy.balance += reward;
|
userEconomy.balance += reward;
|
||||||
|
|
||||||
const rareItemEarned = Math.random() < rareItemChance;
|
const rareItemEarned = Math.random() < rareItemChance;
|
||||||
let rareItemMessage = "";
|
const rareItemMessage = rareItemEarned
|
||||||
|
? "\nYou also found a **rare item** in your reward!"
|
||||||
if (rareItemEarned) {
|
: "";
|
||||||
rareItemMessage = "\nYou also found a **rare item** in your reward!";
|
|
||||||
}
|
|
||||||
|
|
||||||
await userEconomy.save();
|
await userEconomy.save();
|
||||||
|
|
||||||
|
|
|
@ -32,8 +32,17 @@ module.exports = {
|
||||||
const itemDetails = await Promise.all(
|
const itemDetails = await Promise.all(
|
||||||
inventory.map(async (item) => {
|
inventory.map(async (item) => {
|
||||||
const shopItem = await ShopItem.findOne({ itemId: item.itemId });
|
const shopItem = await ShopItem.findOne({ itemId: item.itemId });
|
||||||
|
|
||||||
|
if (!shopItem) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
if (item.quantity > 0) {
|
if (item.quantity > 0) {
|
||||||
return `${shopItem.name} (x${item.quantity}) - **Rarity**: ${shopItem.rarity}`;
|
return `${shopItem.name} (x${item.quantity}) - **Rarity**: ${
|
||||||
|
shopItem.rarity
|
||||||
|
} - **Type**: ${shopItem.type} - **Category**: ${
|
||||||
|
shopItem.category || "N/A"
|
||||||
|
}`;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
})
|
})
|
||||||
|
@ -44,7 +53,9 @@ module.exports = {
|
||||||
const inventoryEmbed = new EmbedBuilder()
|
const inventoryEmbed = new EmbedBuilder()
|
||||||
.setColor("#00ff00")
|
.setColor("#00ff00")
|
||||||
.setTitle(`${user.username}'s Inventory`)
|
.setTitle(`${user.username}'s Inventory`)
|
||||||
.setDescription(filteredItemDetails.join("\n"))
|
.setDescription(
|
||||||
|
filteredItemDetails.join("\n") || "No items in inventory."
|
||||||
|
)
|
||||||
.setTimestamp()
|
.setTimestamp()
|
||||||
.setFooter({
|
.setFooter({
|
||||||
text: `Requested by ${user.username}`,
|
text: `Requested by ${user.username}`,
|
||||||
|
|
208
commands/economy/sell.js
Normal file
208
commands/economy/sell.js
Normal file
|
@ -0,0 +1,208 @@
|
||||||
|
const {
|
||||||
|
SlashCommandBuilder,
|
||||||
|
EmbedBuilder,
|
||||||
|
ActionRowBuilder,
|
||||||
|
StringSelectMenuBuilder,
|
||||||
|
} = require("discord.js");
|
||||||
|
const UserInventory = require("../../models/UserInventory");
|
||||||
|
const ShopItem = require("../../models/ShopItem");
|
||||||
|
const UserEconomy = require("../../models/UserEconomy");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
data: new SlashCommandBuilder()
|
||||||
|
.setName("sell")
|
||||||
|
.setDescription("Sell an item from your inventory."),
|
||||||
|
|
||||||
|
async execute(interaction) {
|
||||||
|
const { user, guild } = interaction;
|
||||||
|
|
||||||
|
const userInventory = await UserInventory.find({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (userInventory.length === 0) {
|
||||||
|
const errorEmbed = new EmbedBuilder()
|
||||||
|
.setColor("#ff0000")
|
||||||
|
.setTitle("🛑 Inventory Empty")
|
||||||
|
.setDescription("You have no items in your inventory to sell.")
|
||||||
|
.setTimestamp()
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
});
|
||||||
|
|
||||||
|
await interaction.reply({ embeds: [errorEmbed], ephemeral: false });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectMenu = new StringSelectMenuBuilder()
|
||||||
|
.setCustomId("sell_item")
|
||||||
|
.setPlaceholder("Select an item to sell")
|
||||||
|
.addOptions(
|
||||||
|
userInventory.map((item) => ({
|
||||||
|
label: item.itemId,
|
||||||
|
value: item.itemId,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
const row = new ActionRowBuilder().addComponents(selectMenu);
|
||||||
|
|
||||||
|
const promptEmbed = new EmbedBuilder()
|
||||||
|
.setColor("#00ffff")
|
||||||
|
.setTitle("📦 Select Item to Sell")
|
||||||
|
.setDescription(
|
||||||
|
"Choose an item from your inventory to sell. The selling price is 50% less than the original item price."
|
||||||
|
)
|
||||||
|
.setTimestamp()
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
});
|
||||||
|
|
||||||
|
await interaction.reply({ embeds: [promptEmbed], components: [row] });
|
||||||
|
|
||||||
|
const filter = (i) => {
|
||||||
|
return i.user.id === user.id;
|
||||||
|
};
|
||||||
|
|
||||||
|
const collector = interaction.channel.createMessageComponentCollector({
|
||||||
|
filter,
|
||||||
|
time: 15000,
|
||||||
|
});
|
||||||
|
|
||||||
|
collector.on("collect", async (i) => {
|
||||||
|
if (i.customId === "sell_item") {
|
||||||
|
const selectedItemId = i.values[0];
|
||||||
|
const selectedItem = userInventory.find(
|
||||||
|
(item) => item.itemId === selectedItemId
|
||||||
|
);
|
||||||
|
|
||||||
|
const shopItem = await ShopItem.findOne({
|
||||||
|
itemId: selectedItemId,
|
||||||
|
guildId: guild.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!shopItem) {
|
||||||
|
await i.reply({
|
||||||
|
content: "Item not found in the shop.",
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const sellingPrice = Math.floor(shopItem.price / 2);
|
||||||
|
|
||||||
|
const quantityPrompt = new EmbedBuilder()
|
||||||
|
.setColor("#ffff00")
|
||||||
|
.setTitle(`💰 Selling ${shopItem.name}`)
|
||||||
|
.setDescription(
|
||||||
|
`How many of **${shopItem.name}** would you like to sell?`
|
||||||
|
)
|
||||||
|
.addFields({
|
||||||
|
name: "Sell Price",
|
||||||
|
value: `Each item sells for **${sellingPrice}** coins.`,
|
||||||
|
inline: true,
|
||||||
|
})
|
||||||
|
.setTimestamp()
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
});
|
||||||
|
|
||||||
|
await i.reply({ embeds: [quantityPrompt], ephemeral: false });
|
||||||
|
|
||||||
|
const quantityFilter = (response) => {
|
||||||
|
return (
|
||||||
|
response.author.id === user.id &&
|
||||||
|
!isNaN(response.content) &&
|
||||||
|
response.content > 0
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const quantityCollector = interaction.channel.createMessageCollector({
|
||||||
|
filter: quantityFilter,
|
||||||
|
time: 15000,
|
||||||
|
});
|
||||||
|
|
||||||
|
quantityCollector.on("collect", async (response) => {
|
||||||
|
const quantityToSell = parseInt(response.content);
|
||||||
|
|
||||||
|
if (quantityToSell > selectedItem.quantity) {
|
||||||
|
await response.reply({
|
||||||
|
content: `You do not have enough **${shopItem.name}** to sell!`,
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let userEconomy = await UserEconomy.findOne({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!userEconomy) {
|
||||||
|
userEconomy = await UserEconomy.create({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
balance: 0,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
userEconomy.balance += sellingPrice * quantityToSell;
|
||||||
|
await userEconomy.save();
|
||||||
|
|
||||||
|
selectedItem.quantity -= quantityToSell;
|
||||||
|
if (selectedItem.quantity === 0) {
|
||||||
|
await UserInventory.deleteOne({
|
||||||
|
userId: user.id,
|
||||||
|
itemId: selectedItemId,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
await selectedItem.save();
|
||||||
|
}
|
||||||
|
|
||||||
|
const successEmbed = new EmbedBuilder()
|
||||||
|
.setColor("#00ff00")
|
||||||
|
.setTitle("🎉 Item Sold!")
|
||||||
|
.setDescription(
|
||||||
|
`You sold **${quantityToSell}** of **${shopItem.name}** for **${
|
||||||
|
sellingPrice * quantityToSell
|
||||||
|
}** coins!`
|
||||||
|
)
|
||||||
|
.addFields({
|
||||||
|
name: "New Balance",
|
||||||
|
value: `${userEconomy.balance} coins`,
|
||||||
|
inline: true,
|
||||||
|
})
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
})
|
||||||
|
.setTimestamp();
|
||||||
|
|
||||||
|
await response.reply({ embeds: [successEmbed], ephemeral: false });
|
||||||
|
quantityCollector.stop();
|
||||||
|
});
|
||||||
|
|
||||||
|
quantityCollector.on("end", async (collected) => {
|
||||||
|
if (collected.size === 0) {
|
||||||
|
await i.followUp({
|
||||||
|
content: "You did not respond in time to sell the item.",
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
collector.on("end", async (collected) => {
|
||||||
|
if (collected.size === 0) {
|
||||||
|
await interaction.followUp({
|
||||||
|
content: "You did not select an item in time.",
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
|
@ -1,20 +1,30 @@
|
||||||
const { SlashCommandBuilder, EmbedBuilder } = require("discord.js");
|
const { SlashCommandBuilder, EmbedBuilder } = require("discord.js");
|
||||||
const ShopItem = require("../../models/ShopItem");
|
const ShopItem = require("../../models/ShopItem");
|
||||||
|
const UserEconomy = require("../../models/UserEconomy");
|
||||||
|
const UserInventory = require("../../models/UserInventory");
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
data: new SlashCommandBuilder()
|
data: new SlashCommandBuilder()
|
||||||
.setName("shop")
|
.setName("shop")
|
||||||
.setDescription("Browse the shop for items with rarity and discounts!"),
|
.setDescription("Browse the shop for items with rarity and discounts!")
|
||||||
|
.addStringOption((option) =>
|
||||||
|
option
|
||||||
|
.setName("item")
|
||||||
|
.setDescription("The item you want to buy (use item name)")
|
||||||
|
.setRequired(false)
|
||||||
|
),
|
||||||
|
|
||||||
async execute(interaction) {
|
async execute(interaction) {
|
||||||
const { user, guild } = interaction;
|
const { user, guild } = interaction;
|
||||||
const discountChance = 0.3;
|
const discountChance = 0.3;
|
||||||
|
const itemName = interaction.options.getString("item");
|
||||||
|
|
||||||
const items = await ShopItem.find({ guildId: guild.id });
|
const items = await ShopItem.find({ guildId: guild.id });
|
||||||
|
|
||||||
if (items.length === 0) {
|
if (items.length === 0) {
|
||||||
const emptyEmbed = new EmbedBuilder()
|
const emptyEmbed = new EmbedBuilder()
|
||||||
.setColor("#ff0000")
|
.setColor("#ff0000")
|
||||||
.setTitle("Shop")
|
.setTitle("🛒 Shop Items")
|
||||||
.setDescription("No items in the shop currently.")
|
.setDescription("No items in the shop currently.")
|
||||||
.setTimestamp()
|
.setTimestamp()
|
||||||
.setFooter({
|
.setFooter({
|
||||||
|
@ -26,6 +36,94 @@ module.exports = {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (itemName) {
|
||||||
|
const item = items.find(
|
||||||
|
(item) => item.name.toLowerCase() === itemName.toLowerCase()
|
||||||
|
);
|
||||||
|
if (!item) {
|
||||||
|
await interaction.reply({
|
||||||
|
content: "Item not found in the shop!",
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let userEconomy = await UserEconomy.findOne({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
});
|
||||||
|
if (!userEconomy) {
|
||||||
|
userEconomy = await UserEconomy.create({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const discount = Math.random() < discountChance ? 0.8 : 1;
|
||||||
|
const price = Math.floor(item.price * discount);
|
||||||
|
|
||||||
|
if (userEconomy.balance < price) {
|
||||||
|
const insufficientFundsEmbed = new EmbedBuilder()
|
||||||
|
.setColor("#ff0000")
|
||||||
|
.setTitle("💸 Insufficient Funds")
|
||||||
|
.setDescription(
|
||||||
|
`You don't have enough coins to buy **${item.name}**!`
|
||||||
|
)
|
||||||
|
.addFields(
|
||||||
|
{
|
||||||
|
name: "Your Balance",
|
||||||
|
value: `${userEconomy.balance} coins`,
|
||||||
|
inline: true,
|
||||||
|
},
|
||||||
|
{ name: "Item Price", value: `${price} coins`, inline: true }
|
||||||
|
)
|
||||||
|
.setTimestamp()
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
});
|
||||||
|
|
||||||
|
await interaction.reply({
|
||||||
|
embeds: [insufficientFundsEmbed],
|
||||||
|
ephemeral: false,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
userEconomy.balance -= price;
|
||||||
|
await userEconomy.save();
|
||||||
|
|
||||||
|
const userInventory = await UserInventory.findOne({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
itemId: item.itemId,
|
||||||
|
});
|
||||||
|
if (userInventory) {
|
||||||
|
userInventory.quantity += 1;
|
||||||
|
await userInventory.save();
|
||||||
|
} else {
|
||||||
|
await UserInventory.create({
|
||||||
|
userId: user.id,
|
||||||
|
guildId: guild.id,
|
||||||
|
itemId: item.itemId,
|
||||||
|
quantity: 1,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const successEmbed = new EmbedBuilder()
|
||||||
|
.setColor("#00ff00")
|
||||||
|
.setTitle("🎉 Purchase Successful")
|
||||||
|
.setDescription(`You bought **${item.name}** for **${price}** coins!`)
|
||||||
|
.setTimestamp()
|
||||||
|
.setFooter({
|
||||||
|
text: `Requested by ${user.username}`,
|
||||||
|
iconURL: user.displayAvatarURL(),
|
||||||
|
});
|
||||||
|
|
||||||
|
await interaction.reply({ embeds: [successEmbed], ephemeral: false });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const shopItemsDetails = items.map((item) => {
|
const shopItemsDetails = items.map((item) => {
|
||||||
const discount = Math.random() < discountChance ? 0.8 : 1;
|
const discount = Math.random() < discountChance ? 0.8 : 1;
|
||||||
const price = Math.floor(item.price * discount);
|
const price = Math.floor(item.price * discount);
|
||||||
|
@ -38,7 +136,7 @@ module.exports = {
|
||||||
|
|
||||||
const shopEmbed = new EmbedBuilder()
|
const shopEmbed = new EmbedBuilder()
|
||||||
.setColor("#00ff00")
|
.setColor("#00ff00")
|
||||||
.setTitle("Shop")
|
.setTitle("🛒 Shop Items")
|
||||||
.setDescription(shopItemsDetails.join("\n"))
|
.setDescription(shopItemsDetails.join("\n"))
|
||||||
.setTimestamp()
|
.setTimestamp()
|
||||||
.setFooter({
|
.setFooter({
|
||||||
|
|
Loading…
Reference in a new issue