Compare commits

..

1 Commits

Author SHA1 Message Date
Gnome
c9983d0248 Make fetch_x methods cache when applicable 2021-10-02 19:34:56 +01:00
22 changed files with 760 additions and 1146 deletions

View File

@ -17,6 +17,18 @@ The Future of enhanced-discord.py
-------------------------- --------------------------
Enhanced discord.py is a fork of Rapptz's discord.py, that went unmaintained (`gist <https://gist.github.com/Rapptz/4a2f62751b9600a31a0d3c78100287f1>`_) Enhanced discord.py is a fork of Rapptz's discord.py, that went unmaintained (`gist <https://gist.github.com/Rapptz/4a2f62751b9600a31a0d3c78100287f1>`_)
It is currently maintained by (in alphabetical order)
- Chillymosh#8175
- Daggy#9889
- dank Had0cK#6081
- Dutchy#6127
- Eyesofcreeper#0001
- Gnome!#6669
- IAmTomahawkx#1000
- Jadon#2494
An overview of added features is available on the `custom features page <https://enhanced-dpy.readthedocs.io/en/latest/index.html#custom-features>`_. An overview of added features is available on the `custom features page <https://enhanced-dpy.readthedocs.io/en/latest/index.html#custom-features>`_.
Key Features Key Features

View File

