2020-02-25 18:15:49 -05:00
|
|
|
import io
|
2020-05-05 20:39:31 -04:00
|
|
|
import os.path
|
2023-03-09 23:01:10 +01:00
|
|
|
|
|
|
|
import discord
|
2020-02-25 18:15:49 -05:00
|
|
|
from discord.ext import commands
|
|
|
|
from discord.ext.commands import Cog
|
|
|
|
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
class Lists(Cog):
|
|
|
|
"""
|
|
|
|
Manages channels that are dedicated to lists.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, bot):
|
|
|
|
self.bot = bot
|
|
|
|
|
|
|
|
# Helpers
|
|
|
|
|
|
|
|
def check_if_target_is_staff(self, target):
|
2023-04-05 12:10:18 +02:00
|
|
|
return any(r.id in self.bot.config.staff_role_ids for r in target.roles)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_edit(self, emoji):
|
2020-04-21 01:05:32 +03:00
|
|
|
return str(emoji)[0] == "✏" or str(emoji)[0] == "📝"
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_delete(self, emoji):
|
2020-04-21 01:05:32 +03:00
|
|
|
return str(emoji)[0] == "❌" or str(emoji)[0] == "❎"
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_recycle(self, emoji):
|
2020-04-21 01:05:32 +03:00
|
|
|
return str(emoji)[0] == "♻"
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_insert_above(self, emoji):
|
2020-04-21 01:05:32 +03:00
|
|
|
return str(emoji)[0] == "⤴️" or str(emoji)[0] == "⬆"
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_insert_below(self, emoji):
|
2020-04-21 01:05:32 +03:00
|
|
|
return str(emoji)[0] == "⤵️" or str(emoji)[0] == "⬇"
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
def is_reaction_valid(self, reaction):
|
|
|
|
allowed_reactions = [
|
2020-04-21 01:05:32 +03:00
|
|
|
"✏",
|
|
|
|
"📝",
|
|
|
|
"❌",
|
|
|
|
"❎",
|
|
|
|
"♻",
|
|
|
|
"⤴️",
|
|
|
|
"⬆",
|
|
|
|
"⬇",
|
|
|
|
"⤵️",
|
2020-02-25 18:15:49 -05:00
|
|
|
]
|
|
|
|
return str(reaction.emoji)[0] in allowed_reactions
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
async def find_reactions(self, user_id, channel_id, limit=None):
|
2020-02-25 18:15:49 -05:00
|
|
|
reactions = []
|
|
|
|
channel = self.bot.get_channel(channel_id)
|
2020-04-21 01:05:32 +03:00
|
|
|
async for message in channel.history(limit=limit):
|
2020-02-25 18:15:49 -05:00
|
|
|
if len(message.reactions) == 0:
|
|
|
|
continue
|
|
|
|
|
|
|
|
for reaction in message.reactions:
|
|
|
|
users = await reaction.users().flatten()
|
|
|
|
user_ids = map(lambda user: user.id, users)
|
|
|
|
if user_id in user_ids:
|
|
|
|
reactions.append(reaction)
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
return reactions
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
def create_log_message(self, emoji, action, user, channel, reason=""):
|
|
|
|
msg = (
|
|
|
|
f"{emoji} **{action}** \n"
|
2020-02-25 18:15:49 -05:00
|
|
|
f"from {self.bot.escape_message(user.name)} ({user.id}), in {channel.mention}"
|
2020-04-21 01:05:32 +03:00
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
if reason != "":
|
|
|
|
msg += f":\n`{reason}`"
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
return msg
|
|
|
|
|
|
|
|
async def clean_up_raw_text_file_message(self, message):
|
|
|
|
embeds = message.embeds
|
|
|
|
if len(embeds) == 0:
|
|
|
|
return
|
|
|
|
|
|
|
|
fields = embeds[0].fields
|
|
|
|
for field in fields:
|
|
|
|
if field.name == "Message ID":
|
2023-04-05 12:10:18 +02:00
|
|
|
files_channel = self.bot.get_channel(self.bot.config.list_files_channel)
|
2020-02-25 18:15:49 -05:00
|
|
|
file_message = await files_channel.fetch_message(int(field.value))
|
|
|
|
await file_message.delete()
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await message.edit(embed=None)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-05-05 19:43:23 -04:00
|
|
|
async def cache_message(self, message):
|
|
|
|
msg = {
|
|
|
|
"has_attachment": False,
|
|
|
|
"attachment_filename": "",
|
|
|
|
"attachment_data": b"",
|
|
|
|
"content": message.content,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(message.attachments) != 0:
|
|
|
|
attachment = next(
|
|
|
|
(
|
|
|
|
a
|
|
|
|
for a in message.attachments
|
2020-05-05 20:39:31 -04:00
|
|
|
if os.path.splitext(a.filename)[1] in [".png", ".jpg", ".jpeg"]
|
2020-05-05 19:43:23 -04:00
|
|
|
),
|
|
|
|
None,
|
|
|
|
)
|
|
|
|
if attachment is not None:
|
|
|
|
msg["has_attachment"] = True
|
|
|
|
msg["attachment_filename"] = attachment.filename
|
|
|
|
msg["attachment_data"] = await attachment.read()
|
|
|
|
|
|
|
|
return msg
|
|
|
|
|
|
|
|
async def send_cached_message(self, channel, message):
|
|
|
|
if message["has_attachment"] == True:
|
|
|
|
file = discord.File(
|
|
|
|
io.BytesIO(message["attachment_data"]),
|
|
|
|
filename=message["attachment_filename"],
|
|
|
|
)
|
|
|
|
await channel.send(content=message["content"], file=file)
|
|
|
|
else:
|
|
|
|
await channel.send(content=message["content"])
|
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
# Commands
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
@commands.command(aliases=["list"])
|
2020-02-25 18:15:49 -05:00
|
|
|
async def listitem(self, ctx, channel: discord.TextChannel, number: int):
|
|
|
|
"""Link to a specific list item."""
|
|
|
|
if number <= 0:
|
|
|
|
await ctx.send(f"Number must be greater than 0.")
|
|
|
|
return
|
|
|
|
|
2023-04-05 12:10:18 +02:00
|
|
|
if channel.id not in self.bot.config.list_channels:
|
2020-02-25 18:15:49 -05:00
|
|
|
await ctx.send(f"{channel.mention} is not a list channel.")
|
|
|
|
return
|
|
|
|
|
|
|
|
counter = 0
|
2020-04-21 01:05:32 +03:00
|
|
|
async for message in channel.history(limit=None, oldest_first=True):
|
2020-02-25 18:15:49 -05:00
|
|
|
if message.content.strip():
|
|
|
|
counter += 1
|
|
|
|
|
|
|
|
if counter == number:
|
|
|
|
embed = discord.Embed(
|
2020-04-21 01:05:32 +03:00
|
|
|
title=f"Item #{number} in #{channel.name}",
|
|
|
|
description=message.content,
|
|
|
|
url=message.jump_url,
|
2020-02-25 18:15:49 -05:00
|
|
|
)
|
2020-04-21 01:05:32 +03:00
|
|
|
await ctx.send(content="", embed=embed)
|
2020-02-25 18:15:49 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
await ctx.send(f"Unable to find item #{number} in {channel.mention}.")
|
|
|
|
|
|
|
|
# Listeners
|
|
|
|
|
|
|
|
@Cog.listener()
|
|
|
|
async def on_raw_reaction_add(self, payload):
|
|
|
|
await self.bot.wait_until_ready()
|
|
|
|
|
|
|
|
# We only care about reactions in Rules, and Support FAQ
|
2023-04-05 12:10:18 +02:00
|
|
|
if payload.channel_id not in self.bot.config.list_channels:
|
2020-02-25 18:15:49 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
channel = self.bot.get_channel(payload.channel_id)
|
|
|
|
message = await channel.fetch_message(payload.message_id)
|
|
|
|
member = channel.guild.get_member(payload.user_id)
|
|
|
|
user = self.bot.get_user(payload.user_id)
|
2020-02-25 20:08:55 -05:00
|
|
|
reaction = next(
|
2020-04-21 01:05:32 +03:00
|
|
|
(
|
|
|
|
reaction
|
|
|
|
for reaction in message.reactions
|
|
|
|
if str(reaction.emoji) == str(payload.emoji)
|
|
|
|
),
|
|
|
|
None,
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
if reaction is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Only staff can add reactions in these channels.
|
|
|
|
if not self.check_if_target_is_staff(member):
|
|
|
|
await reaction.remove(user)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Reactions are only allowed on messages from the bot.
|
|
|
|
if not message.author.bot:
|
|
|
|
await reaction.remove(user)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Only certain reactions are allowed.
|
|
|
|
if not self.is_reaction_valid(reaction):
|
|
|
|
await reaction.remove(user)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Remove all other reactions from user in this channel.
|
|
|
|
for r in await self.find_reactions(payload.user_id, payload.channel_id):
|
2020-04-21 01:05:32 +03:00
|
|
|
if r.message.id != message.id or (
|
|
|
|
r.message.id == message.id and str(r.emoji) != str(reaction.emoji)
|
|
|
|
):
|
2020-02-25 18:15:49 -05:00
|
|
|
await r.remove(user)
|
|
|
|
|
|
|
|
# When editing we want to provide the user a copy of the raw text.
|
2023-04-05 12:10:18 +02:00
|
|
|
if self.is_edit(reaction.emoji) and self.bot.config.list_files_channel != 0:
|
|
|
|
files_channel = self.bot.get_channel(self.bot.config.list_files_channel)
|
2020-02-25 20:08:55 -05:00
|
|
|
file = discord.File(
|
|
|
|
io.BytesIO(message.content.encode("utf-8")),
|
2020-04-21 01:05:32 +03:00
|
|
|
filename=f"{message.id}.txt",
|
|
|
|
)
|
|
|
|
file_message = await files_channel.send(file=file)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
embed = discord.Embed(
|
2020-04-21 01:05:32 +03:00
|
|
|
title="Click here to get the raw text to modify.",
|
|
|
|
url=f"{file_message.attachments[0].url}?",
|
|
|
|
)
|
|
|
|
embed.add_field(name="Message ID", value=file_message.id, inline=False)
|
|
|
|
await message.edit(embed=embed)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
@Cog.listener()
|
|
|
|
async def on_raw_reaction_remove(self, payload):
|
|
|
|
await self.bot.wait_until_ready()
|
|
|
|
|
|
|
|
# We only care about reactions in Rules, and Support FAQ
|
2023-04-05 12:10:18 +02:00
|
|
|
if payload.channel_id not in self.bot.config.list_channels:
|
2020-02-25 18:15:49 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
channel = self.bot.get_channel(payload.channel_id)
|
|
|
|
message = await channel.fetch_message(payload.message_id)
|
|
|
|
|
|
|
|
# Reaction was removed from a message we don"t care about.
|
|
|
|
if not message.author.bot:
|
|
|
|
return
|
|
|
|
|
|
|
|
# We want to remove the embed we added.
|
2023-04-05 12:10:18 +02:00
|
|
|
if self.is_edit(payload.emoji) and self.bot.config.list_files_channel != 0:
|
2020-02-25 18:15:49 -05:00
|
|
|
await self.clean_up_raw_text_file_message(message)
|
|
|
|
|
|
|
|
@Cog.listener()
|
|
|
|
async def on_message(self, message):
|
|
|
|
await self.bot.wait_until_ready()
|
|
|
|
|
|
|
|
# We only care about messages in Rules, and Support FAQ
|
2023-04-05 12:10:18 +02:00
|
|
|
if message.channel.id not in self.bot.config.list_channels:
|
2020-02-25 18:15:49 -05:00
|
|
|
return
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
# We don"t care about messages from bots.
|
|
|
|
if message.author.bot:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Only staff can modify lists.
|
|
|
|
if not self.check_if_target_is_staff(message.author):
|
|
|
|
await message.delete()
|
|
|
|
return
|
|
|
|
|
2023-04-05 12:10:18 +02:00
|
|
|
log_channel = self.bot.get_channel(self.bot.config.log_channel)
|
2020-02-25 18:15:49 -05:00
|
|
|
channel = message.channel
|
|
|
|
content = message.content
|
|
|
|
user = message.author
|
|
|
|
|
|
|
|
attachment_filename = None
|
|
|
|
attachment_data = None
|
|
|
|
if len(message.attachments) != 0:
|
|
|
|
# Lists will only reupload the first image.
|
2020-04-21 01:05:32 +03:00
|
|
|
attachment = next(
|
|
|
|
(
|
|
|
|
a
|
|
|
|
for a in message.attachments
|
2020-05-05 20:39:31 -04:00
|
|
|
if os.path.splitext(a.filename)[1] in [".png", ".jpg", ".jpeg"]
|
2020-04-21 01:05:32 +03:00
|
|
|
),
|
|
|
|
None,
|
|
|
|
)
|
2020-02-25 20:08:55 -05:00
|
|
|
if attachment is not None:
|
2020-02-25 18:15:49 -05:00
|
|
|
attachment_filename = attachment.filename
|
|
|
|
attachment_data = await attachment.read()
|
|
|
|
|
|
|
|
await message.delete()
|
|
|
|
|
|
|
|
reactions = await self.find_reactions(user.id, channel.id)
|
|
|
|
|
2020-02-25 20:08:55 -05:00
|
|
|
# Add to the end of the list if there is no reactions or somehow more
|
|
|
|
# than one.
|
2020-02-25 18:15:49 -05:00
|
|
|
if len(reactions) != 1:
|
2020-02-25 20:08:55 -05:00
|
|
|
if attachment_filename is not None and attachment_data is not None:
|
|
|
|
file = discord.File(
|
2020-04-21 01:05:32 +03:00
|
|
|
io.BytesIO(attachment_data), filename=attachment_filename
|
|
|
|
)
|
|
|
|
await channel.send(content=content, file=file)
|
2020-02-25 18:15:49 -05:00
|
|
|
else:
|
|
|
|
await channel.send(content)
|
|
|
|
|
|
|
|
for reaction in reactions:
|
|
|
|
await reaction.remove(user)
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message("💬", "List item added:", user, channel)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
targeted_reaction = reactions[0]
|
|
|
|
targeted_message = targeted_reaction.message
|
|
|
|
|
|
|
|
if self.is_edit(targeted_reaction):
|
2023-04-05 12:10:18 +02:00
|
|
|
if self.bot.config.list_files_channel != 0:
|
2020-02-25 18:15:49 -05:00
|
|
|
await self.clean_up_raw_text_file_message(targeted_message)
|
2020-04-21 01:05:32 +03:00
|
|
|
await targeted_message.edit(content=content)
|
2020-02-25 18:15:49 -05:00
|
|
|
await targeted_reaction.remove(user)
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message("📝", "List item edited:", user, channel)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
elif self.is_delete(targeted_reaction):
|
|
|
|
await targeted_message.delete()
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message(
|
|
|
|
"❌", "List item deleted:", user, channel, content
|
|
|
|
)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
elif self.is_recycle(targeted_reaction):
|
2020-05-05 19:43:23 -04:00
|
|
|
messages = [await self.cache_message(targeted_message)]
|
|
|
|
|
|
|
|
for message in await channel.history(
|
2020-04-21 01:05:32 +03:00
|
|
|
limit=None, after=targeted_message, oldest_first=True
|
2020-05-05 19:43:23 -04:00
|
|
|
).flatten():
|
|
|
|
messages.append(await self.cache_message(message))
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await channel.purge(limit=len(messages) + 1, bulk=True)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
for message in messages:
|
2020-05-05 19:43:23 -04:00
|
|
|
await self.send_cached_message(channel, message)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message(
|
|
|
|
"♻", "List item recycled:", user, channel, content
|
|
|
|
)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
elif self.is_insert_above(targeted_reaction):
|
2020-05-05 19:43:23 -04:00
|
|
|
messages = [await self.cache_message(targeted_message)]
|
|
|
|
|
|
|
|
for message in await channel.history(
|
2020-04-21 01:05:32 +03:00
|
|
|
limit=None, after=targeted_message, oldest_first=True
|
2020-05-05 19:43:23 -04:00
|
|
|
).flatten():
|
|
|
|
messages.append(await self.cache_message(message))
|
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await channel.purge(limit=len(messages) + 1, bulk=True)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-05-05 19:43:23 -04:00
|
|
|
if attachment_filename is not None and attachment_data is not None:
|
|
|
|
file = discord.File(
|
|
|
|
io.BytesIO(attachment_data), filename=attachment_filename
|
|
|
|
)
|
|
|
|
await channel.send(content=content, file=file)
|
|
|
|
else:
|
|
|
|
await channel.send(content)
|
|
|
|
|
2020-02-25 18:15:49 -05:00
|
|
|
for message in messages:
|
2020-05-05 19:43:23 -04:00
|
|
|
await self.send_cached_message(channel, message)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message("💬", "List item added:", user, channel)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
elif self.is_insert_below(targeted_reaction):
|
2020-05-05 19:43:23 -04:00
|
|
|
await targeted_reaction.remove(user)
|
|
|
|
|
|
|
|
messages = []
|
|
|
|
|
|
|
|
for message in await channel.history(
|
2020-04-21 01:05:32 +03:00
|
|
|
limit=None, after=targeted_message, oldest_first=True
|
2020-05-05 19:43:23 -04:00
|
|
|
).flatten():
|
|
|
|
messages.append(await self.cache_message(message))
|
|
|
|
|
|
|
|
await channel.purge(limit=len(messages), bulk=True)
|
|
|
|
|
|
|
|
if attachment_filename is not None and attachment_data is not None:
|
|
|
|
file = discord.File(
|
|
|
|
io.BytesIO(attachment_data), filename=attachment_filename
|
|
|
|
)
|
|
|
|
await channel.send(content=content, file=file)
|
|
|
|
else:
|
|
|
|
await channel.send(content)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
|
|
|
for message in messages:
|
2020-05-05 19:43:23 -04:00
|
|
|
await self.send_cached_message(channel, message)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-04-21 01:05:32 +03:00
|
|
|
await log_channel.send(
|
|
|
|
self.create_log_message("💬", "List item added:", user, channel)
|
|
|
|
)
|
2020-02-25 18:15:49 -05:00
|
|
|
|
2020-02-25 20:08:55 -05:00
|
|
|
|
2022-05-24 20:35:42 +02:00
|
|
|
async def setup(bot):
|
|
|
|
await bot.add_cog(Lists(bot))
|