diff options
| author | 2021-10-19 21:43:53 -0400 | |
|---|---|---|
| committer | 2022-02-09 18:13:36 -0500 | |
| commit | a7d00f8de9a5cfa2b9c76f1a2b39ac861787e24e (patch) | |
| tree | df7cf2ca2f99c972c1254567ac3502e37f5f6f7c /bot/exts/events/trivianight/_questions.py | |
| parent | finished basic question interface (diff) | |
add trivianight structure in the main cog
Diffstat (limited to 'bot/exts/events/trivianight/_questions.py')
| -rw-r--r-- | bot/exts/events/trivianight/_questions.py | 116 | 
1 files changed, 116 insertions, 0 deletions
| diff --git a/bot/exts/events/trivianight/_questions.py b/bot/exts/events/trivianight/_questions.py new file mode 100644 index 00000000..ef56ee81 --- /dev/null +++ b/bot/exts/events/trivianight/_questions.py @@ -0,0 +1,116 @@ +from random import choice +from time import perf_counter + +from discord import ButtonStyle, Embed, Interaction +from discord.ui import Button, View + +from bot.constants import Colours, NEGATIVE_REPLIES +from ._scoreboard import Scoreboard + + +class QuestionButton(Button): +    """Button subclass for the options of the questions.""" + +    def __init__(self, label: str): +        self._time = perf_counter() +        self.users_picked = {} +        super().__init__(label=label, style=ButtonStyle.green) + +    def answer(self, label: str) -> dict: +        """Returns the dictionary of the users who picked the answer only if it was correct.""" +        return self.users_picked if label == self.label else {} + +    async def callback(self, interaction: Interaction) -> None: +        """When a user interacts with the button, this will be called.""" +        if interaction.user.id not in self.users_picked.keys(): +            self.users_picked[interaction.user.id] = [self.label, 1, perf_counter() - self._time] +        elif self.users_picked[interaction.user.id][1] < 3: +            self.users_picked[interaction.user.id] = [ +                self.label, self.users_picked[interaction.user.id][0] + 1, perf_counter() - self._time +            ] +        else: +            await interaction.response.send_message( +                embed=Embed( +                    title=choice(NEGATIVE_REPLIES), +                    description="You've already changed your answer more than once!", +                    color=Colours.soft_red +                ), +                ephemeral=True +            ) + + +class QuestionView(View): +    """View for the questions.""" + +    def __init__(self): +        self.current_question = {} + +    def create_current_question(self) -> Embed: +        """Helper function to create the embed for the current question.""" +        question_embed = Embed( +            title=f"Question {self.current_question['number']}", +            description=self.current_question["description"], +            color=Colours.python_yellow +        ) +        for label, answer in zip(("A", "B", "C", "D"), self.current_question["answers"]): +            question_embed.add_field(name=label, value=answer, inline=False) + +        self.buttons = [QuestionButton(label) for label in ("A", "B", "C", "D")] +        for button in self.buttons: +            self.add_item(button) +        return question_embed + +    def end_question(self) -> tuple[dict, Embed]: +        """Returns the dictionaries from the corresponding buttons for those who got it correct.""" +        labels = ("A", "B", "C", "D") +        label = labels[self.current_question["correct"].index(self.current_question["answers"])] +        return_dict = {} +        for button in self.buttons: +            return_dict.update(button.answer(label)) +            self.remove_item(button) + +        answer_embed = Embed( +            title=f"The correct answer for Question {self.current_question['number']} was", +            color=Colours.grass_green +        ) +        answer_embed.add_field( +            name=label, +            value=self.current_question["correct"].index(self.current_question["answers"]), +            inline=False +        ) + +        return return_dict, answer_embed + + +class Questions: +    """An interface to use from the TriviaNight cog for questions.""" + +    def __init__(self, scoreboard: Scoreboard): +        self.scoreboard = scoreboard +        self.view = QuestionView() +        self.questions = [] +        self._ptr = -1 + +    def set_questions(self, questions: list) -> None: +        """Setting `self.questions` dynamically via a function to set it.""" +        self.questions = questions + +    def next_question(self) -> None: +        """Advances to the next question.""" +        self._ptr += 1 +        if self._ptr < len(self.questions): +            self.questions[self._ptr]["visited"] = True +            self.view.current_question = self.questions[self._ptr] + +    def current_question(self) -> tuple[Embed, QuestionView]: +        """Returns an embed entailing the current question as an embed with a view.""" +        return self.view.create_current_question(), self.view + +    def end_question(self) -> None: +        """Terminates answering of the question and displays the correct answer.""" +        scores, answer_embed = self.view.end_question() +        for user, score in scores.items(): +            self.scoreboard[f"points: {user}"] = score[1] +            self.scoreboard[f"speed: {user}"] = score[2] + +        return answer_embed | 
