mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-25 02:23:04 +00:00 
			
		
		
		
	Add typings for message related classes
This commit is contained in:
		| @@ -22,10 +22,14 @@ FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | |||||||
| DEALINGS IN THE SOFTWARE. | DEALINGS IN THE SOFTWARE. | ||||||
| """ | """ | ||||||
|  |  | ||||||
|  | from __future__ import annotations | ||||||
|  |  | ||||||
| import asyncio | import asyncio | ||||||
| import datetime | import datetime | ||||||
| import re | import re | ||||||
| import io | import io | ||||||
|  | from os import PathLike | ||||||
|  | from typing import TYPE_CHECKING, Union, List, Optional, Any, Callable, Tuple, ClassVar | ||||||
|  |  | ||||||
| from . import utils | from . import utils | ||||||
| from .reaction import Reaction | from .reaction import Reaction | ||||||
| @@ -42,6 +46,26 @@ from .guild import Guild | |||||||
| from .mixins import Hashable | from .mixins import Hashable | ||||||
| from .sticker import Sticker | from .sticker import Sticker | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |     from .types.message import ( | ||||||
|  |         Message as MessagePayload, | ||||||
|  |         Attachment as AttachmentPayload, | ||||||
|  |         MessageReference as MessageReferencePayload, | ||||||
|  |         MessageApplication as MessageApplicationPayload, | ||||||
|  |         MessageActivity as MessageActivityPayload, | ||||||
|  |         Reaction as ReactionPayload, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     from .types.member import Member as MemberPayload | ||||||
|  |     from .types.user import User as UserPayload | ||||||
|  |     from .types.embed import Embed as EmbedPayload | ||||||
|  |     from .abc import Snowflake | ||||||
|  |     from .abc import GuildChannel, PrivateChannel, Messageable | ||||||
|  |     from .state import ConnectionState | ||||||
|  |     from .channel import TextChannel, GroupChannel, DMChannel | ||||||
|  |  | ||||||
|  |     EmojiInputType = Union[Emoji, PartialEmoji, str] | ||||||
|  |  | ||||||
| __all__ = ( | __all__ = ( | ||||||
|     'Attachment', |     'Attachment', | ||||||
|     'Message', |     'Message', | ||||||
| @@ -116,7 +140,7 @@ class Attachment(Hashable): | |||||||
|  |  | ||||||
|     __slots__ = ('id', 'size', 'height', 'width', 'filename', 'url', 'proxy_url', '_http', 'content_type') |     __slots__ = ('id', 'size', 'height', 'width', 'filename', 'url', 'proxy_url', '_http', 'content_type') | ||||||
|  |  | ||||||
|     def __init__(self, *, data, state): |     def __init__(self, *, data: AttachmentPayload, state: ConnectionState): | ||||||
|         self.id = int(data['id']) |         self.id = int(data['id']) | ||||||
|         self.size = data['size'] |         self.size = data['size'] | ||||||
|         self.height = data.get('height') |         self.height = data.get('height') | ||||||
| @@ -127,17 +151,17 @@ class Attachment(Hashable): | |||||||
|         self._http = state.http |         self._http = state.http | ||||||
|         self.content_type = data.get('content_type') |         self.content_type = data.get('content_type') | ||||||
|  |  | ||||||
|     def is_spoiler(self): |     def is_spoiler(self) -> bool: | ||||||
|         """:class:`bool`: Whether this attachment contains a spoiler.""" |         """:class:`bool`: Whether this attachment contains a spoiler.""" | ||||||
|         return self.filename.startswith('SPOILER_') |         return self.filename.startswith('SPOILER_') | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self) -> str: | ||||||
|         return f'<Attachment id={self.id} filename={self.filename!r} url={self.url!r}>' |         return f'<Attachment id={self.id} filename={self.filename!r} url={self.url!r}>' | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self) -> str: | ||||||
|         return self.url or '' |         return self.url or '' | ||||||
|  |  | ||||||
|     async def save(self, fp, *, seek_begin=True, use_cached=False): |     async def save(self, fp: Union[io.BufferedIOBase, PathLike], *, seek_begin: bool = True, use_cached: bool = False) -> int: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Saves this attachment into a file-like object. |         Saves this attachment into a file-like object. | ||||||
| @@ -181,7 +205,7 @@ class Attachment(Hashable): | |||||||
|             with open(fp, 'wb') as f: |             with open(fp, 'wb') as f: | ||||||
|                 return f.write(data) |                 return f.write(data) | ||||||
|  |  | ||||||
|     async def read(self, *, use_cached=False): |     async def read(self, *, use_cached: bool = False) -> bytes: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Retrieves the content of this attachment as a :class:`bytes` object. |         Retrieves the content of this attachment as a :class:`bytes` object. | ||||||
| @@ -216,7 +240,7 @@ class Attachment(Hashable): | |||||||
|         data = await self._http.get_from_cdn(url) |         data = await self._http.get_from_cdn(url) | ||||||
|         return data |         return data | ||||||
|  |  | ||||||
|     async def to_file(self, *, use_cached=False, spoiler=False): |     async def to_file(self, *, use_cached: bool = False, spoiler: bool = False) -> File: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Converts the attachment into a :class:`File` suitable for sending via |         Converts the attachment into a :class:`File` suitable for sending via | ||||||
| @@ -258,8 +282,8 @@ class Attachment(Hashable): | |||||||
|         data = await self.read(use_cached=use_cached) |         data = await self.read(use_cached=use_cached) | ||||||
|         return File(io.BytesIO(data), filename=self.filename, spoiler=spoiler) |         return File(io.BytesIO(data), filename=self.filename, spoiler=spoiler) | ||||||
|  |  | ||||||
|     def to_dict(self): |     def to_dict(self) -> AttachmentPayload: | ||||||
|         result = { |         result: AttachmentPayload = { | ||||||
|             'filename': self.filename, |             'filename': self.filename, | ||||||
|             'id': self.id, |             'id': self.id, | ||||||
|             'proxy_url': self.proxy_url, |             'proxy_url': self.proxy_url, | ||||||
| @@ -287,24 +311,24 @@ class DeletedReferencedMessage: | |||||||
|  |  | ||||||
|     __slots__ = ('_parent') |     __slots__ = ('_parent') | ||||||
|  |  | ||||||
|     def __init__(self, parent): |     def __init__(self, parent: MessageReference): | ||||||
|         self._parent = parent |         self._parent = parent | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self) -> str: | ||||||
|         return f"<DeletedReferencedMessage id={self.id} channel_id={self.channel_id} guild_id={self.guild_id!r}>" |         return f"<DeletedReferencedMessage id={self.id} channel_id={self.channel_id} guild_id={self.guild_id!r}>" | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def id(self): |     def id(self) -> int: | ||||||
|         """:class:`int`: The message ID of the deleted referenced message.""" |         """:class:`int`: The message ID of the deleted referenced message.""" | ||||||
|         return self._parent.message_id |         return self._parent.message_id | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def channel_id(self): |     def channel_id(self) -> int: | ||||||
|         """:class:`int`: The channel ID of the deleted referenced message.""" |         """:class:`int`: The channel ID of the deleted referenced message.""" | ||||||
|         return self._parent.channel_id |         return self._parent.channel_id | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def guild_id(self): |     def guild_id(self) -> Optional[int]: | ||||||
|         """Optional[:class:`int`]: The guild ID of the deleted referenced message.""" |         """Optional[:class:`int`]: The guild ID of the deleted referenced message.""" | ||||||
|         return self._parent.guild_id |         return self._parent.guild_id | ||||||
|  |  | ||||||
| @@ -345,16 +369,16 @@ class MessageReference: | |||||||
|  |  | ||||||
|     __slots__ = ('message_id', 'channel_id', 'guild_id', 'fail_if_not_exists', 'resolved', '_state') |     __slots__ = ('message_id', 'channel_id', 'guild_id', 'fail_if_not_exists', 'resolved', '_state') | ||||||
|  |  | ||||||
|     def __init__(self, *, message_id, channel_id, guild_id=None, fail_if_not_exists=True): |     def __init__(self, *, message_id: int, channel_id: int, guild_id: Optional[int] = None, fail_if_not_exists: bool = True): | ||||||
|         self._state = None |         self._state: Optional[ConnectionState] = None | ||||||
|         self.resolved = None |         self.resolved: Optional[Union[Message, DeletedReferencedMessage]] = None | ||||||
|         self.message_id = message_id |         self.message_id = message_id | ||||||
|         self.channel_id = channel_id |         self.channel_id = channel_id | ||||||
|         self.guild_id = guild_id |         self.guild_id = guild_id | ||||||
|         self.fail_if_not_exists = fail_if_not_exists |         self.fail_if_not_exists = fail_if_not_exists | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def with_state(cls, state, data): |     def with_state(cls, state: ConnectionState, data: MessageReferencePayload) -> MessageReference: | ||||||
|         self = cls.__new__(cls) |         self = cls.__new__(cls) | ||||||
|         self.message_id = utils._get_as_snowflake(data, 'message_id') |         self.message_id = utils._get_as_snowflake(data, 'message_id') | ||||||
|         self.channel_id = int(data.pop('channel_id')) |         self.channel_id = int(data.pop('channel_id')) | ||||||
| @@ -365,7 +389,7 @@ class MessageReference: | |||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_message(cls, message, *, fail_if_not_exists=True): |     def from_message(cls, message: Message, *, fail_if_not_exists: bool = True): | ||||||
|         """Creates a :class:`MessageReference` from an existing :class:`~discord.Message`. |         """Creates a :class:`MessageReference` from an existing :class:`~discord.Message`. | ||||||
|  |  | ||||||
|         .. versionadded:: 1.6 |         .. versionadded:: 1.6 | ||||||
| @@ -390,12 +414,12 @@ class MessageReference: | |||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def cached_message(self): |     def cached_message(self) -> Optional[Message]: | ||||||
|         """Optional[:class:`~discord.Message`]: The cached message, if found in the internal message cache.""" |         """Optional[:class:`~discord.Message`]: The cached message, if found in the internal message cache.""" | ||||||
|         return self._state._get_message(self.message_id) |         return self._state and self._state._get_message(self.message_id) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def jump_url(self): |     def jump_url(self) -> str: | ||||||
|         """:class:`str`: Returns a URL that allows the client to jump to the referenced message. |         """:class:`str`: Returns a URL that allows the client to jump to the referenced message. | ||||||
|  |  | ||||||
|         .. versionadded:: 1.7 |         .. versionadded:: 1.7 | ||||||
| @@ -403,11 +427,11 @@ class MessageReference: | |||||||
|         guild_id = self.guild_id if self.guild_id is not None else '@me' |         guild_id = self.guild_id if self.guild_id is not None else '@me' | ||||||
|         return f'https://discord.com/channels/{guild_id}/{self.channel_id}/{self.message_id}' |         return f'https://discord.com/channels/{guild_id}/{self.channel_id}/{self.message_id}' | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self) -> str: | ||||||
|         return f'<MessageReference message_id={self.message_id!r} channel_id={self.channel_id!r} guild_id={self.guild_id!r}>' |         return f'<MessageReference message_id={self.message_id!r} channel_id={self.channel_id!r} guild_id={self.guild_id!r}>' | ||||||
|  |  | ||||||
|     def to_dict(self): |     def to_dict(self) -> MessageReferencePayload: | ||||||
|         result = {'message_id': self.message_id} if self.message_id is not None else {} |         result: MessageReferencePayload = {'message_id': self.message_id} if self.message_id is not None else {} | ||||||
|         result['channel_id'] = self.channel_id |         result['channel_id'] = self.channel_id | ||||||
|         if self.guild_id is not None: |         if self.guild_id is not None: | ||||||
|             result['guild_id'] = self.guild_id |             result['guild_id'] = self.guild_id | ||||||
| @@ -460,17 +484,17 @@ class Message(Hashable): | |||||||
|     type: :class:`MessageType` |     type: :class:`MessageType` | ||||||
|         The type of message. In most cases this should not be checked, but it is helpful |         The type of message. In most cases this should not be checked, but it is helpful | ||||||
|         in cases where it might be a system message for :attr:`system_content`. |         in cases where it might be a system message for :attr:`system_content`. | ||||||
|     author: :class:`abc.User` |     author: Union[:class:`Member`, :class:`abc.User`] | ||||||
|         A :class:`Member` that sent the message. If :attr:`channel` is a |         A :class:`Member` that sent the message. If :attr:`channel` is a | ||||||
|         private channel or the user has the left the guild, then it is a :class:`User` instead. |         private channel or the user has the left the guild, then it is a :class:`User` instead. | ||||||
|     content: :class:`str` |     content: :class:`str` | ||||||
|         The actual contents of the message. |         The actual contents of the message. | ||||||
|     nonce |     nonce: Union[:class:`str`, :class:`int`] | ||||||
|         The value used by the discord guild and the client to verify that the message is successfully sent. |         The value used by the discord guild and the client to verify that the message is successfully sent. | ||||||
|         This is not stored long term within Discord's servers and is only used ephemerally. |         This is not stored long term within Discord's servers and is only used ephemerally. | ||||||
|     embeds: List[:class:`Embed`] |     embeds: List[:class:`Embed`] | ||||||
|         A list of embeds the message has. |         A list of embeds the message has. | ||||||
|     channel: Union[:class:`abc.Messageable`] |     channel: Union[:class:`TextChannel`, :class:`DMChannel`, :class:`GroupChannel`] | ||||||
|         The :class:`TextChannel` that the message was sent from. |         The :class:`TextChannel` that the message was sent from. | ||||||
|         Could be a :class:`DMChannel` or :class:`GroupChannel` if it's a private message. |         Could be a :class:`DMChannel` or :class:`GroupChannel` if it's a private message. | ||||||
|     reference: Optional[:class:`~discord.MessageReference`] |     reference: Optional[:class:`~discord.MessageReference`] | ||||||
| @@ -552,7 +576,10 @@ class Message(Hashable): | |||||||
|                  '_cs_system_content', '_cs_guild', '_state', 'reactions', 'reference', |                  '_cs_system_content', '_cs_guild', '_state', 'reactions', 'reference', | ||||||
|                  'application', 'activity', 'stickers') |                  'application', 'activity', 'stickers') | ||||||
|  |  | ||||||
|     def __init__(self, *, state, channel, data): |     _HANDLERS: ClassVar[List[Tuple[str, Callable[..., None]]]] | ||||||
|  |     _CACHED_SLOTS: ClassVar[List[str]] | ||||||
|  |  | ||||||
|  |     def __init__(self, *, state: ConnectionState, channel: Union[TextChannel, DMChannel, GroupChannel], data: MessagePayload): | ||||||
|         self._state = state |         self._state = state | ||||||
|         self.id = int(data['id']) |         self.id = int(data['id']) | ||||||
|         self.webhook_id = utils._get_as_snowflake(data, 'webhook_id') |         self.webhook_id = utils._get_as_snowflake(data, 'webhook_id') | ||||||
| @@ -600,10 +627,10 @@ class Message(Hashable): | |||||||
|             except KeyError: |             except KeyError: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self) -> str: | ||||||
|         return f'<Message id={self.id} channel={self.channel!r} type={self.type!r} author={self.author!r} flags={self.flags!r}>' |         return f'<Message id={self.id} channel={self.channel!r} type={self.type!r} author={self.author!r} flags={self.flags!r}>' | ||||||
|  |  | ||||||
|     def _try_patch(self, data, key, transform=None): |     def _try_patch(self, data, key, transform=None) -> None: | ||||||
|         try: |         try: | ||||||
|             value = data[key] |             value = data[key] | ||||||
|         except KeyError: |         except KeyError: | ||||||
| @@ -614,7 +641,7 @@ class Message(Hashable): | |||||||
|             else: |             else: | ||||||
|                 setattr(self, key, transform(value)) |                 setattr(self, key, transform(value)) | ||||||
|  |  | ||||||
|     def _add_reaction(self, data, emoji, user_id): |     def _add_reaction(self, data, emoji, user_id) -> Reaction: | ||||||
|         reaction = utils.find(lambda r: r.emoji == emoji, self.reactions) |         reaction = utils.find(lambda r: r.emoji == emoji, self.reactions) | ||||||
|         is_me = data['me'] = user_id == self._state.self_id |         is_me = data['me'] = user_id == self._state.self_id | ||||||
|  |  | ||||||
| @@ -628,7 +655,7 @@ class Message(Hashable): | |||||||
|  |  | ||||||
|         return reaction |         return reaction | ||||||
|  |  | ||||||
|     def _remove_reaction(self, data, emoji, user_id): |     def _remove_reaction(self, data: ReactionPayload, emoji: EmojiInputType, user_id: int) -> Reaction: | ||||||
|         reaction = utils.find(lambda r: r.emoji == emoji, self.reactions) |         reaction = utils.find(lambda r: r.emoji == emoji, self.reactions) | ||||||
|  |  | ||||||
|         if reaction is None: |         if reaction is None: | ||||||
| @@ -647,7 +674,7 @@ class Message(Hashable): | |||||||
|  |  | ||||||
|         return reaction |         return reaction | ||||||
|  |  | ||||||
|     def _clear_emoji(self, emoji): |     def _clear_emoji(self, emoji) -> Optional[Reaction]: | ||||||
|         to_check = str(emoji) |         to_check = str(emoji) | ||||||
|         for index, reaction in enumerate(self.reactions): |         for index, reaction in enumerate(self.reactions): | ||||||
|             if str(reaction.emoji) == to_check: |             if str(reaction.emoji) == to_check: | ||||||
| @@ -679,50 +706,50 @@ class Message(Hashable): | |||||||
|             except AttributeError: |             except AttributeError: | ||||||
|                 pass |                 pass | ||||||
|  |  | ||||||
|     def _handle_edited_timestamp(self, value): |     def _handle_edited_timestamp(self, value: str) -> None: | ||||||
|         self._edited_timestamp = utils.parse_time(value) |         self._edited_timestamp = utils.parse_time(value) | ||||||
|  |  | ||||||
|     def _handle_pinned(self, value): |     def _handle_pinned(self, value: int) -> None: | ||||||
|         self.pinned = value |         self.pinned = value | ||||||
|  |  | ||||||
|     def _handle_flags(self, value): |     def _handle_flags(self, value: int) -> None: | ||||||
|         self.flags = MessageFlags._from_value(value) |         self.flags = MessageFlags._from_value(value) | ||||||
|  |  | ||||||
|     def _handle_application(self, value): |     def _handle_application(self, value: MessageApplicationPayload) -> None: | ||||||
|         self.application = value |         self.application = value | ||||||
|  |  | ||||||
|     def _handle_activity(self, value): |     def _handle_activity(self, value: MessageActivityPayload) -> None: | ||||||
|         self.activity = value |         self.activity = value | ||||||
|  |  | ||||||
|     def _handle_mention_everyone(self, value): |     def _handle_mention_everyone(self, value: bool) -> None: | ||||||
|         self.mention_everyone = value |         self.mention_everyone = value | ||||||
|  |  | ||||||
|     def _handle_tts(self, value): |     def _handle_tts(self, value: bool) -> None: | ||||||
|         self.tts = value |         self.tts = value | ||||||
|  |  | ||||||
|     def _handle_type(self, value): |     def _handle_type(self, value: int) -> None: | ||||||
|         self.type = try_enum(MessageType, value) |         self.type = try_enum(MessageType, value) | ||||||
|  |  | ||||||
|     def _handle_content(self, value): |     def _handle_content(self, value: str) -> None: | ||||||
|         self.content = value |         self.content = value | ||||||
|  |  | ||||||
|     def _handle_attachments(self, value): |     def _handle_attachments(self, value: List[AttachmentPayload]) -> None: | ||||||
|         self.attachments = [Attachment(data=a, state=self._state) for a in value] |         self.attachments = [Attachment(data=a, state=self._state) for a in value] | ||||||
|  |  | ||||||
|     def _handle_embeds(self, value): |     def _handle_embeds(self, value: List[EmbedPayload]) -> None: | ||||||
|         self.embeds = [Embed.from_dict(data) for data in value] |         self.embeds = [Embed.from_dict(data) for data in value] | ||||||
|  |  | ||||||
|     def _handle_nonce(self, value): |     def _handle_nonce(self, value: Union[str, int]) -> None: | ||||||
|         self.nonce = value |         self.nonce = value | ||||||
|  |  | ||||||
|     def _handle_author(self, author): |     def _handle_author(self, author: UserPayload) -> None: | ||||||
|         self.author = self._state.store_user(author) |         self.author = self._state.store_user(author) | ||||||
|         if isinstance(self.guild, Guild): |         if isinstance(self.guild, Guild): | ||||||
|             found = self.guild.get_member(self.author.id) |             found = self.guild.get_member(self.author.id) | ||||||
|             if found is not None: |             if found is not None: | ||||||
|                 self.author = found |                 self.author = found | ||||||
|  |  | ||||||
|     def _handle_member(self, member): |     def _handle_member(self, member: MemberPayload) -> None: | ||||||
|         # The gateway now gives us full Member objects sometimes with the following keys |         # The gateway now gives us full Member objects sometimes with the following keys | ||||||
|         # deaf, mute, joined_at, roles |         # deaf, mute, joined_at, roles | ||||||
|         # For the sake of performance I'm going to assume that the only |         # For the sake of performance I'm going to assume that the only | ||||||
| @@ -732,13 +759,13 @@ class Message(Hashable): | |||||||
|         author = self.author |         author = self.author | ||||||
|         try: |         try: | ||||||
|             # Update member reference |             # Update member reference | ||||||
|             author._update_from_message(member) |             author._update_from_message(member)  # type: ignore | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             # It's a user here |             # It's a user here | ||||||
|             # TODO: consider adding to cache here |             # TODO: consider adding to cache here | ||||||
|             self.author = Member._from_message(message=self, data=member) |             self.author = Member._from_message(message=self, data=member) | ||||||
|  |  | ||||||
|     def _handle_mentions(self, mentions): |     def _handle_mentions(self, mentions: List[UserPayload]) -> None: | ||||||
|         self.mentions = r = [] |         self.mentions = r = [] | ||||||
|         guild = self.guild |         guild = self.guild | ||||||
|         state = self._state |         state = self._state | ||||||
| @@ -754,7 +781,7 @@ class Message(Hashable): | |||||||
|             else: |             else: | ||||||
|                 r.append(Member._try_upgrade(data=mention, guild=guild, state=state)) |                 r.append(Member._try_upgrade(data=mention, guild=guild, state=state)) | ||||||
|  |  | ||||||
|     def _handle_mention_roles(self, role_mentions): |     def _handle_mention_roles(self, role_mentions: List[int]) -> None: | ||||||
|         self.role_mentions = [] |         self.role_mentions = [] | ||||||
|         if isinstance(self.guild, Guild): |         if isinstance(self.guild, Guild): | ||||||
|             for role_id in map(int, role_mentions): |             for role_id in map(int, role_mentions): | ||||||
| @@ -762,21 +789,21 @@ class Message(Hashable): | |||||||
|                 if role is not None: |                 if role is not None: | ||||||
|                     self.role_mentions.append(role) |                     self.role_mentions.append(role) | ||||||
|  |  | ||||||
|     def _rebind_channel_reference(self, new_channel): |     def _rebind_channel_reference(self, new_channel: Union[TextChannel, DMChannel, GroupChannel]) -> None: | ||||||
|         self.channel = new_channel |         self.channel = new_channel | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             del self._cs_guild |             del self._cs_guild  # type: ignore | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             pass |             pass | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_guild') |     @utils.cached_slot_property('_cs_guild') | ||||||
|     def guild(self): |     def guild(self) -> Optional[Guild]: | ||||||
|         """Optional[:class:`Guild`]: The guild that the message belongs to, if applicable.""" |         """Optional[:class:`Guild`]: The guild that the message belongs to, if applicable.""" | ||||||
|         return getattr(self.channel, 'guild', None) |         return getattr(self.channel, 'guild', None) | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_raw_mentions') |     @utils.cached_slot_property('_cs_raw_mentions') | ||||||
|     def raw_mentions(self): |     def raw_mentions(self) -> List[int]: | ||||||
|         """List[:class:`int`]: A property that returns an array of user IDs matched with |         """List[:class:`int`]: A property that returns an array of user IDs matched with | ||||||
|         the syntax of ``<@user_id>`` in the message content. |         the syntax of ``<@user_id>`` in the message content. | ||||||
|  |  | ||||||
| @@ -786,28 +813,28 @@ class Message(Hashable): | |||||||
|         return [int(x) for x in re.findall(r'<@!?([0-9]{15,20})>', self.content)] |         return [int(x) for x in re.findall(r'<@!?([0-9]{15,20})>', self.content)] | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_raw_channel_mentions') |     @utils.cached_slot_property('_cs_raw_channel_mentions') | ||||||
|     def raw_channel_mentions(self): |     def raw_channel_mentions(self) -> List[int]: | ||||||
|         """List[:class:`int`]: A property that returns an array of channel IDs matched with |         """List[:class:`int`]: A property that returns an array of channel IDs matched with | ||||||
|         the syntax of ``<#channel_id>`` in the message content. |         the syntax of ``<#channel_id>`` in the message content. | ||||||
|         """ |         """ | ||||||
|         return [int(x) for x in re.findall(r'<#([0-9]{15,20})>', self.content)] |         return [int(x) for x in re.findall(r'<#([0-9]{15,20})>', self.content)] | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_raw_role_mentions') |     @utils.cached_slot_property('_cs_raw_role_mentions') | ||||||
|     def raw_role_mentions(self): |     def raw_role_mentions(self) -> List[int]: | ||||||
|         """List[:class:`int`]: A property that returns an array of role IDs matched with |         """List[:class:`int`]: A property that returns an array of role IDs matched with | ||||||
|         the syntax of ``<@&role_id>`` in the message content. |         the syntax of ``<@&role_id>`` in the message content. | ||||||
|         """ |         """ | ||||||
|         return [int(x) for x in re.findall(r'<@&([0-9]{15,20})>', self.content)] |         return [int(x) for x in re.findall(r'<@&([0-9]{15,20})>', self.content)] | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_channel_mentions') |     @utils.cached_slot_property('_cs_channel_mentions') | ||||||
|     def channel_mentions(self): |     def channel_mentions(self) -> List[GuildChannel]: | ||||||
|         if self.guild is None: |         if self.guild is None: | ||||||
|             return [] |             return [] | ||||||
|         it = filter(None, map(self.guild.get_channel, self.raw_channel_mentions)) |         it = filter(None, map(self.guild.get_channel, self.raw_channel_mentions)) | ||||||
|         return utils._unique(it) |         return utils._unique(it) | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_clean_content') |     @utils.cached_slot_property('_cs_clean_content') | ||||||
|     def clean_content(self): |     def clean_content(self) -> str: | ||||||
|         """:class:`str`: A property that returns the content in a "cleaned up" |         """:class:`str`: A property that returns the content in a "cleaned up" | ||||||
|         manner. This basically means that mentions are transformed |         manner. This basically means that mentions are transformed | ||||||
|         into the way the client shows it. e.g. ``<#id>`` will transform |         into the way the client shows it. e.g. ``<#id>`` will transform | ||||||
| @@ -857,22 +884,22 @@ class Message(Hashable): | |||||||
|         return escape_mentions(result) |         return escape_mentions(result) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def created_at(self): |     def created_at(self) -> datetime.datetime: | ||||||
|         """:class:`datetime.datetime`: The message's creation time in UTC.""" |         """:class:`datetime.datetime`: The message's creation time in UTC.""" | ||||||
|         return utils.snowflake_time(self.id) |         return utils.snowflake_time(self.id) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def edited_at(self): |     def edited_at(self) -> Optional[datetime.datetime]: | ||||||
|         """Optional[:class:`datetime.datetime`]: An aware UTC datetime object containing the edited time of the message.""" |         """Optional[:class:`datetime.datetime`]: An aware UTC datetime object containing the edited time of the message.""" | ||||||
|         return self._edited_timestamp |         return self._edited_timestamp | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def jump_url(self): |     def jump_url(self) -> str: | ||||||
|         """:class:`str`: Returns a URL that allows the client to jump to this message.""" |         """:class:`str`: Returns a URL that allows the client to jump to this message.""" | ||||||
|         guild_id = getattr(self.guild, 'id', '@me') |         guild_id = getattr(self.guild, 'id', '@me') | ||||||
|         return f'https://discord.com/channels/{guild_id}/{self.channel.id}/{self.id}' |         return f'https://discord.com/channels/{guild_id}/{self.channel.id}/{self.id}' | ||||||
|  |  | ||||||
|     def is_system(self): |     def is_system(self) -> bool: | ||||||
|         """:class:`bool`: Whether the message is a system message. |         """:class:`bool`: Whether the message is a system message. | ||||||
|  |  | ||||||
|         .. versionadded:: 1.3 |         .. versionadded:: 1.3 | ||||||
| @@ -943,7 +970,7 @@ class Message(Hashable): | |||||||
|             return f'{self.author.name} has added {self.content} to this channel' |             return f'{self.author.name} has added {self.content} to this channel' | ||||||
|  |  | ||||||
|         if self.type is MessageType.guild_stream: |         if self.type is MessageType.guild_stream: | ||||||
|             return f'{self.author.name} is live! Now streaming {self.author.activity.name}' |             return f'{self.author.name} is live! Now streaming {self.author.activity.name}'  # type: ignore | ||||||
|  |  | ||||||
|         if self.type is MessageType.guild_discovery_disqualified: |         if self.type is MessageType.guild_discovery_disqualified: | ||||||
|             return 'This server has been removed from Server Discovery because it no longer passes all the requirements. Check Server Settings for more details.' |             return 'This server has been removed from Server Discovery because it no longer passes all the requirements. Check Server Settings for more details.' | ||||||
| @@ -960,7 +987,7 @@ class Message(Hashable): | |||||||
|         if self.type is MessageType.guild_invite_reminder: |         if self.type is MessageType.guild_invite_reminder: | ||||||
|             return 'Wondering who to invite?\nStart by inviting anyone who can help you build the server!' |             return 'Wondering who to invite?\nStart by inviting anyone who can help you build the server!' | ||||||
|  |  | ||||||
|     async def delete(self, *, delay=None): |     async def delete(self, *, delay: Optional[float] = None) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Deletes the message. |         Deletes the message. | ||||||
| @@ -988,18 +1015,18 @@ class Message(Hashable): | |||||||
|             Deleting the message failed. |             Deleting the message failed. | ||||||
|         """ |         """ | ||||||
|         if delay is not None: |         if delay is not None: | ||||||
|             async def delete(): |             async def delete(delay: float): | ||||||
|                 await asyncio.sleep(delay) |                 await asyncio.sleep(delay) | ||||||
|                 try: |                 try: | ||||||
|                     await self._state.http.delete_message(self.channel.id, self.id) |                     await self._state.http.delete_message(self.channel.id, self.id) | ||||||
|                 except HTTPException: |                 except HTTPException: | ||||||
|                     pass |                     pass | ||||||
|  |  | ||||||
|             asyncio.create_task(delete()) |             asyncio.create_task(delete(delay)) | ||||||
|         else: |         else: | ||||||
|             await self._state.http.delete_message(self.channel.id, self.id) |             await self._state.http.delete_message(self.channel.id, self.id) | ||||||
|  |  | ||||||
|     async def edit(self, **fields): |     async def edit(self, **fields: Any) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Edits the message. |         Edits the message. | ||||||
| @@ -1102,7 +1129,7 @@ class Message(Hashable): | |||||||
|         if delete_after is not None: |         if delete_after is not None: | ||||||
|             await self.delete(delay=delete_after) |             await self.delete(delay=delete_after) | ||||||
|  |  | ||||||
|     async def publish(self): |     async def publish(self) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Publishes this message to your announcement channel. |         Publishes this message to your announcement channel. | ||||||
| @@ -1120,7 +1147,7 @@ class Message(Hashable): | |||||||
|  |  | ||||||
|         await self._state.http.publish_message(self.channel.id, self.id) |         await self._state.http.publish_message(self.channel.id, self.id) | ||||||
|  |  | ||||||
|     async def pin(self, *, reason=None): |     async def pin(self, *, reason: Optional[str] = None) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Pins the message. |         Pins the message. | ||||||
| @@ -1149,7 +1176,7 @@ class Message(Hashable): | |||||||
|         await self._state.http.pin_message(self.channel.id, self.id, reason=reason) |         await self._state.http.pin_message(self.channel.id, self.id, reason=reason) | ||||||
|         self.pinned = True |         self.pinned = True | ||||||
|  |  | ||||||
|     async def unpin(self, *, reason=None): |     async def unpin(self, *, reason: Optional[str] = None) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Unpins the message. |         Unpins the message. | ||||||
| @@ -1177,7 +1204,7 @@ class Message(Hashable): | |||||||
|         await self._state.http.unpin_message(self.channel.id, self.id, reason=reason) |         await self._state.http.unpin_message(self.channel.id, self.id, reason=reason) | ||||||
|         self.pinned = False |         self.pinned = False | ||||||
|  |  | ||||||
|     async def add_reaction(self, emoji): |     async def add_reaction(self, emoji: EmojiInputType) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Add a reaction to the message. |         Add a reaction to the message. | ||||||
| @@ -1208,7 +1235,7 @@ class Message(Hashable): | |||||||
|         emoji = convert_emoji_reaction(emoji) |         emoji = convert_emoji_reaction(emoji) | ||||||
|         await self._state.http.add_reaction(self.channel.id, self.id, emoji) |         await self._state.http.add_reaction(self.channel.id, self.id, emoji) | ||||||
|  |  | ||||||
|     async def remove_reaction(self, emoji, member): |     async def remove_reaction(self, emoji: Union[EmojiInputType, Reaction], member: Snowflake) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Remove a reaction by the member from the message. |         Remove a reaction by the member from the message. | ||||||
| @@ -1247,7 +1274,7 @@ class Message(Hashable): | |||||||
|         else: |         else: | ||||||
|             await self._state.http.remove_reaction(self.channel.id, self.id, emoji, member.id) |             await self._state.http.remove_reaction(self.channel.id, self.id, emoji, member.id) | ||||||
|  |  | ||||||
|     async def clear_reaction(self, emoji): |     async def clear_reaction(self, emoji: Union[EmojiInputType, Reaction]) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Clears a specific reaction from the message. |         Clears a specific reaction from the message. | ||||||
| @@ -1278,7 +1305,7 @@ class Message(Hashable): | |||||||
|         emoji = convert_emoji_reaction(emoji) |         emoji = convert_emoji_reaction(emoji) | ||||||
|         await self._state.http.clear_single_reaction(self.channel.id, self.id, emoji) |         await self._state.http.clear_single_reaction(self.channel.id, self.id, emoji) | ||||||
|  |  | ||||||
|     async def clear_reactions(self): |     async def clear_reactions(self) -> None: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Removes all the reactions from the message. |         Removes all the reactions from the message. | ||||||
| @@ -1294,7 +1321,7 @@ class Message(Hashable): | |||||||
|         """ |         """ | ||||||
|         await self._state.http.clear_reactions(self.channel.id, self.id) |         await self._state.http.clear_reactions(self.channel.id, self.id) | ||||||
|  |  | ||||||
|     async def reply(self, content=None, **kwargs): |     async def reply(self, content: Optional[str] = None, **kwargs) -> Message: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         A shortcut method to :meth:`.abc.Messageable.send` to reply to the |         A shortcut method to :meth:`.abc.Messageable.send` to reply to the | ||||||
| @@ -1320,7 +1347,7 @@ class Message(Hashable): | |||||||
|  |  | ||||||
|         return await self.channel.send(content, reference=self, **kwargs) |         return await self.channel.send(content, reference=self, **kwargs) | ||||||
|  |  | ||||||
|     def to_reference(self, *, fail_if_not_exists=True): |     def to_reference(self, *, fail_if_not_exists: bool = True) -> MessageReference: | ||||||
|         """Creates a :class:`~discord.MessageReference` from the current message. |         """Creates a :class:`~discord.MessageReference` from the current message. | ||||||
|  |  | ||||||
|         .. versionadded:: 1.6 |         .. versionadded:: 1.6 | ||||||
| @@ -1341,8 +1368,8 @@ class Message(Hashable): | |||||||
|  |  | ||||||
|         return MessageReference.from_message(self, fail_if_not_exists=fail_if_not_exists) |         return MessageReference.from_message(self, fail_if_not_exists=fail_if_not_exists) | ||||||
|  |  | ||||||
|     def to_message_reference_dict(self): |     def to_message_reference_dict(self) -> MessageReferencePayload: | ||||||
|         data = { |         data: MessageReferencePayload = { | ||||||
|             'message_id': self.id, |             'message_id': self.id, | ||||||
|             'channel_id': self.channel.id, |             'channel_id': self.channel.id, | ||||||
|         } |         } | ||||||
| @@ -1411,7 +1438,7 @@ class PartialMessage(Hashable): | |||||||
|         'to_message_reference_dict', |         'to_message_reference_dict', | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *, channel, id): |     def __init__(self, *, channel: Union[GuildChannel, PrivateChannel], id: int): | ||||||
|         if channel.type not in (ChannelType.text, ChannelType.news, ChannelType.private): |         if channel.type not in (ChannelType.text, ChannelType.news, ChannelType.private): | ||||||
|             raise TypeError(f'Expected TextChannel or DMChannel not {type(channel)!r}') |             raise TypeError(f'Expected TextChannel or DMChannel not {type(channel)!r}') | ||||||
|  |  | ||||||
| @@ -1419,29 +1446,29 @@ class PartialMessage(Hashable): | |||||||
|         self._state = channel._state |         self._state = channel._state | ||||||
|         self.id = id |         self.id = id | ||||||
|  |  | ||||||
|     def _update(self, data): |     def _update(self, data) -> None: | ||||||
|         # This is used for duck typing purposes. |         # This is used for duck typing purposes. | ||||||
|         # Just do nothing with the data. |         # Just do nothing with the data. | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     # Also needed for duck typing purposes |     # Also needed for duck typing purposes | ||||||
|     # n.b. not exposed |     # n.b. not exposed | ||||||
|     pinned = property(None, lambda x, y: ...) |     pinned = property(None, lambda x, y: None) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self) -> str: | ||||||
|         return f'<PartialMessage id={self.id} channel={self.channel!r}>' |         return f'<PartialMessage id={self.id} channel={self.channel!r}>' | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def created_at(self): |     def created_at(self) -> datetime.datetime: | ||||||
|         """:class:`datetime.datetime`: The partial message's creation time in UTC.""" |         """:class:`datetime.datetime`: The partial message's creation time in UTC.""" | ||||||
|         return utils.snowflake_time(self.id) |         return utils.snowflake_time(self.id) | ||||||
|  |  | ||||||
|     @utils.cached_slot_property('_cs_guild') |     @utils.cached_slot_property('_cs_guild') | ||||||
|     def guild(self): |     def guild(self) -> Optional[Guild]: | ||||||
|         """Optional[:class:`Guild`]: The guild that the partial message belongs to, if applicable.""" |         """Optional[:class:`Guild`]: The guild that the partial message belongs to, if applicable.""" | ||||||
|         return getattr(self.channel, 'guild', None) |         return getattr(self.channel, 'guild', None) | ||||||
|  |  | ||||||
|     async def fetch(self): |     async def fetch(self) -> Message: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Fetches the partial message to a full :class:`Message`. |         Fetches the partial message to a full :class:`Message`. | ||||||
| @@ -1464,7 +1491,7 @@ class PartialMessage(Hashable): | |||||||
|         data = await self._state.http.get_message(self.channel.id, self.id) |         data = await self._state.http.get_message(self.channel.id, self.id) | ||||||
|         return self._state.create_message(channel=self.channel, data=data) |         return self._state.create_message(channel=self.channel, data=data) | ||||||
|  |  | ||||||
|     async def edit(self, **fields): |     async def edit(self, **fields: Any) -> Optional[Message]: | ||||||
|         """|coro| |         """|coro| | ||||||
|  |  | ||||||
|         Edits the message. |         Edits the message. | ||||||
| @@ -1558,7 +1585,7 @@ class PartialMessage(Hashable): | |||||||
|             data = await self._state.http.edit_message(self.channel.id, self.id, **fields) |             data = await self._state.http.edit_message(self.channel.id, self.id, **fields) | ||||||
|  |  | ||||||
|         if delete_after is not None: |         if delete_after is not None: | ||||||
|             await self.delete(delay=delete_after) |             await self.delete(delay=delete_after)  # type: ignore | ||||||
|  |  | ||||||
|         if fields: |         if fields: | ||||||
|             return self._state.create_message(channel=self.channel, data=data) |             return self._state.create_message(channel=self.channel, data=data)  # type: ignore | ||||||
|   | |||||||
| @@ -51,6 +51,7 @@ class _AttachmentOptional(TypedDict, total=False): | |||||||
|     height: Optional[int] |     height: Optional[int] | ||||||
|     width: Optional[int] |     width: Optional[int] | ||||||
|     content_type: str |     content_type: str | ||||||
|  |     spoiler: bool | ||||||
|  |  | ||||||
|  |  | ||||||
| class Attachment(_AttachmentOptional): | class Attachment(_AttachmentOptional): | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user