diff --git a/.env.template b/.env.template new file mode 100644 index 0000000000000000000000000000000000000000..f26bd8ec99a471d8e5befb9a05eefaa998c2cae9 --- /dev/null +++ b/.env.template @@ -0,0 +1,4 @@ +# General +DISCORD_TOKEN=<Bot Token you can get at https://discord.com/developers/applications/> +DISCORD_ACTIVITY=<What should be shown, Bot is playing right now> +DISCORD_DATE_TIME_FORMAT=<Date and time format used for commands like %d.%m.%Y %H:%M> \ No newline at end of file diff --git a/.gitignore b/.gitignore old mode 100644 new mode 100755 diff --git a/.idea/.gitignore b/.idea/.gitignore old mode 100644 new mode 100755 diff --git a/.idea/inspectionProfiles/Project_Default.xml b/.idea/inspectionProfiles/Project_Default.xml old mode 100644 new mode 100755 diff --git a/.idea/inspectionProfiles/profiles_settings.xml b/.idea/inspectionProfiles/profiles_settings.xml old mode 100644 new mode 100755 diff --git a/.idea/misc.xml b/.idea/misc.xml old mode 100644 new mode 100755 diff --git a/.idea/modules.xml b/.idea/modules.xml old mode 100644 new mode 100755 diff --git a/.idea/vcs.xml b/.idea/vcs.xml old mode 100644 new mode 100755 diff --git a/.idea/xanathar.iml b/.idea/xanathar.iml index 74d515a027de98657e9d3d5f0f1831882fd81374..f9707ccca5ad0d6351a37cb3b72e63139204afdf 100644 --- a/.idea/xanathar.iml +++ b/.idea/xanathar.iml @@ -4,7 +4,7 @@ <content url="file://$MODULE_DIR$"> <excludeFolder url="file://$MODULE_DIR$/venv" /> </content> - <orderEntry type="inheritedJdk" /> + <orderEntry type="jdk" jdkName="Python 3.10 (xanathar)" jdkType="Python SDK" /> <orderEntry type="sourceFolder" forTests="false" /> </component> </module> \ No newline at end of file diff --git a/cogs/appointments.py b/cogs/appointments.py old mode 100644 new mode 100755 index 44f537d28c4e04858d58f71bf1c516a8987428fa..a3737aae970ca6f1230dd15b9d10665358f2e67b --- a/cogs/appointments.py +++ b/cogs/appointments.py @@ -5,7 +5,7 @@ import json import os import uuid -import discord +from discord import app_commands, errors, Embed, File, Interaction from discord.ext import tasks, commands import utils @@ -52,7 +52,8 @@ def get_ics_file(title, date_time, reminder, recurring): return ics_file -class Appointments(commands.Cog): +@app_commands.guild_only() +class Appointments(commands.GroupCog, name="appointments", description="Handle Appointments in Channels"): def __init__(self, bot): self.bot = bot self.fmt = os.getenv("DISCORD_DATE_TIME_FORMAT") @@ -84,15 +85,15 @@ class Appointments(commands.Cog): message = await channel.fetch_message(int(message_id)) reactions = message.reactions diff = int(round(((date_time - now).total_seconds() / 60), 0)) - answer = f"Benachrichtigung!\nDer Termin \"{appointment['title']}\" ist " + answer = f"Benachrichtigung!\nDer Termin \"{appointment['title']}\" startet " if appointment["reminder"] > 0 and diff > 0: - answer += f"startet <t:{int(date_time.timestamp())}:R>." + answer += f"<t:{int(date_time.timestamp())}:R>." if (reminder := appointment.get("reminder")) and appointment.get("recurring"): - appointment["original_reminder"] = str(reminder) + appointment["original_reminder"] = reminder appointment["reminder"] = 0 else: - answer += f"jetzt fällig. :loudspeaker: " + answer += f"jetzt!!! :loudspeaker: " delete.append(message_id) answer += f"\n" @@ -106,7 +107,7 @@ class Appointments(commands.Cog): if str(message.id) in delete: await message.delete() - except discord.errors.NotFound: + except errors.NotFound: delete.append(message_id) if len(delete) > 0: @@ -125,7 +126,7 @@ class Appointments(commands.Cog): await self.add_appointment(channel, channel_appointment["author_id"], splitted_new_date_time_str[0], splitted_new_date_time_str[1], - str(reminder), + reminder, channel_appointment["title"], channel_appointment["recurring"]) channel_appointments.pop(key) @@ -135,9 +136,16 @@ class Appointments(commands.Cog): async def before_timer(self): await asyncio.sleep(60 - datetime.datetime.now().second) - @commands.command(name="add-appointment") - async def cmd_add_appointment(self, ctx, date, time, reminder, title, recurring: int = None): - await self.add_appointment(ctx.channel, ctx.author.id, date, time, reminder, title, recurring) + @app_commands.command(name="add", description="Füge dem Kanal einen neuen Termin hinzu.") + @app_commands.describe(date="Tag des Termins", time="Uhrzeit des Termins", + reminder="Wie viele Minuten bevor der Termin startet, soll eine Erinnerung verschickt werden?", + title="Titel des Termins", + recurring="In welchem Intervall (in Tagen) soll der Termin wiederholt werden?") + async def cmd_add_appointment(self, interaction: Interaction, date: str, time: str, reminder: int, title: str, + recurring: int = None): + """ Add an appointment to a channel """ + await self.add_appointment(interaction.channel, interaction.user.id, date, time, reminder, title, recurring) + await interaction.response.send_message("Termin erfolgreich erstellt!", ephemeral=True) async def add_appointment(self, channel, author_id, date, time, reminder, title, recurring: int = None): """ Add appointment to a channel """ @@ -148,17 +156,11 @@ class Appointments(commands.Cog): await channel.send("Fehler! Ungültiges Datums und/oder Zeit Format!") return - if not utils.is_valid_time(reminder): - await channel.send("Fehler! Benachrichtigung in ungültigem Format!") - return - else: - reminder = utils.to_minutes(reminder) - - embed = discord.Embed(title="Neuer Termin hinzugefügt!", - description=f"Wenn du eine Benachrichtigung zum Beginn des Termins" - f"{f', sowie {reminder} Minuten vorher, ' if reminder > 0 else f''} " - f"erhalten möchtest, reagiere mit :thumbsup: auf diese Nachricht.", - color=19607) + embed = Embed(title="Neuer Termin hinzugefügt!", + description=f"Wenn du eine Benachrichtigung zum Beginn des Termins" + f"{f', sowie {reminder} Minuten vorher, ' if reminder > 0 else f''} " + f"erhalten möchtest, reagiere mit :thumbsup: auf diese Nachricht.", + color=19607) embed.add_field(name="Titel", value=title, inline=False) embed.add_field(name="Startzeitpunkt", value=f"{date_time.strftime(self.fmt)}", inline=False) @@ -167,8 +169,8 @@ class Appointments(commands.Cog): if recurring: embed.add_field(name="Wiederholung", value=f"Alle {recurring} Tage", inline=False) - message = await channel.send(embed=embed, file=discord.File(get_ics_file(title, date_time, reminder, recurring), - filename=f"{title}.ics")) + message = await channel.send(embed=embed, file=File(get_ics_file(title, date_time, reminder, recurring), + filename=f"{title}.ics")) await message.add_reaction("ðŸ‘") await message.add_reaction("🗑ï¸") @@ -181,21 +183,22 @@ class Appointments(commands.Cog): self.save_appointments() - @commands.command(name="appointments") - async def cmd_appointments(self, ctx): + @app_commands.command(name="list", description="Listet alle Termine dieses Kanals auf.") + @app_commands.describe(show_all="Zeige die Liste für alle an.") + async def cmd_appointments_list(self, interaction: Interaction, show_all: bool = False): """ List (and link) all Appointments in the current channel """ - if str(ctx.channel.id) in self.appointments: - channel_appointments = self.appointments.get(str(ctx.channel.id)) + if str(interaction.channel.id) in self.appointments: + channel_appointments = self.appointments.get(str(interaction.channel.id)) answer = f'Termine dieses Channels:\n' delete = [] for message_id, appointment in channel_appointments.items(): try: - message = await ctx.channel.fetch_message(int(message_id)) + message = await interaction.channel.fetch_message(int(message_id)) answer += f'{appointment["date_time"]}: {appointment["title"]} => ' \ f'{message.jump_url}\n' - except discord.errors.NotFound: + except errors.NotFound: delete.append(message_id) if len(delete) > 0: @@ -203,9 +206,10 @@ class Appointments(commands.Cog): channel_appointments.pop(key) self.save_appointments() - await ctx.channel.send(answer) + await interaction.response.send_message(answer, ephemeral=not show_all) else: - await ctx.send("Für diesen Channel existieren derzeit keine Termine") + await interaction.response.send_message("Für diesen Channel existieren derzeit keine Termine", + ephemeral=not show_all) def save_appointments(self): appointments_file = open(self.app_file, mode='w') @@ -234,3 +238,7 @@ class Appointments(commands.Cog): message = await channel.fetch_message(payload.message_id) if len(message.embeds) > 0 and message.embeds[0].title == "Neuer Termin hinzugefügt!": await self.handle_reactions(payload) + + +async def setup(bot: commands.Bot) -> None: + await bot.add_cog(Appointments(bot)) diff --git a/cogs/components/poll/poll.py b/cogs/components/poll/poll.py old mode 100644 new mode 100755 diff --git a/cogs/polls.py b/cogs/polls.py old mode 100644 new mode 100755 index bb0b4b15017f3aac3feef34593b34ad4db36e7e1..86557d7e9e7445c465b7790cb14896ac39b2d6aa --- a/cogs/polls.py +++ b/cogs/polls.py @@ -11,7 +11,7 @@ class Polls(commands.Cog): @commands.group(name="poll", pass_context=True, invoke_without_command=True) async def cmd_poll(self, ctx, question, *answers): - """ Create poll """ + """ Create a new poll """ await Poll(self.bot, question, list(answers), ctx.author.id).send_poll(ctx) @@ -42,3 +42,7 @@ class Polls(commands.Cog): await poll.delete_poll() else: await poll.close_poll() + + +async def setup(bot: commands.Bot) -> None: + await bot.add_cog(Polls(bot)) diff --git a/requirements.txt b/requirements.txt old mode 100644 new mode 100755 index a8fc1564a2e0f1b8e6d02119c6b21a980e1c2068..8bd5af4977fad55c7e8471ac6b2357e12f1fc806 --- a/requirements.txt +++ b/requirements.txt @@ -3,7 +3,7 @@ aiosignal==1.2.0 async-timeout==4.0.2 attrs==21.4.0 charset-normalizer==2.1.0 -discord.py @ git+https://github.com/Rapptz/discord.py.git@dc50736bfc3340d7b999d9f165808f8dcb8f1a60 +discord.py emoji==2.0.0 frozenlist==1.3.0 idna==3.3 diff --git a/utils.py b/utils.py old mode 100644 new mode 100755 index 84f4309034282c4ae07ce5115895009119673686..294e10fbadc8f8517668f1459c48d4039f082dab --- a/utils.py +++ b/utils.py @@ -43,11 +43,3 @@ def date_to_string(date: datetime): def date_from_string(date: str): return datetime.strptime(date, DATE_TIME_FMT) - - -async def files_from_attachments(attachments): - files = [] - for attachment in attachments: - files.append(await attachment.to_file(spoiler=attachment.is_spoiler())) - - return files diff --git a/xanathar.py b/xanathar.py old mode 100644 new mode 100755 index f28e9155598f256884f8678d1b6f5aa2b9b3df99..5b73007894e9e13ba29ac83c6b36ffa011e82cd3 --- a/xanathar.py +++ b/xanathar.py @@ -1,25 +1,32 @@ import os +from typing import List import discord from discord.ext import commands from dotenv import load_dotenv -from cogs import polls, appointments - # .env file is necessary in the same directory, that contains several strings. load_dotenv() TOKEN = os.getenv('DISCORD_TOKEN') ACTIVITY = os.getenv('DISCORD_ACTIVITY') intents = discord.Intents.all() -bot = commands.Bot(command_prefix='!', help_command=None, activity=discord.Game(ACTIVITY), intents=intents) +help_command = commands.DefaultHelpCommand(dm_help=True) +extensions = ["cogs.appointments", "cogs.polls"] + + +class Xanathar(commands.Bot): + def __init__(self, *args, initial_extensions: List[str], **kwargs): + super().__init__(*args, **kwargs) + self.initial_extensions = initial_extensions + async def setup_hook(self) -> None: + for extension in self.initial_extensions: + await self.load_extension(extension) + await self.tree.sync() -@bot.event -async def on_ready(): - print("Client started!") - await bot.add_cog(polls.Polls(bot)) - await bot.add_cog(appointments.Appointments(bot)) +bot = Xanathar(command_prefix='!', help_command=help_command, activity=discord.Game(ACTIVITY), intents=intents, + initial_extensions=extensions) bot.run(TOKEN)