Source code for byte_bot.byte.views.forums
"""Discord UI views used in forums."""
from discord import ButtonStyle, Interaction, Member
from discord.ext.commands import Bot
from discord.ui import Button, View, button
from byte_bot.byte.lib.log import get_logger
from byte_bot.server.domain.guilds.dependencies import provides_guilds_service
from byte_bot.server.lib.db import config
__all__ = ("HelpThreadView",)
logger = get_logger()
[docs]
class HelpThreadView(View):
"""View for the help thread."""
[docs]
def __init__(self, author: Member, guild_id: int, bot: Bot, *args: list, **kwargs: dict) -> None:
"""Initialize the view."""
super().__init__(*args, **kwargs)
self.author = author
self.bot = bot
self.guild_id = guild_id
[docs]
async def setup(self) -> None:
"""Asynchronously setup guild details and add button.
.. todo:: Think about this more - If we plan on decoupling this
should be a call to an endpoint like we do in ``byte.bot.Byte.on_guild_join``.
"""
# noinspection PyBroadException
try:
async with config.get_session() as session:
guilds_service = await anext(provides_guilds_service(db_session=session))
guild_settings = await guilds_service.get(self.guild_id, id_attribute="guild_id")
if guild_settings and guild_settings.github_config:
guild_repo = guild_settings.github_config.github_repository
self.add_item(
Button(label="Open GitHub Issue", style=ButtonStyle.blurple, url=f"{guild_repo}/new/choose")
)
else:
logger.warning("no github configuration found for guild %s", self.guild_id)
except Exception:
logger.exception("failed to setup view for guild %s", self.guild_id)
[docs]
async def delete_interaction_check(self, interaction: Interaction) -> bool:
"""Check if the user is the author or an admin.
Args:
interaction (Interaction): Interaction object.
Returns:
bool: True if the user is the author or an admin, False otherwise.
"""
return interaction.user == self.author or (
hasattr(interaction.user, "guild_permissions") and interaction.user.guild_permissions.administrator # type: ignore[attr-defined]
)
[docs]
@button(label="Solve", style=ButtonStyle.green, custom_id="solve_button")
async def solve_button_callback(self, interaction: Interaction, button: Button) -> None: # noqa: ARG002
"""Mark the thread as solved.
Args:
interaction: Interaction object.
button: Button object.
"""
await interaction.response.defer()
if interaction.message:
ctx = await self.bot.get_context(interaction.message)
solve_command = self.bot.get_command("solve")
if solve_command is not None:
ctx.command = solve_command
ctx.invoked_with = "solve"
ctx.args.append(ctx)
logger.info(
"invoking solve command for %s by %s on thread %s",
ctx.channel,
interaction.user,
interaction.channel,
)
# noinspection PyBroadException
try:
await solve_command.invoke(ctx)
await interaction.followup.send("Marked as solved and closed the help forum!", ephemeral=True)
except Exception:
logger.exception("failed to invoke solve command")
await interaction.followup.send("Failed to mark as solved. Please try again.", ephemeral=True)
[docs]
@button(label="Remove", style=ButtonStyle.red, custom_id="remove_button")
async def remove_button_callback(self, interaction: Interaction, button: Button) -> None: # noqa: ARG002
"""Remove the view and embed.
Args:
interaction: Interaction object.
button: Button object.
"""
if interaction.message:
content = interaction.message.content or "\u200b"
logger.info("removing view for %s by %s", interaction.channel, interaction.user)
await interaction.message.edit(content=content, embed=None, view=None)