Skip to content

Commit 765c8bd

Browse files
committed
Add changes from revive branch
1 parent 2eccbf9 commit 765c8bd

File tree

7 files changed

+776
-1
lines changed

7 files changed

+776
-1
lines changed

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -133,3 +133,6 @@ dmypy.json
133133

134134
# config
135135
config.toml
136+
137+
.vscode/
138+
.idea/

core/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -25,3 +25,4 @@
2525
from . import errors as errors, utils as utils
2626
from .converters import *
2727
from .core import *
28+
from .context import *

core/config.py

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
"""MIT License
2+
3+
Copyright (c) 2020 - Current PythonistaGuild
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.
22+
"""
23+
import toml
24+
25+
__all__ = ("CONFIG",)
26+
27+
with open("config.toml", "r") as fp:
28+
CONFIG = toml.load(fp)

core/context.py

+30-1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44

55
import discord
66
from discord.ext import commands
7+
from constants import GUILD_ID, Roles
78

89

910
if TYPE_CHECKING:
@@ -19,7 +20,35 @@
1920

2021

2122
class Context(commands.Context["Bot"]):
22-
pass
23+
bot: Bot
24+
25+
def author_is_mod(self) -> bool:
26+
member: discord.Member
27+
if self.guild is None: # dms
28+
guild = self.bot.get_guild(GUILD_ID)
29+
if not guild:
30+
return False
31+
32+
_member = guild.get_member(self.author.id)
33+
if _member is not None:
34+
member = _member
35+
36+
else:
37+
return False
38+
39+
else:
40+
member = self.author # type: ignore
41+
42+
roles = member._roles
43+
return roles.has(Roles.ADMIN) or roles.has(Roles.MODERATOR)
44+
45+
@discord.utils.copy_doc(commands.Context.reply)
46+
async def reply(self, content: str, **kwargs) -> discord.Message:
47+
if "mention_author" not in kwargs:
48+
kwargs["mention_author"] = False
49+
50+
return await super().reply(content, **kwargs)
51+
2352

2453

2554
class GuildContext(Context):

core/paginator.py

