mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-25 10:32:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			207 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| The MIT License (MIT)
 | |
| 
 | |
| Copyright (c) 2015-present Rapptz
 | |
| 
 | |
| Permission is hereby granted, free of charge, to any person obtaining a
 | |
| copy of this software and associated documentation files (the "Software"),
 | |
| to deal in the Software without restriction, including without limitation
 | |
| the rights to use, copy, modify, merge, publish, distribute, sublicense,
 | |
| and/or sell copies of the Software, and to permit persons to whom the
 | |
| Software is furnished to do so, subject to the following conditions:
 | |
| 
 | |
| The above copyright notice and this permission notice shall be included in
 | |
| all copies or substantial portions of the Software.
 | |
| 
 | |
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
 | |
| OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 | |
| FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | |
| DEALINGS IN THE SOFTWARE.
 | |
| """
 | |
| 
 | |
| from __future__ import annotations
 | |
| 
 | |
| import asyncio
 | |
| import logging
 | |
| import os
 | |
| import sys
 | |
| import traceback
 | |
| from copy import deepcopy
 | |
| from typing import TYPE_CHECKING, Any, Dict, Optional, Sequence, ClassVar, List
 | |
| 
 | |
| from ..utils import MISSING, find
 | |
| from .item import Item
 | |
| from .view import View
 | |
| 
 | |
| if TYPE_CHECKING:
 | |
|     from typing_extensions import Self
 | |
| 
 | |
|     from ..interactions import Interaction
 | |
|     from ..types.interactions import ModalSubmitComponentInteractionData as ModalSubmitComponentInteractionDataPayload
 | |
| 
 | |
| 
 | |
| # fmt: off
 | |
| __all__ = (
 | |
|     'Modal',
 | |
| )
 | |
| # fmt: on
 | |
| 
 | |
| 
 | |
| _log = logging.getLogger(__name__)
 | |
| 
 | |
| 
 | |
| class Modal(View):
 | |
|     """Represents a UI modal.
 | |
| 
 | |
|     This object must be inherited to create a modal popup window within discord.
 | |
| 
 | |
|     .. versionadded:: 2.0
 | |
| 
 | |
|     Examples
 | |
|     ----------
 | |
| 
 | |
|     .. code-block:: python3
 | |
| 
 | |
|         from discord import ui
 | |
| 
 | |
|         class Questionnaire(ui.Modal, title='Questionnaire Response'):
 | |
|             name = ui.TextInput(label='Name')
 | |
|             answer = ui.TextInput(label='Answer', style=discord.TextStyle.paragraph)
 | |
| 
 | |
|             async def on_submit(self, interaction: discord.Interaction):
 | |
|                 await interaction.response.send_message(f'Thanks for your response, {self.name}!', ephemeral=True)
 | |
| 
 | |
|     Parameters
 | |
|     -----------
 | |
|     title: :class:`str`
 | |
|         The title of the modal.
 | |
|     timeout: Optional[:class:`float`]
 | |
|         Timeout in seconds from last interaction with the UI before no longer accepting input.
 | |
|         If ``None`` then there is no timeout.
 | |
|     custom_id: :class:`str`
 | |
|         The ID of the modal that gets received during an interaction.
 | |
|         If not given then one is generated for you.
 | |
| 
 | |
|     Attributes
 | |
|     ------------
 | |
|     title: :class:`str`
 | |
|         The title of the modal.
 | |
|     children: List[:class:`Item`]
 | |
|         The list of children attached to this view.
 | |
|     custom_id: :class:`str`
 | |
|         The ID of the modal that gets received during an interaction.
 | |
