mirror of https://github.com/stijndcl/didier
227 lines
8.5 KiB
Python
227 lines
8.5 KiB
Python
from data import constants
|
|
import datetime
|
|
from decorators import help
|
|
import discord
|
|
from discord.ext import commands
|
|
from enums.help_categories import Category
|
|
from functions import timeFormatters, stringFormatters
|
|
from functions.database import birthdays
|
|
|
|
|
|
class Birthdays(commands.Cog):
|
|
def __init__(self, client):
|
|
self.client = client
|
|
|
|
# Don't allow any commands to work when locked
|
|
def cog_check(self, ctx):
|
|
return not self.client.locked
|
|
|
|
@commands.group(name="Birthday", aliases=["Bd", "Birthdays"], case_insensitive=True, invoke_without_command=True)
|
|
@help.Category(Category.Other)
|
|
async def birthday(self, ctx, member: discord.Member = None):
|
|
"""
|
|
Command to check the birthday of yourself/another person.
|
|
:param ctx: Discord Context
|
|
:param member: The member to check
|
|
"""
|
|
if member is not None:
|
|
# A member was tagged
|
|
nameStr = "**{}**'s".format(member.display_name)
|
|
res = birthdays.get_user(member.id)
|
|
else:
|
|
# No member passed -> check the user's birthday
|
|
nameStr = "Jouw"
|
|
res = birthdays.get_user(ctx.author.id)
|
|
|
|
if not res:
|
|
# Nothing found in the db for this member
|
|
return await ctx.send("{} verjaardag zit nog niet in de database.".format(nameStr))
|
|
|
|
# Create a datetime object of the upcoming birthday,
|
|
# and a formatted string displaying the date
|
|
dayDatetime, timeString = self.dmToDatetime(res[0][0], res[0][1])
|
|
|
|
# Find the weekday related to this day
|
|
weekday = timeFormatters.intToWeekday(dayDatetime.weekday()).lower()
|
|
|
|
return await ctx.send("{} verjaardag staat ingesteld op **{} {}**.".format(
|
|
nameStr, weekday, timeString
|
|
))
|
|
|
|
@birthday.command(name="Today", aliases=["Now"])
|
|
async def today(self, ctx):
|
|
"""
|
|
Command that lists all birthdays of the day.
|
|
:param ctx: Discord Context
|
|
"""
|
|
# Create a datetime object for today
|
|
dt = timeFormatters.dateTimeNow()
|
|
await ctx.send(self.getBirthdayOnDate(dt))
|
|
|
|
@birthday.command(name="Tomorrow", aliases=["Tm", "Tmw"])
|
|
async def tomorrow(self, ctx):
|
|
"""
|
|
Command that lists all birthdays of tomorrow.
|
|
:param ctx: Discord Context
|
|
"""
|
|
# Create a datetime object for tomorrow
|
|
dt = timeFormatters.dateTimeNow() + datetime.timedelta(days=1)
|
|
await ctx.send(self.getBirthdayOnDate(dt).replace("Vandaag", "Morgen").replace("vandaag", "morgen"))
|
|
|
|
@birthday.command(name="Week")
|
|
async def week(self, ctx):
|
|
"""
|
|
Command that lists all birthdays for the coming week.
|
|
:param ctx: Discord Context
|
|
"""
|
|
# Dict of all birthdays this week
|
|
this_week = {}
|
|
|
|
# Create a datetime object starting yesterday so the first line
|
|
# of the loop can add a day every time,
|
|
# as premature returning would prevent this from happening
|
|
# & get the day stuck
|
|
dt = timeFormatters.dateTimeNow() - datetime.timedelta(days=1)
|
|
|
|
# Create an embed
|
|
embed = discord.Embed(colour=discord.Colour.blue())
|
|
embed.set_author(name="Verjaardagen deze week")
|
|
|
|
# Add all people of the coming week
|
|
for dayCounter in range(7):
|
|
dt += datetime.timedelta(days=1)
|
|
res = birthdays.get_users_on_date(dt.day, dt.month)
|
|
|
|
# No birthdays on this day
|
|
if not res:
|
|
continue
|
|
|
|
# Add everyone from this day into the dict
|
|
this_week[str(dayCounter)] = {"day": dt.day, "month": dt.month, "users": []}
|
|
|
|
for user in res:
|
|
this_week[str(dayCounter)]["users"].append(user[0])
|
|
|
|
# No one found
|
|
if not this_week:
|
|
embed.description = "Deze week is er niemand jarig."
|
|
return await ctx.send(embed=embed)
|
|
|
|
COC = self.client.get_guild(int(constants.CallOfCode))
|
|
|
|
# For every day, add the list of users into the embed
|
|
for day, value in this_week.items():
|
|
|
|
dayDatetime, timeString = self.dmToDatetime(int(value["day"]), int(value["month"]))
|
|
weekday = timeFormatters.intToWeekday(dayDatetime.weekday())
|
|
|
|
embed.add_field(name="{} {}".format(weekday, timeString),
|
|
value=", ".join(COC.get_member(user).mention for user in value["users"]),
|
|
inline=False)
|
|
|
|
await ctx.send(embed=embed)
|
|
|
|
def getBirthdayOnDate(self, dt):
|
|
"""
|
|
Function to get all birthdays on a certain date.
|
|
Returns a string right away to avoid more code duplication.
|
|
:param dt: the date (Python datetime instance)
|
|
:return: A formatted string containing all birthdays on [dt]
|
|
"""
|
|
res = birthdays.get_users_on_date(dt.day, dt.month)
|
|
|
|
# Nobody's birthday
|
|
if not res:
|
|
return "Vandaag is er niemand jarig."
|
|
|
|
COC = self.client.get_guild(int(constants.CallOfCode))
|
|
|
|
# Create a list of member objects of the people that have a birthday on this date
|
|
people = [COC.get_member(int(user[0])) for user in res]
|
|
|
|
if len(people) == 1:
|
|
return "Vandaag is **{}** jarig.".format(people[0].display_name)
|
|
return "Vandaag zijn {} en {} jarig.".format(
|
|
", ".join("**" + user.display_name + "**" for user in people[:-1]),
|
|
people[-1].display_name
|
|
)
|
|
|
|
def dmToDatetime(self, day, month):
|
|
"""
|
|
Converts a day + month to a datetime instance.
|
|
:param day: the day in the date
|
|
:param month: the month in the date
|
|
:return: a datetime instance representing the next time this date occurs,
|
|
and a formatted string for this date
|
|
"""
|
|
now = timeFormatters.dateTimeNow()
|
|
year = now.year
|
|
|
|
# Add an extra year to the date in case it has already passed
|
|
if month < now.month or (month == now.month and day < now.day):
|
|
year += 1
|
|
|
|
# Create a datetime object for this birthday
|
|
timeString = "{}/{}/{}".format(
|
|
stringFormatters.leadingZero(str(day)),
|
|
stringFormatters.leadingZero(str(month)),
|
|
year
|
|
)
|
|
|
|
dayDatetime = datetime.datetime.strptime(timeString, "%d/%m/%Y")
|
|
return dayDatetime, timeString
|
|
|
|
@birthday.command(name="Set", usage="[DD/MM/YYYY]")
|
|
async def set(self, ctx, date=None, member: discord.Member = None):
|
|
"""
|
|
Command to add your birthday into the database.
|
|
:param ctx: Discord Context
|
|
:param date: the date of your birthday
|
|
:param member: another member whose birthday has to be added/changed
|
|
"""
|
|
# No date passed
|
|
if date is None:
|
|
return await ctx.send("Geef een datum op.")
|
|
|
|
# Invalid format used
|
|
if date.count("/") != 2:
|
|
return await ctx.send("Ongeldig formaat (gebruik DD/MM/YYYY).")
|
|
|
|
# Check if anything is wrong with the date
|
|
try:
|
|
day = int(date.split("/")[0])
|
|
month = int(date.split("/")[1])
|
|
year = int(date.split("/")[2])
|
|
|
|
# This is not used, but creating an invalid datetime object throws a ValueError
|
|
# so it prevents invalid dates like 69/420/360
|
|
dt = datetime.datetime(year=year, month=month, day=day)
|
|
|
|
# Assume no one in the Discord is more than 5 years younger, or 10 years older
|
|
# (which are also virtually impossible, but just to be sure)
|
|
if year >= timeFormatters.dateTimeNow().year - 15 or year < 1990:
|
|
raise ValueError
|
|
|
|
except ValueError:
|
|
return await ctx.send("Dit is geen geldige datum.")
|
|
|
|
# A member was tagged, check if I did it
|
|
if member is not None:
|
|
if str(ctx.author.id) != str(constants.myId):
|
|
return await ctx.send("Je kan andere mensen hun verjaardag niet instellen, {}.".format(ctx.author.display_name))
|
|
else:
|
|
birthdays.add_user(member.id, day, month, year)
|
|
return await ctx.message.add_reaction("✅")
|
|
|
|
# Birthday is already added
|
|
if birthdays.get_user(ctx.author.id) and str(ctx.author.id) != constants.myId:
|
|
return await ctx.send("Je verjaardag zit al in de database.")
|
|
|
|
# Add into the db
|
|
birthdays.add_user(ctx.author.id, day, month, year)
|
|
return await ctx.send("Je verjaardag is toegevoegd aan de database.")
|
|
|
|
|
|
def setup(client):
|
|
client.add_cog(Birthdays(client))
|