+304
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,304 @@
1+
"""MIT License
2+
3+
Copyright (c) 2020 - Current PythonistaGuild
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.
22+
"""
23+
import asyncio
24+
from typing import Any, Dict, List, Optional
25+
26+
import discord
27+
from discord import ui # shortcut because I'm lazy
28+
from discord.ext.commands import CommandError, Paginator as _Paginator
29+
from discord.utils import MISSING
30+
31+
__all__ = ("CannotPaginate", "Pager", "KVPager", "TextPager")
32+
33+
34+
class CannotPaginate(CommandError):
35+
pass
36+
37+
38+
class Pager(ui.View):
39+
message: Optional[discord.Message] = None
40+
41+
def __init__(
42+
self,
43+
ctx,
44+
*,
45+
entries,
46+
per_page=12,
47+
show_entry_count=True,
48+
title=None,
49+
embed_color=discord.Color.blurple(),
50+
nocount=False,
51+
delete_after=True,
52+
author=None,
53+
author_url=None,
54+
stop=False,
55+
):
56+
super().__init__()
57+
self.bot = ctx.bot
58+
self.stoppable = stop
59+
self.ctx = ctx
60+
self.delete_after = delete_after
61+
self.entries = entries
62+
self.embed_author = author, author_url
63+
self.channel = ctx.channel
64+
self.author = ctx.author
65+
self.nocount = nocount
66+
self.title = title
67+
self.per_page = per_page
68+
69+
pages, left_over = divmod(len(self.entries), self.per_page)
70+
if left_over:
71+
pages += 1
72+
73+
self.maximum_pages = pages
74+
self.embed = discord.Embed(colour=embed_color)
75+
self.paginating = len(entries) > per_page
76+
self.show_entry_count = show_entry_count
77+
self.reaction_emojis = [
78+
("\N{BLACK LEFT-POINTING DOUBLE TRIANGLE WITH VERTICAL BAR}", self.first_page),
79+
("\N{BLACK LEFT-POINTING TRIANGLE}", self.previous_page),
80+
("\N{BLACK RIGHT-POINTING TRIANGLE}", self.next_page),
81+
("\N{BLACK RIGHT-POINTING DOUBLE TRIANGLE WITH VERTICAL BAR}", self.last_page),
82+
]
83+
84+
if stop:
85+
self.reaction_emojis.append(("\N{BLACK SQUARE FOR STOP}", self.stop_pages)) # type: ignore
86+
87+
if ctx.guild is not None:
88+
self.permissions = self.channel.permissions_for(ctx.guild.me)
89+
else:
90+
self.permissions = self.channel.permissions_for(ctx.bot.user)
91+
92+
if not self.permissions.embed_links:
93+
raise CannotPaginate("Bot does not have embed links permission.")
94+
95+
if not self.permissions.send_messages:
96+
raise CannotPaginate("Bot cannot send messages.")
97+
98+
def setup_buttons(self):
99+
self.clear_items()
100+
for emoji, button in self.reaction_emojis:
101+
btn = ui.Button(emoji=emoji)
102+
btn.callback = button # type: ignore
103+
self.add_item(btn)
104+
105+
def get_page(self, page: int):
106+
base = (page - 1) * self.per_page
107+
return self.entries[base : base + self.per_page]
108+
109+
def get_content(self, entries: List[Any], page: int, *, first=False):
110+
return None
111+
112+
def get_embed(self, entries: List[Any], page: int, *, first=False):
113+
self.prepare_embed(entries, page, first=first)
114+
return self.embed
115+
116+
def prepare_embed(self, entries: List[Any], page: int, *, first=False):
117+
p = []
118+
for index, entry in enumerate(entries, 1 + ((page - 1) * self.per_page)):
119+
if self.nocount:
120+
p.append(entry)
121+
else:
122+
p.append(f"{index}. {entry}")
123+
124+
if self.maximum_pages > 1:
125+
if self.show_entry_count:
126+
text = f"Page {page}/{self.maximum_pages} ({len(self.entries)} entries)"
127+
else:
128+
text = f"Page {page}/{self.maximum_pages}"
129+
130+
self.embed.set_footer(text=text)
131+
132+
if self.paginating and first:
133+
p.append("")
134+
135+
if self.embed_author[0]:
136+
self.embed.set_author(name=self.embed_author[0], icon_url=self.embed_author[1] or MISSING)
137+
138+
self.embed.description = "\n".join(p)
139+
self.embed.title = self.title or MISSING
140+
141+
async def show_page(self, page: int, *, first=False, msg_kwargs: Dict[str, Any] | None = None):
142+
self.current_page = page
143+
entries = self.get_page(page)
144+
content = self.get_content(entries, page, first=first)
145+
embed = self.get_embed(entries, page, first=first)
146+
147+
if not self.paginating:
148+
return await self.channel.send(content=content, embed=embed, view=self, **msg_kwargs or {})
149+
150+
if not first:
151+
if self.message:
152+
await self.message.edit(content=content, embed=embed, view=self)
153+
return
154+
155+
self.message = await self.channel.send(content=content, embed=embed, view=self)
156+
157+
async def checked_show_page(self, page: int):
158+
if page != 0 and page <= self.maximum_pages:
159+
await self.show_page(page)
160+
161+
async def first_page(self, inter: discord.Interaction):
162+
await inter.response.defer()
163+
await self.show_page(1)
164+
165+
async def last_page(self, inter: discord.Interaction):
166+
await inter.response.defer()
167+
await self.show_page(self.maximum_pages)
168+
169+
async def next_page(self, inter: discord.Interaction):
170+
await inter.response.defer()
171+
await self.checked_show_page(self.current_page + 1)
172+
173+
async def previous_page(self, inter: discord.Interaction):
174+
await inter.response.defer()
175+
await self.checked_show_page(self.current_page - 1)
176+
177+
async def show_current_page(self, inter: discord.Interaction):
178+
await inter.response.defer()
179+
if self.paginating:
180+
await self.show_page(self.current_page)
181+
182+
async def numbered_page(self, inter: discord.Interaction):
183+
await inter.response.defer()
184+
to_delete = [await self.channel.send("What page do you want to go to?")]
185+
186+
def message_check(m):
187+
return m.author == self.author and self.channel == m.channel and m.content.isdigit()
188+
189+
try:
190+
msg = await self.bot.wait_for("message", check=message_check, timeout=30.0)
191+
except asyncio.TimeoutError:
192+
to_delete.append(await self.channel.send("Took too long."))
193+
await asyncio.sleep(5)
194+
else:
195+
page = int(msg.content)
196+
to_delete.append(msg)
197+
if page != 0 and page <= self.maximum_pages:
198+
await self.show_page(page)
199+
else:
200+
to_delete.append(await self.channel.send(f"Invalid page given. ({page}/{self.maximum_pages})"))
201+
await asyncio.sleep(5)
202+
203+
try:
204+
await self.channel.delete_messages(to_delete)
205+
except Exception:
206+
pass
207+
208+
async def stop_pages(self, interaction: discord.Interaction | None = None):
209+
"""stops the interactive pagination session"""
210+
if self.delete_after and self.message:
211+
await self.message.delete()
212+
213+
super().stop()
214+
215+
stop = stop_pages # type: ignore
216+
217+
def _check(self, interaction: discord.Interaction):
218+
if interaction.user.id != self.author.id:
219+
return False
220+
221+
return True
222+
223+
async def interaction_check(self, interaction: discord.Interaction) -> bool:
224+
resp = self._check(interaction)
225+
226+
if not resp:
227+
await interaction.response.send_message("You cannot use this menu", ephemeral=True)
228+
229+
return resp
230+
231+
async def paginate(self, msg_kwargs: Dict[str, Any] | None = None):
232+
if self.maximum_pages > 1:
233+
self.paginating = True
234+
235+
self.setup_buttons()
236+
await self.show_page(1, first=True, msg_kwargs=msg_kwargs)
237+
238+
await self.wait()
239+
if self.delete_after and self.paginating and self.message:
240+
try:
241+
await self.message.delete()
242+
except discord.HTTPException:
243+
pass
244+
245+
246+
class KVPager(Pager):
247+
def __init__(
248+
self,
249+
ctx,
250+
*,
251+
entries: list[tuple[str, str]],
252+
per_page=12,
253+
show_entry_count=True,
254+
description=None,
255+
title=None,
256+
embed_color=discord.Color.blurple(),
257+
**kwargs,
258+
):
259+
super().__init__(
260+
ctx,
261+
entries=entries,
262+
per_page=per_page,
263+
show_entry_count=show_entry_count,
264+
title=title,
265+
embed_color=embed_color,
266+
**kwargs,
267+
)
268+
self.description = description
269+
270+
def prepare_embed(self, entries: List[Any], page: int, *, first=False):
271+
self.embed.clear_fields()
272+
self.embed.description = self.description or MISSING
273+
self.embed.title = self.title or MISSING
274+
275+
for key, value in entries:
276+
self.embed.add_field(name=key, value=value, inline=False)
277+
278+
if self.maximum_pages > 1:
279+
if self.show_entry_count:
280+
text = f"Page {page}/{self.maximum_pages} ({len(self.entries)} entries)"
281+
else:
282+
text = f"Page {page}/{self.maximum_pages}"
283+
284+
self.embed.set_footer(text=text)
285+
286+
287+
class TextPager(Pager):
288+
def __init__(self, ctx, text, *, prefix="```", suffix="```", max_size=2000, stop=False):
289+
paginator = _Paginator(prefix=prefix, suffix=suffix, max_size=max_size - 200)
290+
for line in text.split("\n"):
291+
paginator.add_line(line)
292+
293+
super().__init__(ctx, entries=paginator.pages, per_page=1, show_entry_count=False, stop=stop)
294+
295+
def get_page(self, page):
296+
return self.entries[page - 1]
297+
298+
def get_embed(self, entries, page, *, first=False):
299+
return None
300+
301+
def get_content(self, entry, page, *, first=False):
302+
if self.maximum_pages > 1:
303+
return f"{entry}\nPage {page}/{self.maximum_pages}"
304+
return entry

0 commit comments

Comments
 (0)