I’m trying to make a horse race command for my discord bot (TypeScript).
The code itself works fine, but I have to update an embed which contains the race and the participants. The thing is that for it to properly update, I have to set its description every time that collector.on("collect")
fires. I want to ask if there’s a better, more efficient and cleaner way to update it. Thank you!
code:
JavaScript
x
130
130
1
const bet = interaction.options.get("bet").value;
2
3
class Horse {
4
name: string;
5
owner: User;
6
speed: number;
7
position: Array<string>;
8
constructor(name: string) {
9
this.name = name;
10
this.owner = null;
11
this.speed = 0;
12
this.position = ["🐴"];
13
}
14
}
15
16
const horses = [
17
new Horse(aimless.pick(names, { remove: true })),
18
new Horse(aimless.pick(names, { remove: true })),
19
new Horse(aimless.pick(names, { remove: true })),
20
new Horse(aimless.pick(names, { remove: false })),
21
];
22
const hasJoined: Array<Horse["owner"]> = [];
23
24
const row = new MessageActionRow();
25
for (const horse of horses) {
26
row.addComponents(
27
new MessageButton()
28
.setCustomId(horse.name)
29
.setLabel(horse.name)
30
.setStyle("SECONDARY")
31
);
32
}
33
const embed = new MessageEmbed()
34
.setTitle("Place your bets!")
35
.setDescription(
36
`**${horses[0].name} - ${
37
horses[0].owner !== null ? horses[0].owner.username : "Nobody"
38
}
39
${horses[0].position}
40
41
${horses[1].name} - ${
42
horses[1].owner !== null ? horses[1].owner.username : "Nobody"
43
}
44
${horses[1].position}
45
46
${horses[2].name} - ${
47
horses[2].owner !== null ? horses[2].owner.username : "Nobody"
48
}
49
${horses[2].position}
50
51
${horses[3].name} - ${
52
horses[3].owner !== null ? horses[3].owner.username : "Nobody"
53
}
54
${horses[3].position}**`
55
);
56
await interaction.editReply({
57
embeds: ,
58
components: [row],
59
});
60
const filter = async (i: MessageComponentInteraction) => {
61
let profile: any;
62
try {
63
profile = await profileModel.findOne({ userID: i.user.id });
64
if (!profile) {
65
await profileModel.create({
66
userID: i.user.id,
67
serverID: i.guild?.id,
68
username: i.user.username,
69
bananas: 100,
70
deposit: 0,
71
});
72
profile.save();
73
}
74
} catch (e) {
75
await i.editReply("Something went wrong! :( Please retry.");
76
} finally {
77
if (hasJoined.includes(i.user)) {
78
return false;
79
}
80
if (profile.bananas < bet) {
81
interaction.editReply(`${i.user} you don't have enough bananas!`);
82
}
83
return profile.bananas >= bet;
84
}
85
};
86
const collector = interaction.channel.createMessageComponentCollector({
87
filter,
88
time: 60000,
89
});
90
collector.on("collect", async (int) => {
91
await int.deferUpdate();
92
for (const btn of row.components) {
93
if (btn.customId === (int.component as MessageButton).customId) {
94
(btn as MessageButton).setDisabled(true).setStyle("SUCCESS");
95
hasJoined.push(int.user);
96
horses.find((h) => h.name === btn.customId).owner = int.user;
97
console.log(horses);
98
}
99
}
100
101
embed.setDescription(
102
`**${horses[0].name} - ${
103
horses[0].owner !== null ? horses[0].owner.username : "Nobody"
104
}
105
${horses[0].position}
106
107
${horses[1].name} - ${
108
horses[1].owner !== null ? horses[1].owner.username : "Nobody"
109
}
110
${horses[1].position}
111
112
${horses[2].name} - ${
113
horses[2].owner !== null ? horses[2].owner.username : "Nobody"
114
}
115
${horses[2].position}
116
117
${horses[3].name} - ${
118
horses[3].owner !== null ? horses[3].owner.username : "Nobody"
119
}
120
${horses[3].position}**`
121
);
122
123
await int.editReply({
124
embeds: ,
125
components: [row],
126
});
127
});
128
},
129
});`
130
Advertisement
Answer
You could make it into a function:
JavaScript
1
22
22
1
const displayHorses = (horses: Array<Horse>) => {
2
return `**${horses[0].name} - ${
3
horses[0].owner !== null ? horses[0].owner.username : "Nobody"
4
}
5
${horses[0].position}
6
7
${horses[1].name} - ${
8
horses[1].owner !== null ? horses[1].owner.username : "Nobody"
9
}
10
${horses[1].position}
11
12
${horses[2].name} - ${
13
horses[2].owner !== null ? horses[2].owner.username : "Nobody"
14
}
15
${horses[2].position}
16
17
${horses[3].name} - ${
18
horses[3].owner !== null ? horses[3].owner.username : "Nobody"
19
}
20
${horses[3].position}**`;
21
};
22
And then every time:
JavaScript
1
2
1
embed.setDescription(displayHorses(horses));
2
If you want to compact it even further, you could map the horse array.
JavaScript
1
8
1
const displayHorses = (horses: Array<Horse>) => {
2
return horses.map(
3
({ name, owner, position }) =>
4
`**${name}** - ${owner !== null ? owner.username : "Nobody"}
5
${position}`
6
);
7
};
8
Lastly, as a tip, if you’re using 14+, you could compact it even further to:
JavaScript
1
8
1
const displayHorses = (horses: Array<Horse>) => {
2
return horses.map(
3
({ name, owner, position }) =>
4
`**${name}** - ${owner?.username ?? "Nobody"}
5
${position}`
6
);
7
};
8