@ -1434,7 +1434,7 @@ class Messageable:
components=components, components=components,
) )
ret = state.create_message(channel=channel, data=data) ret = state.store_message(channel=channel, data=data)
if view: if view:
state.store_view(view, ret.id) state.store_view(view, ret.id)
@ -1501,7 +1501,7 @@ class Messageable:
channel = await self._get_channel() channel = await self._get_channel()
data = await self._state.http.get_message(channel.id, id) data = await self._state.http.get_message(channel.id, id)
return self._state.create_message(channel=channel, data=data) return self._state.store_message(channel=channel, data=data)
async def pins(self) -> List[Message]: async def pins(self) -> List[Message]:
"""|coro| """|coro|
@ -1528,7 +1528,7 @@ class Messageable:
channel = await self._get_channel() channel = await self._get_channel()
state = self._state state = self._state
data = await state.http.pins_from(channel.id) data = await state.http.pins_from(channel.id)
return [state.create_message(channel=channel, data=m) for m in data] return [state.store_message(channel=channel, data=m) for m in data]
def history( def history(
self, self,

View File

@ -53,7 +53,7 @@ from .widget import Widget
from .guild import Guild from .guild import Guild
from .emoji import Emoji from .emoji import Emoji
from .channel import _threaded_channel_factory, PartialMessageable from .channel import _threaded_channel_factory, PartialMessageable
from .enums import ChannelType from .enums import ChannelType, StickerType
from .mentions import AllowedMentions from .mentions import AllowedMentions
from .errors import * from .errors import *
from .enums import Status, VoiceRegion from .enums import Status, VoiceRegion
@ -76,7 +76,8 @@ from .threads import Thread
from .sticker import GuildSticker, StandardSticker, StickerPack, _sticker_factory from .sticker import GuildSticker, StandardSticker, StickerPack, _sticker_factory
if TYPE_CHECKING: if TYPE_CHECKING:
from .abc import SnowflakeTime, PrivateChannel, GuildChannel, Snowflake from .abc import SnowflakeTime, PrivateChannel, GuildChannel as GuildChannelABC, Snowflake
from .guild import GuildChannel
from .channel import DMChannel from .channel import DMChannel
from .message import Message from .message import Message
from .member import Member from .member import Member
@ -780,7 +781,7 @@ class Client:
"""List[:class:`~discord.User`]: Returns a list of all the users the bot can see.""" """List[:class:`~discord.User`]: Returns a list of all the users the bot can see."""
return list(self._connection._users.values()) return list(self._connection._users.values())
def get_channel(self, id: int, /) -> Optional[Union[GuildChannel, Thread, PrivateChannel]]: def get_channel(self, id: int, /) -> Optional[Union[GuildChannelABC, Thread, PrivateChannel]]:
"""Returns a channel or thread with the given ID. """Returns a channel or thread with the given ID.
Parameters Parameters
@ -933,7 +934,7 @@ class Client:
""" """
return self._connection.get_sticker(id) return self._connection.get_sticker(id)
def get_all_channels(self) -> Generator[GuildChannel, None, None]: def get_all_channels(self) -> Generator[GuildChannelABC, None, None]:
"""A generator that retrieves every :class:`.abc.GuildChannel` the client can 'access'. """A generator that retrieves every :class:`.abc.GuildChannel` the client can 'access'.
This is equivalent to: :: This is equivalent to: ::
@ -1488,6 +1489,7 @@ class Client:
"""|coro| """|coro|
Retrieves the bot's application information. Retrieves the bot's application information.
This will fill up :attr:`application_id` and :attr:`application_flags`.
Raises Raises
------- -------
@ -1502,6 +1504,8 @@ class Client:
data = await self.http.application_info() data = await self.http.application_info()
if "rpc_origins" not in data: if "rpc_origins" not in data:
data["rpc_origins"] = None data["rpc_origins"] = None
self._connection.store_appinfo(data)
return AppInfo(self._connection, data) return AppInfo(self._connection, data)
async def fetch_user(self, user_id: int, /) -> User: async def fetch_user(self, user_id: int, /) -> User:
@ -1535,10 +1539,11 @@ class Client:
data = await self.http.get_user(user_id) data = await self.http.get_user(user_id)
return User(state=self._connection, data=data) return User(state=self._connection, data=data)
async def fetch_channel(self, channel_id: int, /) -> Union[GuildChannel, PrivateChannel, Thread]: async def fetch_channel(self, channel_id: int, /) -> Union[GuildChannelABC, PrivateChannel, Thread]:
"""|coro| """|coro|
Retrieves a :class:`.abc.GuildChannel`, :class:`.abc.PrivateChannel`, or :class:`.Thread` with the specified ID. Retrieves a :class:`.abc.GuildChannel`, :class:`.abc.PrivateChannel`, or :class:`.Thread` with the specified ID.
If found, will store the Channel in the internal cache, meaning :meth:``get_channel`` will succeed afterwards.
.. note:: .. note::
@ -1570,14 +1575,18 @@ class Client:
if ch_type in (ChannelType.group, ChannelType.private): if ch_type in (ChannelType.group, ChannelType.private):
# the factory will be a DMChannel or GroupChannel here # the factory will be a DMChannel or GroupChannel here
channel = factory(me=self.user, data=data, state=self._connection) # type: ignore channel: PrivateChannel = factory(me=self.user, data=data, state=self._connection) # type: ignore
self._connection._add_private_channel(channel) # type: ignore
else: else:
# the factory can't be a DMChannel or GroupChannel here # the factory can't be a DMChannel or GroupChannel here
guild_id = int(data["guild_id"]) # type: ignore guild_id = int(data["guild_id"]) # type: ignore
guild = self.get_guild(guild_id) or Object(id=guild_id) guild = self.get_guild(guild_id)
# GuildChannels expect a Guild, we may be passing an Object
channel = factory(guild=guild, state=self._connection, data=data) # type: ignore
if guild is None:
return factory(guild=Object(guild_id), state=self._connection, data=data) # type: ignore
channel: GuildChannel = factory(guild=guild, state=self._connection, data=data) # type: ignore
guild._add_channel(channel)
return channel return channel
async def fetch_webhook(self, webhook_id: int, /) -> Webhook: async def fetch_webhook(self, webhook_id: int, /) -> Webhook:
@ -1606,6 +1615,7 @@ class Client:
"""|coro| """|coro|
Retrieves a :class:`.Sticker` with the specified ID. Retrieves a :class:`.Sticker` with the specified ID.
If found, will store the sticker in the internal cache, meaning :meth:``get_sticker`` will succeed afterwards.
.. versionadded:: 2.0 .. versionadded:: 2.0
@ -1623,7 +1633,11 @@ class Client:
""" """
data = await self.http.get_sticker(sticker_id) data = await self.http.get_sticker(sticker_id)
cls, _ = _sticker_factory(data["type"]) # type: ignore cls, _ = _sticker_factory(data["type"]) # type: ignore
return cls(state=self._connection, data=data) # type: ignore
if data["type"] == StickerType.guild: # type: ignore
return self._connection.store_sticker(data) # type: ignore
else:
return cls(state=self._connection, data=data) # type: ignore
async def fetch_premium_sticker_packs(self) -> List[StickerPack]: async def fetch_premium_sticker_packs(self) -> List[StickerPack]:
"""|coro| """|coro|

View File

@ -55,7 +55,6 @@ __all__ = (
"InteractionType", "InteractionType",
"InteractionResponseType", "InteractionResponseType",
"NSFWLevel", "NSFWLevel",
"ProtocolURL",
) )
@ -530,7 +529,6 @@ class InteractionType(Enum):
ping = 1 ping = 1
application_command = 2 application_command = 2
component = 3 component = 3
application_command_autocomplete = 4
class InteractionResponseType(Enum): class InteractionResponseType(Enum):
@ -541,7 +539,6 @@ class InteractionResponseType(Enum):
deferred_channel_message = 5 # (with source) deferred_channel_message = 5 # (with source)
deferred_message_update = 6 # for components deferred_message_update = 6 # for components
message_update = 7 # for components message_update = 7 # for components
application_command_autocomplete_result = 8
class VideoQualityMode(Enum): class VideoQualityMode(Enum):
@ -593,74 +590,6 @@ class NSFWLevel(Enum, comparable=True):
age_restricted = 3 age_restricted = 3
class ProtocolURL(Enum):
# General
home = "discord://-/channels/@me/"
nitro = "discord://-/store"
apps = "discord://-/apps" # Breaks the client on windows (Shows download links for different OS)
guild_discovery = "discord://-/guild-discovery"
guild_create = "discord://-/guilds/create"
guild_invite = "discord://-/invite/{invite_code}"
# Settings
account_settings = "discord://-/settings/account"
profile_settings = "discord://-/settings/profile-customization"
privacy_settings = "discord://-/settings/privacy-and-safety"
safety_settings = "discord://-/settings/privacy-and-safety" # Alias
authorized_apps_settings = "discord://-/settings/authorized-apps"
connections_settings = "discord://-/settings/connections"
nitro_settings = "discord://-/settings/premium" # Same as store, but inside of settings
guild_premium_subscription = "discord://-/settings/premium-guild-subscription"
subscription_settings = "discord://-/settings/subscriptions"
gift_inventory_settings = "discord://-/settings/inventory"
billing_settings = "discord://-/settings/billing"
appearance_settings = "discord://-/settings/appearance"
accessibility_settings = "discord://-/settings/accessibility"
voice_video_settings = "discord://-/settings/voice"
text_images_settings = "discord://-/settings/text"
notifications_settings = "discord://-/settings/notifications"
keybinds_settings = "discord://-/settings/keybinds"
language_settings = "discord://-/settings/locale"
windows_settings = "discord://-/settings/windows" # Doesnt work if used on wrong platform
linux_settings = "discord://-/settings/linux" # Doesnt work if used on wrong platform
streamer_mode_settings = "discord://-/settings/streamer-mode"
advanced_settings = "discord://-/settings/advanced"
activity_status_settings = "discord://-/settings/activity-status"
game_overlay_settings = "discord://-/settings/overlay"
hypesquad_settings = "discord://-/settings/hypesquad-online"
changelogs = "discord://-/settings/changelogs"
# Doesn't work if you don't have it actually activated. Just blank screen.
experiments = "discord://-/settings/experiments"
developer_options = "discord://-/settings/developer-options" # Same as experiments
hotspot_options = "discord://-/settings/hotspot-options" # Same as experiments
# Users, Guilds, and DMs
user_profile = "discord://-/users/{user_id}"
dm_channel = "discord://-/channels/@me/{channel_id}"
dm_message = "discord://-/channels/@me/{channel_id}/{message_id}"
guild_channel = "discord://-/channels/{guild_id}/{channel_id}"
guild_message = "discord://-/channels/{guild_id}/{channel_id}/{message_id}"
guild_membership_screening = "discord://-/member-verification/{guild_id}"
# Library
games_library = "discord://-/library"
library_settings = "discord://-/library/settings"
def __str__(self) -> str:
return self.value
def format(self, **kwargs: Any) -> str:
return self.value.format(**kwargs)
T = TypeVar("T") T = TypeVar("T")

View File

@ -28,7 +28,6 @@ from __future__ import annotations
import asyncio import asyncio
import collections import collections
import collections.abc import collections.abc
from functools import cached_property
import inspect import inspect
import importlib.util import importlib.util
@ -73,9 +72,7 @@ from .cog import Cog
if TYPE_CHECKING: if TYPE_CHECKING:
import importlib.machinery import importlib.machinery
from discord.role import Role
from discord.message import Message from discord.message import Message
from discord.abc import PartialMessageableChannel
from ._types import ( from ._types import (
Check, Check,
CoroFunc, CoroFunc,
@ -97,16 +94,9 @@ CXT = TypeVar("CXT", bound="Context")
class _FakeSlashMessage(discord.PartialMessage): class _FakeSlashMessage(discord.PartialMessage):
activity = application = edited_at = reference = webhook_id = None activity = application = edited_at = reference = webhook_id = None
attachments = components = reactions = stickers = [] attachments = components = reactions = stickers = mentions = []
tts = False
raw_mentions = discord.Message.raw_mentions
clean_content = discord.Message.clean_content
channel_mentions = discord.Message.channel_mentions
raw_role_mentions = discord.Message.raw_role_mentions
raw_channel_mentions = discord.Message.raw_channel_mentions
author: Union[discord.User, discord.Member] author: Union[discord.User, discord.Member]
tts = False
@classmethod @classmethod
def from_interaction( def from_interaction(
@ -118,22 +108,6 @@ class _FakeSlashMessage(discord.PartialMessage):
return self return self
@cached_property
def mentions(self) -> List[Union[discord.Member, discord.User]]:
client = self._state._get_client()
if self.guild:
ensure_user = lambda id: self.guild.get_member(id) or client.get_user(id) # type: ignore
else:
ensure_user = client.get_user
return discord.utils._unique(filter(None, map(ensure_user, self.raw_mentions)))
@cached_property
def role_mentions(self) -> List[Role]:
if self.guild is None:
return []
return discord.utils._unique(filter(None, map(self.guild.get_role, self.raw_role_mentions)))
def when_mentioned(bot: Union[Bot, AutoShardedBot], msg: Message) -> List[str]: def when_mentioned(bot: Union[Bot, AutoShardedBot], msg: Message) -> List[str]:
"""A callable that implements a command prefix equivalent to being mentioned. """A callable that implements a command prefix equivalent to being mentioned.
@ -188,17 +162,16 @@ def _is_submodule(parent: str, child: str) -> bool:
def _unwrap_slash_groups( def _unwrap_slash_groups(
data: ApplicationCommandInteractionData, data: ApplicationCommandInteractionData,
) -> Tuple[str, Dict[str, ApplicationCommandInteractionDataOption]]: ) -> Tuple[str, List[ApplicationCommandInteractionDataOption]]:
command_name = data["name"] command_name = data["name"]
command_options: Any = data.get("options") or [] command_options = data.get("options") or []
while True: while any(o["type"] in {1, 2} for o in command_options): # type: ignore
try: for option in command_options: # type: ignore
option = next(o for o in command_options if o["type"] in {1, 2}) if option["type"] in {1, 2}: # type: ignore
except StopIteration: command_name += f' {option["name"]}' # type: ignore
return command_name, {o["name"]: o for o in command_options} command_options = option.get("options") or []
else:
command_name += f' {option["name"]}' return command_name, command_options
command_options = option.get("options") or []
def _quote_string_safe(string: str) -> str: def _quote_string_safe(string: str) -> str:
@ -1293,14 +1266,14 @@ class BotBase(GroupMixin):
# Make our fake message so we can pass it to ext.commands # Make our fake message so we can pass it to ext.commands
message: discord.Message = _FakeSlashMessage.from_interaction(interaction, channel) # type: ignore message: discord.Message = _FakeSlashMessage.from_interaction(interaction, channel) # type: ignore
message.content = f"/{command_name}" message.content = f"/{command_name} "
# Add arguments to fake message content, in the right order # Add arguments to fake message content, in the right order
ignore_params: List[inspect.Parameter] = [] ignore_params: List[inspect.Parameter] = []
for name, param in command.clean_params.items(): for name, param in command.clean_params.items():
if inspect.isclass(param.annotation) and issubclass(param.annotation, FlagConverter): if inspect.isclass(param.annotation) and issubclass(param.annotation, FlagConverter):
for name, flag in param.annotation.get_flags().items(): for name, flag in param.annotation.get_flags().items():
option = command_options.get(name) option = next((o for o in command_options if o["name"] == name), None)
if option is None: if option is None:
if flag.required: if flag.required:
@ -1308,10 +1281,10 @@ class BotBase(GroupMixin):
else: else:
prefix = param.annotation.__commands_flag_prefix__ prefix = param.annotation.__commands_flag_prefix__
delimiter = param.annotation.__commands_flag_delimiter__ delimiter = param.annotation.__commands_flag_delimiter__
message.content += f" {prefix}{name}{delimiter}{option['value']}" # type: ignore message.content += f"{prefix}{name} {option['value']}{delimiter}" # type: ignore
continue continue
option = command_options.get(name) option = next((o for o in command_options if o["name"] == name), None)
if option is None: if option is None:
if param.default is param.empty and not command._is_typing_optional(param.annotation): if param.default is param.empty and not command._is_typing_optional(param.annotation):
raise errors.MissingRequiredArgument(param) raise errors.MissingRequiredArgument(param)
@ -1324,9 +1297,9 @@ class BotBase(GroupMixin):
): ):
# String with space in without "consume rest" # String with space in without "consume rest"
option = cast(_ApplicationCommandInteractionDataOptionString, option) option = cast(_ApplicationCommandInteractionDataOptionString, option)
message.content += f" {_quote_string_safe(option['value'])}" message.content += f"{_quote_string_safe(option['value'])} "
else: else:
message.content += f' {option.get("value", "")}' message.content += f'{option.get("value", "")} '
ctx = await self.get_context(message) ctx = await self.get_context(message)
ctx._ignored_params = ignore_params ctx._ignored_params = ignore_params

View File

@ -22,12 +22,11 @@ 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 from __future__ import annotations
import asyncio
import inspect import inspect
import re import re
from datetime import timedelta from datetime import timedelta
from typing import Any, Dict, Generic, List, Literal, NoReturn, Optional, TYPE_CHECKING, TypeVar, Union, overload from typing import Any, Dict, Generic, List, Literal, Optional, TYPE_CHECKING, TypeVar, Union, overload
import discord.abc import discord.abc
import discord.utils import discord.utils
@ -156,7 +155,6 @@ class Context(discord.abc.Messageable, Generic[BotT]):
self.command_failed: bool = command_failed self.command_failed: bool = command_failed
self.current_parameter: Optional[inspect.Parameter] = current_parameter self.current_parameter: Optional[inspect.Parameter] = current_parameter
self._ignored_params: List[inspect.Parameter] = [] self._ignored_params: List[inspect.Parameter] = []
self._typing_task: Optional[asyncio.Task[NoReturn]] = None
self._state: ConnectionState = self.message._state self._state: ConnectionState = self.message._state
async def invoke(self, command: Command[CogT, P, T], /, *args: P.args, **kwargs: P.kwargs) -> T: async def invoke(self, command: Command[CogT, P, T], /, *args: P.args, **kwargs: P.kwargs) -> T:
@ -457,10 +455,6 @@ class Context(discord.abc.Messageable, Generic[BotT]):
In a normal context, it always returns a :class:`.Message` In a normal context, it always returns a :class:`.Message`
""" """
if self._typing_task is not None:
self._typing_task.cancel()
self._typing_task = None
if self.interaction is None or ( if self.interaction is None or (
self.interaction.response.responded_at is not None self.interaction.response.responded_at is not None
and discord.utils.utcnow() - self.interaction.response.responded_at >= timedelta(minutes=15) and discord.utils.utcnow() - self.interaction.response.responded_at >= timedelta(minutes=15)
@ -506,30 +500,3 @@ class Context(discord.abc.Messageable, Generic[BotT]):
self, content: Optional[str] = None, return_message: bool = True, **kwargs: Any self, content: Optional[str] = None, return_message: bool = True, **kwargs: Any
) -> Optional[Union[Message, WebhookMessage]]: ) -> Optional[Union[Message, WebhookMessage]]:
return await self.send(content, return_message=return_message, reference=self.message, **kwargs) # type: ignore return await self.send(content, return_message=return_message, reference=self.message, **kwargs) # type: ignore
async def defer(self, *, ephemeral: bool = False, trigger_typing: bool = True) -> None:
"""|coro|
Defers the Slash Command interaction if ran in a slash command **or**
Loops triggering ``Bot is typing`` in the channel if run in a message command.
Parameters
------------
trigger_typing: :class:`bool`
Indicates whether to trigger typing in a message command.
ephemeral: :class:`bool`
Indicates whether the deferred message will eventually be ephemeral in a slash command.
"""
if self.interaction is None:
if self._typing_task is None and trigger_typing:
async def typing_task():
while True:
await self.trigger_typing()
await asyncio.sleep(10)
self._typing_task = self.bot.loop.create_task(typing_task())
else:
await self.interaction.response.defer(ephemeral=ephemeral)

View File

@ -1032,8 +1032,6 @@ class Option(Generic[T, DT]): # type: ignore
The default for this option, overwrites Option during parsing. The default for this option, overwrites Option during parsing.
description: :class:`str` description: :class:`str`
The description for this option, is unpacked to :attr:`.Command.option_descriptions` The description for this option, is unpacked to :attr:`.Command.option_descriptions`
name: :class:`str`
The name of the option. This defaults to the parameter name.
""" """
description: DT description: DT
@ -1041,18 +1039,17 @@ class Option(Generic[T, DT]): # type: ignore
__slots__ = ( __slots__ = (
"default", "default",
"description", "description",
"name",
) )
def __init__( def __init__(self, default: T = inspect.Parameter.empty, *, description: DT) -> None:
self, default: T = inspect.Parameter.empty, *, description: DT, name: str = discord.utils.MISSING
) -> None:
self.description = description self.description = description
self.default = default self.default = default
self.name: str = name
Option: Any if TYPE_CHECKING:
# Terrible workaround for type checking reasons
def Option(default: T = inspect.Parameter.empty, *, description: str) -> T:
...
def _convert_to_bool(argument: str) -> bool: def _convert_to_bool(argument: str) -> bool:

View File

@ -136,14 +136,6 @@ application_option_type_lookup = {
discord.Role: 8, discord.Role: 8,
float: 10, float: 10,
} }
application_option_channel_types = {
discord.VoiceChannel: [2],
discord.TextChannel: [0, 5, 6],
discord.CategoryChannel: [4],
discord.Thread: [10, 11, 12],
discord.StageChannel: [13],
}
if TYPE_CHECKING: if TYPE_CHECKING:
P = ParamSpec("P") P = ParamSpec("P")
@ -174,12 +166,8 @@ def get_signature_parameters(
annotation = parameter.annotation annotation = parameter.annotation
if isinstance(parameter.default, Option): # type: ignore if isinstance(parameter.default, Option): # type: ignore
option = parameter.default option = parameter.default
parameter = parameter.replace(default=option.default)
if option.name is not MISSING:
name = option.name
parameter.replace(name=name)
descriptions[name] = option.description descriptions[name] = option.description
parameter = parameter.replace(default=option.default)
if annotation is parameter.empty: if annotation is parameter.empty:
params[name] = parameter params[name] = parameter
@ -1238,25 +1226,15 @@ class Command(_BaseCommand, Generic[CogT, P, T]):
ctx.command = original ctx.command = original
def _param_to_options( def _param_to_options(
self, name: str, annotation: Any, required: bool, varadic: bool, description: Optional[str] = None self, name: str, annotation: Any, required: bool, varadic: bool
) -> List[Optional[ApplicationCommandInteractionDataOption]]: ) -> List[Optional[ApplicationCommandInteractionDataOption]]:
if description is not None:
self.option_descriptions[name] = description
description = self.option_descriptions[name]
origin = getattr(annotation, "__origin__", None) origin = getattr(annotation, "__origin__", None)
if inspect.isclass(annotation) and issubclass(annotation, FlagConverter): if inspect.isclass(annotation) and issubclass(annotation, FlagConverter):
return [ return [
param param
for name, flag in annotation.get_flags().items() for name, flag in annotation.get_flags().items()
for param in self._param_to_options( for param in self._param_to_options(
name, name, flag.annotation, required=flag.required, varadic=flag.annotation is tuple
flag.annotation,
required=flag.required,
varadic=flag.annotation is tuple,
description=flag.description if flag.description is not MISSING else None,
) )
] ]
@ -1264,16 +1242,15 @@ class Command(_BaseCommand, Generic[CogT, P, T]):
annotation = str annotation = str
origin = None origin = None
if not required and origin is Union and annotation.__args__[-1] is type(None): if not required and origin is not None and len(annotation.__args__) == 2:
# Unpack Optional[T] (Union[T, None]) into just T # Unpack Optional[T] (Union[T, None]) into just T
annotation = annotation.__args__[0] annotation, origin = annotation.__args__[0], None
origin = getattr(annotation, "__origin__", None)
option: Dict[str, Any] = { option: Dict[str, Any] = {
"type": 3, "type": 3,
"name": name, "name": name,
"required": required, "required": required,
"description": description, "description": self.option_descriptions[name],
} }
if origin is None: if origin is None:
@ -1288,23 +1265,12 @@ class Command(_BaseCommand, Generic[CogT, P, T]):
for python_type, discord_type in application_option_type_lookup.items(): for python_type, discord_type in application_option_type_lookup.items():
if issubclass(annotation, python_type): if issubclass(annotation, python_type):
option["type"] = discord_type option["type"] = discord_type
# Set channel types
if discord_type == 7:
option["channel_types"] = application_option_channel_types[annotation]
break break
elif origin is Union: elif origin is Union:
if annotation in {Union[discord.Member, discord.Role], Union[MemberConverter, RoleConverter]}: if annotation in {Union[discord.Member, discord.Role], Union[MemberConverter, RoleConverter]}:
option["type"] = 9 option["type"] = 9
elif all([arg in application_option_channel_types for arg in annotation.__args__]):
option["type"] = 7
option["channel_types"] = [
discord_value
for arg in annotation.__args__
for discord_value in application_option_channel_types[arg]
]
elif origin is Literal: elif origin is Literal:
literal_values = annotation.__args__ literal_values = annotation.__args__
python_type = type(literal_values[0]) python_type = type(literal_values[0])

View File

@ -81,8 +81,6 @@ class Flag:
------------ ------------
name: :class:`str` name: :class:`str`
The name of the flag. The name of the flag.
description: :class:`str`
The description of the flag.
aliases: List[:class:`str`] aliases: List[:class:`str`]
The aliases of the flag name. The aliases of the flag name.
attribute: :class:`str` attribute: :class:`str`
@ -99,7 +97,6 @@ class Flag:
""" """
name: str = MISSING name: str = MISSING
description: str = MISSING
aliases: List[str] = field(default_factory=list) aliases: List[str] = field(default_factory=list)
attribute: str = MISSING attribute: str = MISSING
annotation: Any = MISSING annotation: Any = MISSING
@ -120,7 +117,6 @@ class Flag:
def flag( def flag(
*, *,
name: str = MISSING, name: str = MISSING,
description: str = MISSING,
aliases: List[str] = MISSING, aliases: List[str] = MISSING,
default: Any = MISSING, default: Any = MISSING,
max_args: int = MISSING, max_args: int = MISSING,
@ -133,8 +129,6 @@ def flag(
------------ ------------
name: :class:`str` name: :class:`str`
The flag name. If not given, defaults to the attribute name. The flag name. If not given, defaults to the attribute name.
description: :class:`str`
Description of the flag for the slash commands options. The default value is `'no description'`.
aliases: List[:class:`str`] aliases: List[:class:`str`]
Aliases to the flag name. If not given no aliases are set. Aliases to the flag name. If not given no aliases are set.
default: Any default: Any
@ -149,9 +143,7 @@ def flag(
Whether multiple given values overrides the previous value. The default Whether multiple given values overrides the previous value. The default
value depends on the annotation given. value depends on the annotation given.
""" """
return Flag( return Flag(name=name, aliases=aliases, default=default, max_args=max_args, override=override)
name=name, description=description, aliases=aliases, default=default, max_args=max_args, override=override
)
def validate_flag_name(name: str, forbidden: Set[str]): def validate_flag_name(name: str, forbidden: Set[str]):

View File

@ -951,6 +951,15 @@ class MemberCacheFlags(BaseFlags):
""" """
return 2 return 2
@flag_value
def fetched(self):
""":class:`bool`: Whether to cache members that are fetched via :meth:``Guild.fetch_member``
or :meth:``Guild.fetch_members``
Members that leave the guild are no longer cached.
"""
return 4
@classmethod @classmethod
def from_intents(cls: Type[MemberCacheFlags], intents: Intents) -> MemberCacheFlags: def from_intents(cls: Type[MemberCacheFlags], intents: Intents) -> MemberCacheFlags:
"""A factory method that creates a :class:`MemberCacheFlags` based on """A factory method that creates a :class:`MemberCacheFlags` based on
@ -968,6 +977,8 @@ class MemberCacheFlags(BaseFlags):
""" """
self = cls.none() self = cls.none()
self.fetched = True
if intents.members: if intents.members:
self.joined = True self.joined = True
if intents.voice_states: if intents.voice_states:

View File

@ -428,7 +428,7 @@ class Guild(Hashable):
self.mfa_level: MFALevel = guild.get("mfa_level") self.mfa_level: MFALevel = guild.get("mfa_level")
self.emojis: Tuple[Emoji, ...] = tuple(map(lambda d: state.store_emoji(self, d), guild.get("emojis", []))) self.emojis: Tuple[Emoji, ...] = tuple(map(lambda d: state.store_emoji(self, d), guild.get("emojis", [])))
self.stickers: Tuple[GuildSticker, ...] = tuple( self.stickers: Tuple[GuildSticker, ...] = tuple(
map(lambda d: state.store_sticker(self, d), guild.get("stickers", [])) map(lambda d: state.store_sticker(d), guild.get("stickers", []))
) )
self.features: List[GuildFeature] = guild.get("features", []) self.features: List[GuildFeature] = guild.get("features", [])
self._splash: Optional[str] = guild.get("splash") self._splash: Optional[str] = guild.get("splash")
@ -1594,6 +1594,7 @@ class Guild(Hashable):
"""|coro| """|coro|
Retrieves all :class:`abc.GuildChannel` that the guild has. Retrieves all :class:`abc.GuildChannel` that the guild has.
Will store the Channels in the internal cache, meaning :meth:``get_channel`` will succeed afterwards.
.. note:: .. note::
@ -1616,11 +1617,12 @@ class Guild(Hashable):
data = await self._state.http.get_all_guild_channels(self.id) data = await self._state.http.get_all_guild_channels(self.id)
def convert(d): def convert(d):
factory, ch_type = _guild_channel_factory(d["type"]) factory, _ = _guild_channel_factory(d["type"])
if factory is None: if factory is None:
raise InvalidData("Unknown channel type {type} for channel ID {id}.".format_map(d)) raise InvalidData("Unknown channel type {type} for channel ID {id}.".format_map(d))
channel = factory(guild=self, state=self._state, data=d) channel = factory(guild=self, state=self._state, data=d)
self._add_channel(channel)
return channel return channel
return [convert(d) for d in data] return [convert(d) for d in data]
@ -1712,6 +1714,8 @@ class Guild(Hashable):
"""|coro| """|coro|
Retrieves a :class:`Member` from a guild ID, and a member ID. Retrieves a :class:`Member` from a guild ID, and a member ID.
If found, will store the Member in the internal cache, filling up :attr:`members`
and meaning :meth:``get_member`` will succeed afterwards.
.. note:: .. note::
@ -1737,7 +1741,11 @@ class Guild(Hashable):
The member from the member ID. The member from the member ID.
""" """
data = await self._state.http.get_member(self.id, member_id) data = await self._state.http.get_member(self.id, member_id)
return Member(data=data, state=self._state, guild=self) member = Member(data=data, state=self._state, guild=self)
if self._state.member_cache_flags.fetched:
self._add_member(member)
return member
async def try_member(self, member_id: int, /) -> Optional[Member]: async def try_member(self, member_id: int, /) -> Optional[Member]:
"""|coro| """|coro|
@ -2257,7 +2265,7 @@ class Guild(Hashable):
payload["tags"] = emoji payload["tags"] = emoji
data = await self._state.http.create_guild_sticker(self.id, payload, file, reason) data = await self._state.http.create_guild_sticker(self.id, payload, file, reason)
return self._state.store_sticker(self, data) return self._state.store_sticker(data)
async def delete_sticker(self, sticker: Snowflake, *, reason: Optional[str] = None) -> None: async def delete_sticker(self, sticker: Snowflake, *, reason: Optional[str] = None) -> None:
"""|coro| """|coro|

View File

@ -188,8 +188,8 @@ class HTTPClient:
self.proxy_auth: Optional[aiohttp.BasicAuth] = proxy_auth self.proxy_auth: Optional[aiohttp.BasicAuth] = proxy_auth
self.use_clock: bool = not unsync_clock self.use_clock: bool = not unsync_clock
u_agent = "DiscordBot (https://github.com/iDevision/enhanced-discord.py {0}) Python/{1[0]}.{1[1]} aiohttp/{2}" user_agent = "DiscordBot (https://github.com/Rapptz/discord.py {0}) Python/{1[0]}.{1[1]} aiohttp/{2}"
self.user_agent: str = u_agent.format(__version__, sys.version_info, aiohttp.__version__) self.user_agent: str = user_agent.format(__version__, sys.version_info, aiohttp.__version__)
def recreate(self) -> None: def recreate(self) -> None:
if self.__session.closed: if self.__session.closed:

View File

@ -51,7 +51,6 @@ if TYPE_CHECKING:
from .types.interactions import ( from .types.interactions import (
Interaction as InteractionPayload, Interaction as InteractionPayload,
ApplicationCommandOptionChoice,
InteractionData, InteractionData,
) )
from .guild import Guild from .guild import Guild
@ -138,7 +137,7 @@ class Interaction:
self.message: Optional[Message] self.message: Optional[Message]
try: try:
self.message = Message(state=self._state, channel=self.channel, data=data["message"]) # type: ignore self.message = self._state.store_message(channel=self.channel, data=data["message"]) # type: ignore
except KeyError: except KeyError:
self.message = None self.message = None
@ -633,43 +632,6 @@ class InteractionResponse:
self.responded_at = utils.utcnow() self.responded_at = utils.utcnow()
async def autocomplete_result(self, choices: List[ApplicationCommandOptionChoice]):
"""|coro|
Responds to this autocomplete interaction with the resulting choices.
This should rarely be used.
Parameters
-----------
choices: List[Dict[:class:`str`, :class:`str`]]
The choices to be shown in the autocomplete UI of the user.
Must be a list of dictionaries with the ``name`` and ``value`` keys.
Raises
-------
HTTPException
Responding to the interaction failed.
InteractionResponded
This interaction has already been responded to before.
"""
if self.is_done():
raise InteractionResponded(self._parent)
parent = self._parent
if parent.type is not InteractionType.application_command_autocomplete:
return
adapter = async_context.get()
await adapter.create_interaction_response(
parent.id,
parent.token,
session=parent._session,
type=InteractionResponseType.application_command_autocomplete_result.value,
data={"choices": choices},
)
self.responded_at = utils.utcnow()
class _InteractionMessageState: class _InteractionMessageState:
__slots__ = ("_parent", "_interaction") __slots__ = ("_parent", "_interaction")

View File

@ -42,22 +42,12 @@ __all__ = (
) )
if TYPE_CHECKING: if TYPE_CHECKING:
from .types.audit_log import ( from .types.member import MemberWithUser as MemberWithUserPayload
AuditLog as AuditLogPayload, from .types.user import PartialUser as PartialUserPayload
) from .types.audit_log import AuditLog as AuditLogPayload
from .types.guild import ( from .types.message import Message as MessagePayload
Guild as GuildPayload, from .types.threads import Thread as ThreadPayload
) from .types.guild import Guild as GuildPayload
from .types.message import (
Message as MessagePayload,
)
from .types.user import (
PartialUser as PartialUserPayload,
)
from .types.threads import (
Thread as ThreadPayload,
)
from .member import Member from .member import Member
from .user import User from .user import User
@ -354,7 +344,7 @@ class HistoryIterator(_AsyncIterator["Message"]):
channel = self.channel channel = self.channel
for element in data: for element in data:
await self.messages.put(self.state.create_message(channel=channel, data=element)) await self.messages.put(self.state.store_message(channel=channel, data=element))
async def _retrieve_messages(self, retrieve) -> List[Message]: async def _retrieve_messages(self, retrieve) -> List[Message]:
"""Retrieve messages and update next parameters.""" """Retrieve messages and update next parameters."""
@ -615,14 +605,18 @@ class MemberIterator(_AsyncIterator["Member"]):
if isinstance(after, datetime.datetime): if isinstance(after, datetime.datetime):
after = Object(id=time_snowflake(after, high=True)) after = Object(id=time_snowflake(after, high=True))
self.guild = guild
self.limit = limit self.limit = limit
self.guild: Guild = guild
self.after = after or OLDEST_OBJECT self.after = after or OLDEST_OBJECT
self.state = self.guild._state self.state = self.guild._state
self.get_members = self.state.http.get_members self.get_members = self.state.http.get_members
self.members = asyncio.Queue() self.members = asyncio.Queue()
self.create_member = (
self.create_member_cache if self.state.member_cache_flags.fetched else self.create_member_no_cache
)
async def next(self) -> Member: async def next(self) -> Member:
if self.members.empty(): if self.members.empty():
await self.fill_members() await self.fill_members()
@ -657,11 +651,16 @@ class MemberIterator(_AsyncIterator["Member"]):
for element in reversed(data): for element in reversed(data):
await self.members.put(self.create_member(element)) await self.members.put(self.create_member(element))
def create_member(self, data): def create_member_no_cache(self, data: MemberWithUserPayload) -> Member:
from .member import Member from .member import Member
return Member(data=data, guild=self.guild, state=self.state) return Member(data=data, guild=self.guild, state=self.state)
def create_member_cache(self, data: MemberWithUserPayload) -> Member:
member = self.create_member_no_cache(data)
self.guild._add_member(member)
return member
class ArchivedThreadIterator(_AsyncIterator["Thread"]): class ArchivedThreadIterator(_AsyncIterator["Thread"]):
def __init__( def __init__(

View File

@ -1331,7 +1331,7 @@ class Message(Hashable):
payload["components"] = [] payload["components"] = []
data = await self._state.http.edit_message(self.channel.id, self.id, **payload) data = await self._state.http.edit_message(self.channel.id, self.id, **payload)
message = Message(state=self._state, channel=self.channel, data=data) message = self._state.store_message(channel=self.channel, data=data)
if view and not view.is_finished(): if view and not view.is_finished():
self._state.store_view(view, self.id) self._state.store_view(view, self.id)
@ -1756,7 +1756,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.store_message(channel=self.channel, data=data)
async def edit(self, **fields: Any) -> Optional[Message]: async def edit(self, **fields: Any) -> Optional[Message]:
"""|coro| """|coro|
@ -1873,7 +1873,7 @@ class PartialMessage(Hashable):
if fields: if fields:
# data isn't unbound # data isn't unbound
msg = self._state.create_message(channel=self.channel, data=data) # type: ignore msg = self._state.store_message(channel=self.channel, data=data) # type: ignore
if view and not view.is_finished(): if view and not view.is_finished():
self._state.store_view(view, self.id) self._state.store_view(view, self.id)
return msg return msg

View File

@ -27,7 +27,6 @@ from __future__ import annotations
import asyncio import asyncio
from collections import deque, OrderedDict from collections import deque, OrderedDict
import copy import copy
import datetime
import itertools import itertools
import logging import logging
from typing import Dict, Optional, TYPE_CHECKING, Union, Callable, Any, List, TypeVar, Coroutine, Sequence, Tuple, Deque from typing import Dict, Optional, TYPE_CHECKING, Union, Callable, Any, List, TypeVar, Coroutine, Sequence, Tuple, Deque
@ -75,6 +74,7 @@ if TYPE_CHECKING:
from .types.sticker import GuildSticker as GuildStickerPayload from .types.sticker import GuildSticker as GuildStickerPayload
from .types.guild import Guild as GuildPayload from .types.guild import Guild as GuildPayload
from .types.message import Message as MessagePayload from .types.message import Message as MessagePayload
from .types.appinfo import AppInfo as AppInfoPayload
T = TypeVar("T") T = TypeVar("T")
CS = TypeVar("CS", bound="ConnectionState") CS = TypeVar("CS", bound="ConnectionState")
@ -323,6 +323,13 @@ class ConnectionState:
for vc in self.voice_clients: for vc in self.voice_clients:
vc.main_ws = ws # type: ignore vc.main_ws = ws # type: ignore
def store_message(self, channel: MessageableChannel, data: MessagePayload) -> Message:
message = Message(state=self, channel=channel, data=data)
if self._messages is not None:
self._messages.append(message)
return message
def store_user(self, data: UserPayload) -> User: def store_user(self, data: UserPayload) -> User:
user_id = int(data["id"]) user_id = int(data["id"])
try: try:
@ -353,11 +360,20 @@ class ConnectionState:
self._emojis[emoji_id] = emoji = Emoji(guild=guild, state=self, data=data) self._emojis[emoji_id] = emoji = Emoji(guild=guild, state=self, data=data)
return emoji return emoji
def store_sticker(self, guild: Guild, data: GuildStickerPayload) -> GuildSticker: def store_sticker(self, data: GuildStickerPayload) -> GuildSticker:
sticker_id = int(data["id"]) sticker_id = int(data["id"])
self._stickers[sticker_id] = sticker = GuildSticker(state=self, data=data) self._stickers[sticker_id] = sticker = GuildSticker(state=self, data=data)
return sticker return sticker
def store_appinfo(self, data: AppInfoPayload):
self.application_id = utils._get_as_snowflake(data, "id")
flags = data.get("flags")
if flags is not None:
self.application_flags = ApplicationFlags._from_value(flags)
return data
def store_view(self, view: View, message_id: Optional[int] = None) -> None: def store_view(self, view: View, message_id: Optional[int] = None) -> None:
self._view_store.add_view(view, message_id) self._view_store.add_view(view, message_id)
@ -563,9 +579,7 @@ class ConnectionState:
except KeyError: except KeyError:
pass pass
else: else:
self.application_id = utils._get_as_snowflake(application, "id") self.store_appinfo(application)
# flags will always be present here
self.application_flags = ApplicationFlags._from_value(application["flags"]) # type: ignore
for guild_data in data["guilds"]: for guild_data in data["guilds"]:
self._add_guild_from_data(guild_data) self._add_guild_from_data(guild_data)
@ -581,8 +595,7 @@ class ConnectionState:
# channel would be the correct type here # channel would be the correct type here
message = Message(channel=channel, data=data, state=self) # type: ignore message = Message(channel=channel, data=data, state=self) # type: ignore
self.dispatch("message", message) self.dispatch("message", message)
if self._messages is not None: self.store_message(channel, data)
self._messages.append(message)
# we ensure that the channel is either a TextChannel or Thread # we ensure that the channel is either a TextChannel or Thread
if channel and channel.__class__ in (TextChannel, Thread): if channel and channel.__class__ in (TextChannel, Thread):
channel.last_message_id = message.id # type: ignore channel.last_message_id = message.id # type: ignore
@ -1032,7 +1045,7 @@ class ConnectionState:
for emoji in before_stickers: for emoji in before_stickers:
self._stickers.pop(emoji.id, None) self._stickers.pop(emoji.id, None)
# guild won't be None here # guild won't be None here
guild.stickers = tuple(map(lambda d: self.store_sticker(guild, d), data["stickers"])) # type: ignore guild.stickers = tuple(map(lambda d: self.store_sticker(d), data["stickers"])) # type: ignore
self.dispatch("guild_stickers_update", guild, before_stickers, guild.stickers) self.dispatch("guild_stickers_update", guild, before_stickers, guild.stickers)
def _get_create_guild(self, data): def _get_create_guild(self, data):
@ -1403,11 +1416,6 @@ class ConnectionState:
if channel is not None: if channel is not None:
return channel return channel
def create_message(
self, *, channel: Union[TextChannel, Thread, DMChannel, GroupChannel, PartialMessageable], data: MessagePayload
) -> Message:
return Message(state=self, channel=channel, data=data)
class AutoShardedConnectionState(ConnectionState): class AutoShardedConnectionState(ConnectionState):
def __init__(self, *args: Any, **kwargs: Any) -> None: def __init__(self, *args: Any, **kwargs: Any) -> None:

View File

@ -24,7 +24,7 @@ DEALINGS IN THE SOFTWARE.
from __future__ import annotations from __future__ import annotations
from typing import Any, Callable, Optional, TYPE_CHECKING, Tuple, Type, TypeVar, Union from typing import Callable, Optional, TYPE_CHECKING, Tuple, Type, TypeVar, Union
import inspect import inspect
import os import os
@ -60,7 +60,7 @@ class Button(Item[V]):
The ID of the button that gets received during an interaction. The ID of the button that gets received during an interaction.
If this button is for a URL, it does not have a custom ID. If this button is for a URL, it does not have a custom ID.
url: Optional[:class:`str`] url: Optional[:class:`str`]
The URL this button sends you to. This param is automatically casted to :class:`str`. The URL this button sends you to.
disabled: :class:`bool` disabled: :class:`bool`
Whether the button is disabled or not. Whether the button is disabled or not.
label: Optional[:class:`str`] label: Optional[:class:`str`]
@ -91,7 +91,7 @@ class Button(Item[V]):
label: Optional[str] = None, label: Optional[str] = None,
disabled: bool = False, disabled: bool = False,
custom_id: Optional[str] = None, custom_id: Optional[str] = None,
url: Optional[Any] = None, url: Optional[str] = None,
emoji: Optional[Union[str, Emoji, PartialEmoji]] = None, emoji: Optional[Union[str, Emoji, PartialEmoji]] = None,
row: Optional[int] = None, row: Optional[int] = None,
): ):
@ -117,7 +117,7 @@ class Button(Item[V]):
self._underlying = ButtonComponent._raw_construct( self._underlying = ButtonComponent._raw_construct(
type=ComponentType.button, type=ComponentType.button,
custom_id=custom_id, custom_id=custom_id,
url=str(url) if url else None, url=url,
disabled=disabled, disabled=disabled,
label=label, label=label,
style=style, style=style,

View File

@ -72,7 +72,7 @@ class Select(Item[V]):
The placeholder text that is shown if nothing is selected, if any. The placeholder text that is shown if nothing is selected, if any.
min_values: :class:`int` min_values: :class:`int`
The minimum number of items that must be chosen for this select menu. The minimum number of items that must be chosen for this select menu.
Defaults to 1 and must be between 0 and 25. Defaults to 1 and must be between 1 and 25.
max_values: :class:`int` max_values: :class:`int`
The maximum number of items that must be chosen for this select menu. The maximum number of items that must be chosen for this select menu.
Defaults to 1 and must be between 1 and 25. Defaults to 1 and must be between 1 and 25.
@ -327,7 +327,7 @@ def select(
ordering. The row number must be between 0 and 4 (i.e. zero indexed). ordering. The row number must be between 0 and 4 (i.e. zero indexed).
min_values: :class:`int` min_values: :class:`int`
The minimum number of items that must be chosen for this select menu. The minimum number of items that must be chosen for this select menu.
Defaults to 1 and must be between 0 and 25. Defaults to 1 and must be between 1 and 25.
max_values: :class:`int` max_values: :class:`int`
The maximum number of items that must be chosen for this select menu. The maximum number of items that must be chosen for this select menu.
Defaults to 1 and must be between 1 and 25. Defaults to 1 and must be between 1 and 25.

File diff suppressed because it is too large Load Diff

View File

@ -49,7 +49,7 @@ autodoc_typehints = "none"
# napoleon_attr_annotations = False # napoleon_attr_annotations = False
extlinks = { extlinks = {
"issue": ("https://github.com/iDevision/enhanced-discord.py/issues/%s", "GH-"), "issue": ("https://github.com/Rapptz/discord.py/issues/%s", "GH-"),
} }
# Links used for cross-referencing stuff in other documentation # Links used for cross-referencing stuff in other documentation
@ -168,8 +168,9 @@ html_context = {
resource_links = { resource_links = {
"discord": "https://discord.gg/TvqYBrGXEm", "discord": "https://discord.gg/TvqYBrGXEm",
"issues": "https://github.com/iDevision/enhanced-discord.py/issues", "issues": "https://github.com/Rapptz/discord.py/issues",
"examples": f"https://github.com/iDevision/enhanced-discord.py/tree/{branch}/examples", "discussions": "https://github.com/Rapptz/discord.py/discussions",
"examples": f"https://github.com/Rapptz/discord.py/tree/{branch}/examples",
} }
# Theme options are theme-specific and customize the look and feel of a theme # Theme options are theme-specific and customize the look and feel of a theme

Binary file not shown.

Before

Width:  |  Height:  |  Size: 52 KiB

View File

@ -38,6 +38,7 @@ If you're having trouble with something, these resources might help.
- Ask us and hang out with us in our :resource:`Discord <discord>` server. - Ask us and hang out with us in our :resource:`Discord <discord>` server.
- If you're looking for something specific, try the :ref:`index <genindex>` or :ref:`searching <search>`. - If you're looking for something specific, try the :ref:`index <genindex>` or :ref:`searching <search>`.
- Report bugs in the :resource:`issue tracker <issues>`. - Report bugs in the :resource:`issue tracker <issues>`.
- Ask in our :resource:`GitHub discussions page <discussions>`.
Extensions Extensions
------------ ------------