/[sudobot]/branches/6.x/src/commands/automation/BallotVoteListCommand.ts
ViewVC logotype

Annotation of /branches/6.x/src/commands/automation/BallotVoteListCommand.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (hide annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 5853 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 rakinar2 577 /**
2     * This file is part of SudoBot.
3     *
4     * Copyright (C) 2021-2023 OSN Developers.
5     *
6     * SudoBot is free software; you can redistribute it and/or modify it
7     * under the terms of the GNU Affero General Public License as published by
8     * the Free Software Foundation, either version 3 of the License, or
9     * (at your option) any later version.
10     *
11     * SudoBot is distributed in the hope that it will be useful, but
12     * WITHOUT ANY WARRANTY; without even the implied warranty of
13     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14     * GNU Affero General Public License for more details.
15     *
16     * You should have received a copy of the GNU Affero General Public License
17     * along with SudoBot. If not, see <https://www.gnu.org/licenses/>.
18     */
19    
20     import { ActionRowBuilder, ButtonBuilder, ButtonStyle, EmbedBuilder } from "discord.js";
21     import Command, { BasicCommandContext, CommandMessage, CommandReturn } from "../../core/Command";
22     import Pagination from "../../utils/Pagination";
23     import { safeUserFetch } from "../../utils/fetch";
24    
25     export default class BallotVoteListCommand extends Command {
26     public readonly name = "ballot__votelist";
27     public readonly permissions = [];
28     public readonly description = "Shows a list of each vote in a poll/ballot.";
29    
30     async execute(message: CommandMessage, context: BasicCommandContext): Promise<CommandReturn> {
31     if (context.isLegacy && context.args[0] === undefined) {
32     await this.error(message, "Please provide the ballot ID!");
33     return;
34     }
35    
36     await this.deferIfInteraction(message);
37    
38     type Mode = "all" | "upvotes" | "downvotes";
39     const id = context.isLegacy ? parseInt(context.args[0]) : context.options.getInteger("id", true);
40     const mode = <Mode>((context.isLegacy ? null : context.options.getString("mode")) ?? "all");
41    
42     if (isNaN(id)) {
43     await this.error(message, "Invalid ballot ID given! Ballot IDs must be numeric values.");
44     return;
45     }
46    
47     const ballot = await this.client.ballotManager.get({
48     id,
49     guildId: message.guildId!
50     });
51    
52     if (!ballot) {
53     await this.error(message, "No such ballot exists with that ID!");
54     return;
55     }
56    
57     const user = ballot.anonymous ? null : await safeUserFetch(this.client, ballot.userId);
58     const url = `https://discord.com/channels/${encodeURIComponent(ballot.guildId)}/${encodeURIComponent(
59     ballot.channelId
60     )}/${encodeURIComponent(ballot.messageId)}`;
61    
62     const data = [
63     ...(mode === "all" || mode === "upvotes"
64     ? ballot.upvotes.map(userId => ({
65     userId,
66     type: "upvote" as const
67     }))
68     : []),
69     ...(mode === "all" || mode === "downvotes"
70     ? ballot.downvotes.map(userId => ({
71     userId,
72     type: "downvote" as const
73     }))
74     : [])
75     ];
76    
77     const pagination = new Pagination(data, {
78     channelId: message.channelId!,
79     client: this.client,
80     guildId: message.guildId!,
81     limit: 10,
82     userId: message.member?.user.id,
83     timeout: 120_000,
84     messageOptions: {
85     files: ballot.files.map(url => ({ attachment: url })),
86     embeds: [
87     {
88     author: {
89     name: ballot.anonymous ? "Staff" : user?.username ?? "Unknown",
90     icon_url: ballot.anonymous ? message.guild!.iconURL() ?? undefined : user?.displayAvatarURL(),
91     url
92     },
93     description: ballot.content,
94     color: 0x007bff,
95     fields: [
96     {
97     name: "Total Votes",
98     value: `⚪ **${ballot.upvotes.length - ballot.downvotes.length}**`,
99     inline: true
100     },
101     {
102     name: "Upvotes",
103     value: `${this.emoji("ArrowTop")} ${ballot.upvotes.length}`,
104     inline: true
105     },
106     {
107     name: "Downvotes",
108     value: `${this.emoji("ArrowDown")} ${ballot.downvotes.length}`,
109     inline: true
110     }
111     ]
112     }
113     ],
114     components: [
115     new ActionRowBuilder<ButtonBuilder>().addComponents(
116     new ButtonBuilder().setURL(url).setStyle(ButtonStyle.Link).setLabel("Go to ballot message")
117     )
118     ]
119     },
120     embedBuilder({ currentPage, data, maxPages }) {
121     let description = "";
122    
123     for (const vote of data) {
124     description += ` * **${vote.type[0].toUpperCase()}${vote.type.substring(1)}** - <@${vote.userId}> [${
125     vote.userId
126     }]\n`;
127     }
128    
129     description = description === "" ? "*No vote was recorded*" : description;
130    
131     return new EmbedBuilder({
132     color: 0x007bff,
133     description,
134     footer: {
135     text: `Page ${currentPage} of ${maxPages}`
136     }
137     }).setTimestamp();
138     }
139     });
140    
141     const reply = await this.deferredReply(message, await pagination.getMessageOptions(1));
142     await pagination.start(reply);
143     }
144     }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26