/[sudobot]/branches/5.x/src/utils/Pagination.ts
ViewVC logotype

Contents of /branches/5.x/src/utils/Pagination.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (show annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 10215 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 /*
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 {
21 ActionRowBuilder,
22 ButtonBuilder,
23 ButtonInteraction,
24 ButtonStyle,
25 ComponentType,
26 EmbedBuilder,
27 InteractionCollector,
28 InteractionReplyOptions,
29 InteractionType,
30 Message,
31 MessageEditOptions,
32 MessageReplyOptions
33 } from "discord.js";
34 import * as uuid from "uuid";
35 import Client from "../core/Client";
36 import { log } from "./logger";
37 import { getEmoji } from "./utils";
38
39 export default class Pagination<T> {
40 protected readonly id: string;
41 protected readonly client: Client<true>;
42 protected maxPage: number = 0;
43 protected currentPage: number = 1;
44 protected currentData: T[] = [];
45
46 constructor(protected readonly data: Array<T> | null = [], protected readonly options: PaginationOptions<T>) {
47 this.id = uuid.v4();
48 this.client = options.client;
49 }
50
51 getOffset(page: number = 1) {
52 return (page - 1) * this.options.limit;
53 }
54
55 async getPaginatedData(page: number = 1) {
56 log(page, this.getOffset(page));
57
58 if (this.options.fetchData)
59 this.currentData = await this.options.fetchData({
60 currentPage: page,
61 limit: this.options.limit,
62 offset: this.getOffset(page)
63 });
64
65 return this.data ? this.data.slice(this.getOffset(page), this.getOffset(page) + this.options.limit) : this.currentData;
66 }
67
68 async getEmbed(page: number = 1): Promise<EmbedBuilder> {
69 const data = await this.getPaginatedData(page);
70
71 return this.options.embedBuilder({
72 data: this.data ? data : this.currentData,
73 currentPage: this.currentPage,
74 maxPages: Math.ceil((this.data?.length ?? this.maxPage) / this.options.limit)
75 });
76 }
77
78 async getMessageOptions(
79 page: number = 1,
80 actionRowOptions: { first: boolean; last: boolean; next: boolean; back: boolean } | undefined = undefined,
81 optionsToMerge: MessageOptions = {}
82 ) {
83 const options = { ...this.options.messageOptions, ...optionsToMerge };
84 const actionRowOptionsDup = actionRowOptions
85 ? { ...actionRowOptions }
86 : { first: true, last: true, next: true, back: true };
87
88 if (this.options.maxData && this.maxPage === 0)
89 this.maxPage = await this.options.maxData({
90 currentPage: page,
91 limit: this.options.limit,
92 offset: this.getOffset(page)
93 });
94
95 log("Max Page", this.maxPage);
96
97 if (actionRowOptionsDup && page <= 1) {
98 actionRowOptionsDup.back = false;
99 actionRowOptionsDup.first = false;
100 }
101
102 if (actionRowOptionsDup && page >= Math.ceil((this.data?.length ?? this.maxPage) / this.options.limit)) {
103 actionRowOptionsDup.last = false;
104 actionRowOptionsDup.next = false;
105 }
106
107 options.embeds ??= [];
108 options.embeds.push(await this.getEmbed(page));
109
110 options.components ??= [];
111 options.components = [this.getActionRow(actionRowOptionsDup), ...options.components];
112
113 return options;
114 }
115
116 getActionRow(
117 { first, last, next, back }: { first: boolean; last: boolean; next: boolean; back: boolean } = {
118 first: true,
119 last: true,
120 next: true,
121 back: true
122 }
123 ) {
124 if (this.options.actionRowBuilder) {
125 return this.options.actionRowBuilder({ first, last, next, back }, this.id);
126 }
127
128 const actionRow = new ActionRowBuilder<ButtonBuilder>();
129
130 actionRow.addComponents(
131 new ButtonBuilder()
132 .setCustomId(`pagination_first_${this.id}`)
133 .setStyle(ButtonStyle.Secondary)
134 .setDisabled(!first)
135 .setEmoji(getEmoji(this.client, "ArrowLeft") ?? "⏮️"),
136 new ButtonBuilder()
137 .setCustomId(`pagination_back_${this.id}`)
138 .setStyle(ButtonStyle.Secondary)
139 .setDisabled(!back)
140 .setEmoji(getEmoji(this.client, "ChevronLeft") ?? "◀️"),
141 new ButtonBuilder()
142 .setCustomId(`pagination_next_${this.id}`)
143 .setStyle(ButtonStyle.Secondary)
144 .setDisabled(!next)
145 .setEmoji(getEmoji(this.client, "ChevronRight") ?? "▶️"),
146 new ButtonBuilder()
147 .setCustomId(`pagination_last_${this.id}`)
148 .setStyle(ButtonStyle.Secondary)
149 .setDisabled(!last)
150 .setEmoji(getEmoji(this.client, "ArrowRight") ?? "⏭️")
151 );
152
153 return actionRow;
154 }
155
156 async start(message: Message) {
157 const collector = new InteractionCollector(this.client, {
158 guild: this.options.guildId,
159 channel: this.options.channelId,
160 interactionType: InteractionType.MessageComponent,
161 componentType: ComponentType.Button,
162 message,
163 time: this.options.timeout ?? 60_000,
164 filter: interaction => {
165 if (interaction.inGuild() && (!this.options.userId || interaction.user.id === this.options.userId)) {
166 return true;
167 }
168
169 if (interaction.isRepliable()) {
170 interaction.reply({
171 content: "That's not under your control or the button controls are expired",
172 ephemeral: true
173 });
174 }
175
176 return false;
177 }
178 });
179
180 collector.on("collect", async (interaction: ButtonInteraction) => {
181 log("Here 2");
182 if (!interaction.customId.endsWith(this.id)) {
183 return;
184 }
185
186 const maxPage = Math.ceil((this.data?.length ?? this.maxPage) / this.options.limit);
187 const componentOptions = { first: true, last: true, next: true, back: true };
188
189 if ([`pagination_next_${this.id}`, `pagination_back_${this.id}`].includes(interaction.customId)) {
190 log("here");
191
192 if (this.currentPage >= maxPage && interaction.customId === `pagination_next_${this.id}`) {
193 log("here");
194 await interaction.reply({
195 content: maxPage === 1 ? "This is the only page!" : "You've reached the last page!",
196 ephemeral: true
197 });
198 return;
199 }
200
201 if (this.currentPage <= 1 && interaction.customId === `pagination_back_${this.id}`) {
202 log("here");
203 await interaction.reply({
204 content: maxPage === 1 ? "This is the only page!" : "You're in the very first page!",
205 ephemeral: true
206 });
207 return;
208 }
209 }
210
211 if (interaction.customId === `pagination_first_${this.id}`) this.currentPage = 1;
212 else if (interaction.customId === `pagination_last_${this.id}`) this.currentPage = maxPage;
213
214 await interaction.update(
215 await this.getMessageOptions(
216 interaction.customId === `pagination_first_${this.id}`
217 ? 1
218 : interaction.customId === `pagination_last_${this.id}`
219 ? maxPage
220 : interaction.customId === `pagination_next_${this.id}`
221 ? this.currentPage >= maxPage
222 ? this.currentPage
223 : ++this.currentPage
224 : --this.currentPage,
225 componentOptions,
226 {
227 embeds: [],
228 ...(this.options.messageOptions ?? {})
229 }
230 )
231 );
232 });
233
234 collector.on("end", async () => {
235 const [, ...components] = message.components!; // this.getActionRow({ first: false, last: false, next: false, back: false })
236
237 try {
238 await message.edit({
239 components: [
240 this.getActionRow({
241 back: false,
242 first: false,
243 last: false,
244 next: false
245 }),
246 ...components
247 ]
248 });
249 } catch (e) {
250 log(e);
251 }
252 });
253 }
254 }
255
256 export interface EmbedBuilderOptions<T> {
257 data: Array<T>;
258 currentPage: number;
259 maxPages: number;
260 }
261
262 export interface FetchDataOption {
263 currentPage: number;
264 offset: number;
265 limit: number;
266 }
267
268 export type MessageOptions = MessageReplyOptions & InteractionReplyOptions & MessageEditOptions;
269
270 export interface PaginationOptions<T> {
271 client: Client;
272 limit: number;
273 guildId: string;
274 channelId: string;
275 userId?: string;
276 timeout?: number;
277 maxData?: (options: FetchDataOption) => Promise<number>;
278 fetchData?: (options: FetchDataOption) => Promise<T[]>;
279 messageOptions?: MessageOptions;
280 embedBuilder: (options: EmbedBuilderOptions<T>) => EmbedBuilder;
281 actionRowBuilder?: (
282 options: { first: boolean; last: boolean; next: boolean; back: boolean },
283 id: string
284 ) => ActionRowBuilder<ButtonBuilder>;
285 }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26