|
|
|
import asyncio
|
|
|
|
import random
|
|
|
|
|
|
|
|
import mariadb
|
|
|
|
from discord import DMChannel, Embed
|
|
|
|
from discord.ext import commands
|
|
|
|
|
|
|
|
import db
|
|
|
|
from settings import blank_space, enso_embedmod_colours, time, enso_guild_ID, enso_modmail_ID, hammyMention, ensoMention
|
|
|
|
|
|
|
|
|
|
|
|
# Method to send the prompt/embed to start sending modmail to the user
|
|
|
|
def startModMail(author):
|
|
|
|
# Set up embed to let the user how to start sending modmail
|
|
|
|
startModMailEmbed = Embed(title="**Welcome to Modmail!**",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
startModMailEmbed.set_thumbnail(url=author.avatar_url)
|
|
|
|
startModMailEmbed.set_footer(text=f"Sent by {author}")
|
|
|
|
|
|
|
|
fields = [
|
|
|
|
(blank_space, "**React to this message if you want to send a message to the Staff Team!**", False),
|
|
|
|
(blank_space, "**Use :white_check_mark: for** `Yes`", True),
|
|
|
|
(blank_space, "**Use :x: for** `No`", True),
|
|
|
|
(blank_space, blank_space, True),
|
|
|
|
(blank_space,
|
|
|
|
"We encourage all suggestions/thoughts and opinions on the server! As long as it is **valid** criticism. "
|
|
|
|
"Purely negative feedback will not be considered.", True)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
startModMailEmbed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return startModMailEmbed
|
|
|
|
|
|
|
|
|
|
|
|
# Method to ask the user if they want to be anonymous or not
|
|
|
|
def AnonOrNot(author):
|
|
|
|
# Set up embed to let the user how to start sending modmail
|
|
|
|
AnonModMailEmbed = Embed(title="**Want to send it Anonymously?**",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
AnonModMailEmbed.set_thumbnail(url=author.avatar_url)
|
|
|
|
AnonModMailEmbed.set_footer(text=f"Sent by {author}")
|
|
|
|
|
|
|
|
fields = [(blank_space, "**We understand that for some things, you may want to remain Anonymous."
|
|
|
|
"\nFeel free to use the reactions below to choose!**", False),
|
|
|
|
(blank_space, "**Use :white_check_mark: for** `Yes`", True),
|
|
|
|
(blank_space, "**Use :x: for** `No`", True),
|
|
|
|
(blank_space, blank_space, True),
|
|
|
|
(blank_space,
|
|
|
|
"This will make sure that Staff do not know who is sending the mail."
|
|
|
|
"\nAgain, purely negative feedback will not be considered.", True)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
AnonModMailEmbed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return AnonModMailEmbed
|
|
|
|
|
|
|
|
|
|
|
|
# Method to send an embed to to let the user know to type into chat
|
|
|
|
def SendInstructions(author):
|
|
|
|
# Set up embed to let the user know that they have aborted the modmail
|
|
|
|
SendModMailEmbed = Embed(title="**Please enter a message for it to be sent to the staff!**",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
SendModMailEmbed.set_thumbnail(url=author.avatar_url)
|
|
|
|
SendModMailEmbed.set_footer(text=f"Sent by {author}")
|
|
|
|
|
|
|
|
fields = [(blank_space,
|
|
|
|
"**Make sure that the message is above 50 characters! Include as much detail as possible :P**",
|
|
|
|
False)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
SendModMailEmbed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return SendModMailEmbed
|
|
|
|
|
|
|
|
|
|
|
|
# Method to send an embed into chat to let the user know that their mail has been sent successfully
|
|
|
|
def MessageSentConfirmation(author):
|
|
|
|
# Set up embed to let the user know that they have sent the mail
|
|
|
|
ConfirmationEmbed = Embed(title="**Message relayed to Staff!!**",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
ConfirmationEmbed.set_thumbnail(url=author.avatar_url)
|
|
|
|
ConfirmationEmbed.set_footer(text=f"Sent by {author}")
|
|
|
|
|
|
|
|
fields = [(blank_space,
|
|
|
|
"Thank you for your input! The staff team appreciate it very much!"
|
|
|
|
f"\n As mentioned previously, please don't be hesistant to DM {hammyMention} for anything! :P",
|
|
|
|
False)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
ConfirmationEmbed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return ConfirmationEmbed
|
|
|
|
|
|
|
|
|
|
|
|
# Method to actually allow the message to be sent to #mod-mail
|
|
|
|
def SendMsgToModMail(self, msg, author):
|
|
|
|
if self.anon:
|
|
|
|
|
|
|
|
avatars = ["https://cdn.discordapp.com/embed/avatars/0.png",
|
|
|
|
"https://cdn.discordapp.com/embed/avatars/1.png",
|
|
|
|
"https://cdn.discordapp.com/embed/avatars/2.png",
|
|
|
|
"https://cdn.discordapp.com/embed/avatars/3.png",
|
|
|
|
"https://cdn.discordapp.com/embed/avatars/4.png"]
|
|
|
|
|
|
|
|
embed = Embed(title="Modmail",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
embed.set_thumbnail(url=random.choice(avatars))
|
|
|
|
embed.set_footer(text=f"Requested by Anon Member")
|
|
|
|
|
|
|
|
fields = [("Member", "Anon Member", False),
|
|
|
|
("Message", msg.content, False)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
embed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return embed
|
|
|
|
|
|
|
|
else:
|
|
|
|
embed = Embed(title="Modmail",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
embed.set_thumbnail(url=author.avatar_url)
|
|
|
|
embed.set_footer(text=f"Requested by {author}")
|
|
|
|
|
|
|
|
fields = [("Member", author, False),
|
|
|
|
("Message", msg.content, False)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
embed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return embed
|
|
|
|
|
|
|
|
|
|
|
|
def logModMail(ctx, anon):
|
|
|
|
# Set up the connection to the database
|
|
|
|
conn = db.connection()
|
|
|
|
|
|
|
|
# With the connection
|
|
|
|
with conn:
|
|
|
|
# Make sure that mariaDB errors are handled properly
|
|
|
|
try:
|
|
|
|
if anon:
|
|
|
|
Anon = "True"
|
|
|
|
else:
|
|
|
|
Anon = "False"
|
|
|
|
|
|
|
|
msg_name = ctx.message.author.name
|
|
|
|
msg_discrim = ctx.message.author.discriminator
|
|
|
|
time = ctx.message.created_at
|
|
|
|
|
|
|
|
# Get:
|
|
|
|
msg_time = time.strftime('%Y-%m-%d %H:%M:%S') # Time of the Message
|
|
|
|
msg_author = f"{msg_name}#{msg_discrim}" # DiscordID
|
|
|
|
msg_content = f"{ctx.message.content}" # Content of the message
|
|
|
|
|
|
|
|
# Store the variables
|
|
|
|
val = Anon, msg_time, msg_author, msg_content
|
|
|
|
|
|
|
|
# Define the Insert Into Statement inserting into the database
|
|
|
|
insert_query = """INSERT INTO modmail (Anon, messageTime, discordID, messageContent) VALUES (?, ?, ?, ?)"""
|
|
|
|
cursor = conn.cursor()
|
|
|
|
# Execute the SQL Query
|
|
|
|
cursor.execute(insert_query, val)
|
|
|
|
conn.commit()
|
|
|
|
print(cursor.rowcount, "Record inserted successfully into Modmail")
|
|
|
|
|
|
|
|
except mariadb.Error as ex:
|
|
|
|
print("Parameterized Query Failed: {}".format(ex))
|
|
|
|
|
|
|
|
|
|
|
|
# Method to send an embed to let the user know that they have aborted the modmail process
|
|
|
|
def Abort(author):
|
|
|
|
# Set up embed to let the user know that they have aborted the modmail
|
|
|
|
AbortEmbed = Embed(title="**Aborting ModMail!**",
|
|
|
|
colour=enso_embedmod_colours,
|
|
|
|
timestamp=time)
|
|
|
|
|
|
|
|
AbortEmbed.set_thumbnail(url=author.avatar_url)
|
|
|
|
AbortEmbed.set_footer(text=f"Sent by {author}")
|
|
|
|
|
|
|
|
fields = [
|
|
|
|
(blank_space, "**If you change your mind, you can do `~mm` or `~modmail` at anytime!**", False),
|
|
|
|
(blank_space, f"If you want to speak to me personally, you can DM {hammyMention} anytime!", True)]
|
|
|
|
|
|
|
|
for name, value, inline in fields:
|
|
|
|
AbortEmbed.add_field(name=name, value=value, inline=inline)
|
|
|
|
|
|
|
|
return AbortEmbed
|
|
|
|
|
|
|
|
|
|
|
|
# Set up the Cog
|
|
|
|
class Modmail(commands.Cog):
|
|
|
|
def __init__(self, bot):
|
|
|
|
self.bot = bot
|
|
|
|
self.reaction = None
|
|
|
|
self.anon = None
|
|
|
|
|
|
|
|
# Allows for the modmail system
|
|
|
|
@commands.command(name="modmail", aliases=["mm"])
|
|
|
|
async def mod_mail(self, ctx):
|
|
|
|
self.anon = None
|
|
|
|
|
|
|
|
# Get the mod-mail channel
|
|
|
|
channel = self.bot.get_channel(enso_modmail_ID)
|
|
|
|
# Get the guild Enso
|
|
|
|
guild = self.bot.get_guild(enso_guild_ID)
|
|
|
|
# Get the member
|
|
|
|
member = guild.get_member(ctx.author.id)
|
|
|
|
|
|
|
|
# Making sure the user is in a DM channel with the bot
|
|
|
|
if isinstance(ctx.message.channel, DMChannel):
|
|
|
|
|
|
|
|
# Asking if the user wants to send staff mail
|
|
|
|
modmail = await ctx.send(embed=startModMail(member))
|
|
|
|
# Add reactions to the message
|
|
|
|
await modmail.add_reaction('✅')
|
|
|
|
await modmail.add_reaction('❌')
|
|
|
|
|
|
|
|
# Surround with try/except to catch any exceptions that may occur
|
|
|
|
try:
|
|
|
|
|
|
|
|
# Checking if the user reacted with ✅ with response to sending staff a message
|
|
|
|
def emoji_check(reaction, user):
|
|
|
|
return user == ctx.author and str(reaction.emoji) in ['✅', '❌']
|
|
|
|
|
|
|
|
# Surround with try/except to catch any exceptions that may occur
|
|
|
|
try:
|
|
|
|
# Wait for the user to add a reaction
|
|
|
|
reaction, user = await self.bot.wait_for('reaction_add', timeout=120.0, check=emoji_check)
|
|
|
|
except Exception as ex:
|
|
|
|
print(ex)
|
|
|
|
return
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
if str(reaction.emoji) == "✅":
|
|
|
|
|
|
|
|
# Delete the old embed
|
|
|
|
await modmail.delete()
|
|
|
|
|
|
|
|
# Ask the user if they want the mail to be anonymized
|
|
|
|
anonornot = await ctx.send(embed=AnonOrNot(member))
|
|
|
|
# Add reactions to the message
|
|
|
|
await anonornot.add_reaction('✅')
|
|
|
|
await anonornot.add_reaction('❌')
|
|
|
|
|
|
|
|
# Surround with try/except to catch any exceptions that may occur
|
|
|
|
try:
|
|
|
|
|
|
|
|
# Wait for the user to add a reaction
|
|
|
|
reaction, user = await self.bot.wait_for('reaction_add', timeout=120.0, check=emoji_check)
|
|
|
|
except Exception as ex:
|
|
|
|
print(ex)
|
|
|
|
return
|
|
|
|
|
|
|
|
else:
|
|
|
|
if str(reaction.emoji) == "✅":
|
|
|
|
self.anon = True
|
|
|
|
|
|
|
|
# Delete the old embed
|
|
|
|
await anonornot.delete()
|
|
|
|
|
|
|
|
# Tell the user to type their mail into the chat
|
|
|
|
instructions = await ctx.send(embed=SendInstructions(member))
|
|
|
|
|
|
|
|
# Making sure that the reply is from the author
|
|
|
|
def check(m):
|
|
|
|
return m.author == ctx.author
|
|
|
|
|
|
|
|
# Wait for the message from the author
|
|
|
|
msg = await self.bot.wait_for('message', check=check, timeout=300)
|
|
|
|
|
|
|
|
while len(msg.content) < 50:
|
|
|
|
await ctx.send("**Make sure your mail is above 50 characters!!**"
|
|
|
|
"\n**This helps us reduce spam and allows you to include more detail in your mail**")
|
|
|
|
|
|
|
|
# Wait for the message from the author
|
|
|
|
msg = await self.bot.wait_for('message', check=check, timeout=300)
|
|
|
|
|
|
|
|
await channel.send(embed=SendMsgToModMail(self, msg, member))
|
|
|
|
await ctx.send(embed=MessageSentConfirmation(member))
|
|
|
|
logModMail(ctx, self.anon)
|
|
|
|
await instructions.delete()
|
|
|
|
|
|
|
|
if str(reaction.emoji) == "❌":
|
|
|
|
self.anon = False
|
|
|
|
|
|
|
|
# Delete the old embed
|
|
|
|
await anonornot.delete()
|
|
|
|
|
|
|
|
# Tell the user to type their mail into the chat
|
|
|
|
instructions = await ctx.send(embed=SendInstructions(member))
|
|
|
|
|
|
|
|
# Making sure that the reply is from the author
|
|
|
|
def check(m):
|
|
|
|
return m.author == ctx.author
|
|
|
|
|
|
|
|
# Wait for the message from the author
|
|
|
|
msg = await self.bot.wait_for('message', check=check, timeout=300)
|
|
|
|
|
|
|
|
while len(msg.content) < 50:
|
|
|
|
await ctx.send("Make sure your mail is above **50*8 characters!!"
|
|
|
|
"\nThis helps us reduce spam and allows you to include more detail in your mail")
|
|
|
|
|
|
|
|
# Wait for the message from the author
|
|
|
|
msg = await self.bot.wait_for('message', check=check, timeout=300)
|
|
|
|
|
|
|
|
await channel.send(embed=SendMsgToModMail(self, msg, member))
|
|
|
|
await ctx.send(embed=MessageSentConfirmation(member))
|
|
|
|
logModMail(ctx, self.anon)
|
|
|
|
await instructions.delete()
|
|
|
|
|
|
|
|
if self.anon is None:
|
|
|
|
if str(reaction.emoji) == "❌":
|
|
|
|
# Delete the old embed
|
|
|
|
await modmail.delete()
|
|
|
|
|
|
|
|
# Send the Abort embed to the user
|
|
|
|
await ctx.send(embed=Abort(member))
|
|
|
|
return
|
|
|
|
|
|
|
|
except Exception as ex:
|
|
|
|
print(ex)
|
|
|
|
|
|
|
|
# Send out an error message if the user waited too long
|
|
|
|
await ctx.send("ModMail Timed Out! Do `~mm` or `~modmail` if you want to use the ModMail system!")
|
|
|
|
|
|
|
|
else:
|
|
|
|
message = await ctx.send(
|
|
|
|
f"{ctx.author.mention} **ModMail can only be sent through DM's!** "
|
|
|
|
f"\nSuggestions and Opinions on the server are always appreciated!\n"
|
|
|
|
f"Make sure you DM {ensoMention} and then use `~modmail` or `~mm`")
|
|
|
|
|
|
|
|
# Let the User read the message for 10 seconds
|
|
|
|
await asyncio.sleep(10.0)
|
|
|
|
# Delete the message
|
|
|
|
await message.delete()
|
|
|
|
|
|
|
|
|
|
|
|
def setup(bot):
|
|
|
|
bot.add_cog(Modmail(bot))
|