|     """
 | |
| 
 | |
|     if TYPE_CHECKING:
 | |
|         title: str
 | |
| 
 | |
|     __discord_ui_modal__ = True
 | |
|     __modal_children_items__: ClassVar[Dict[str, Item[Self]]] = {}
 | |
| 
 | |
|     def __init_subclass__(cls, *, title: str = MISSING) -> None:
 | |
|         if title is not MISSING:
 | |
|             cls.title = title
 | |
| 
 | |
|         children = {}
 | |
|         for base in reversed(cls.__mro__):
 | |
|             for name, member in base.__dict__.items():
 | |
|                 if isinstance(member, Item):
 | |
|                     children[name] = member
 | |
| 
 | |
|         cls.__modal_children_items__ = children
 | |
| 
 | |
|     def _init_children(self) -> List[Item]:
 | |
|         children = []
 | |
|         for name, item in self.__modal_children_items__.items():
 | |
|             item = deepcopy(item)
 | |
|             setattr(self, name, item)
 | |
|             item._view = self
 | |
|             children.append(item)
 | |
|         return children
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         *,
 | |
|         title: str = MISSING,
 | |
|         timeout: Optional[float] = None,
 | |
|         custom_id: str = MISSING,
 | |
|     ) -> None:
 | |
|         if title is MISSING and getattr(self, 'title', MISSING) is MISSING:
 | |
|             raise ValueError('Modal must have a title')
 | |
|         elif title is not MISSING:
 | |
|             self.title = title
 | |
|         self.custom_id: str = os.urandom(16).hex() if custom_id is MISSING else custom_id
 | |
| 
 | |
|         super().__init__(timeout=timeout)
 | |
| 
 | |
|     async def on_submit(self, interaction: Interaction) -> None:
 | |
|         """|coro|
 | |
| 
 | |
|         Called when the modal is submitted.
 | |
| 
 | |
|         Parameters
 | |
|         -----------
 | |
|         interaction: :class:`.Interaction`
 | |
|             The interaction that submitted this modal.
 | |
|         """
 | |
|         pass
 | |
| 
 | |
|     async def on_error(self, error: Exception, interaction: Interaction) -> None:
 | |
|         """|coro|
 | |
| 
 | |
|         A callback that is called when :meth:`on_submit`
 | |
|         fails with an error.
 | |
| 
 | |
|         The default implementation prints the traceback to stderr.
 | |
| 
 | |
|         Parameters
 | |
|         -----------
 | |
|         error: :class:`Exception`
 | |
|             The exception that was raised.
 | |
|         interaction: :class:`~discord.Interaction`
 | |
|             The interaction that led to the failure.
 | |
|         """
 | |
|         print(f'Ignoring exception in modal {self}:', file=sys.stderr)
 | |
|         traceback.print_exception(error.__class__, error, error.__traceback__, file=sys.stderr)
 | |
| 
 | |
|     def _refresh(self, components: Sequence[ModalSubmitComponentInteractionDataPayload]) -> None:
 | |
|         for component in components:
 | |
|             if component['type'] == 1:
 | |
|                 self._refresh(component['components'])
 | |
|             else:
 | |
|                 item = find(lambda i: i.custom_id == component['custom_id'], self.children)  # type: ignore
 | |
|                 if item is None:
 | |
|                     _log.debug("Modal interaction referencing unknown item custom_id %s. Discarding", component['custom_id'])
 | |
|                     continue
 | |
|                 item._refresh_state(component)  # type: ignore
 | |
| 
 | |
|     async def _scheduled_task(self, interaction: Interaction):
 | |
|         try:
 | |
|             self._refresh_timeout()
 | |
|             allow = await self.interaction_check(interaction)
 | |
|             if not allow:
 | |
|                 return
 | |
| 
 | |
|             await self.on_submit(interaction)
 | |
|         except Exception as e:
 | |
|             return await self.on_error(e, interaction)
 | |
|         else:
 | |
|             # No error, so assume this will always happen
 | |
|             # In the future, maybe this will require checking if we set an error response.
 | |
|             self.stop()
 | |
| 
 | |
|     def _dispatch_submit(self, interaction: Interaction) -> None:
 | |
|         asyncio.create_task(self._scheduled_task(interaction), name=f'discord-ui-modal-dispatch-{self.id}')
 | |
| 
 | |
|     def to_dict(self) -> Dict[str, Any]:
 | |
|         payload = {
 | |
|             'custom_id': self.custom_id,
 | |
|             'title': self.title,
 | |
|             'components': self.to_components(),
 | |
|         }
 | |
| 
 | |
|         return payload
 |