diff --git a/discord/__init__.py b/discord/__init__.py index 87174fe15..08c53d341 100644 --- a/discord/__init__.py +++ b/discord/__init__.py @@ -40,6 +40,7 @@ from .colour import * from .integrations import * from .invite import * from .template import * +from .welcome_screen import * from .widget import * from .object import * from .reaction import * diff --git a/discord/guild.py b/discord/guild.py index 65259e607..b3cb226ef 100644 --- a/discord/guild.py +++ b/discord/guild.py @@ -86,6 +86,7 @@ from .sticker import GuildSticker from .file import File from .audit_logs import AuditLogEntry from .object import OLDEST_OBJECT, Object +from .welcome_screen import WelcomeScreen, WelcomeChannel __all__ = ( @@ -3137,6 +3138,77 @@ class Guild(Hashable): return roles + async def welcome_screen(self) -> WelcomeScreen: + """|coro| + + Returns the guild's welcome screen. + + The guild must have ``COMMUNITY`` in :attr:`~Guild.features`. + + You must have the :attr:`~Permissions.manage_guild` permission to use + this as well. + + .. versionadded:: 2.0 + + Raises + ------- + Forbidden + You do not have the proper permissions to get this. + HTTPException + Retrieving the welcome screen failed. + + Returns + -------- + :class:`WelcomeScreen` + The welcome screen. + """ + data = await self._state.http.get_welcome_screen(self.id) + return WelcomeScreen(data=data, guild=self) + + async def edit_welcome_screen( + self, + *, + description: str = MISSING, + welcome_channels: List[WelcomeChannel] = MISSING, + enabled: bool = MISSING, + reason: Optional[str] = None, + ) -> WelcomeScreen: + """|coro| + + A shorthand method of :attr:`WelcomeScreen.edit` without needing + to fetch the welcome screen beforehand. + + The guild must have ``COMMUNITY`` in :attr:`~Guild.features`. + + You must have the :attr:`~Permissions.manage_guild` permission to use + this as well. + + .. versionadded:: 2.0 + + Returns + -------- + :class:`WelcomeScreen` + The edited welcome screen. + """ + fields = {} + + if welcome_channels is not MISSING: + welcome_channels_serialised = [] + for wc in welcome_channels: + if not isinstance(wc, WelcomeChannel): + raise TypeError('welcome_channels parameter must be a list of WelcomeChannel') + welcome_channels_serialised.append(wc.to_dict()) + fields['welcome_channels'] = welcome_channels_serialised + + if description is not MISSING: + fields['description'] = description + + if enabled is not MISSING: + fields['enabled'] = enabled + + data = await self._state.http.edit_welcome_screen(self.id, reason=reason, **fields) + return WelcomeScreen(data=data, guild=self) + async def kick(self, user: Snowflake, *, reason: Optional[str] = None) -> None: """|coro| diff --git a/discord/http.py b/discord/http.py index c9909e71b..cc795eeb5 100644 --- a/discord/http.py +++ b/discord/http.py @@ -87,6 +87,7 @@ if TYPE_CHECKING: threads, scheduled_event, sticker, + welcome_screen, ) from .types.snowflake import Snowflake, SnowflakeList @@ -1211,6 +1212,22 @@ class HTTPClient: return self.request(Route('GET', '/guilds/{guild_id}/bans', guild_id=guild_id), params=params) + def get_welcome_screen(self, guild_id: Snowflake) -> Response[welcome_screen.WelcomeScreen]: + return self.request(Route('GET', '/guilds/{guild_id}/welcome-screen', guild_id=guild_id)) + + def edit_welcome_screen( + self, guild_id: Snowflake, *, reason: Optional[str] = None, **fields: Any + ) -> Response[welcome_screen.WelcomeScreen]: + valid_keys = ( + 'description', + 'welcome_channels', + 'enabled', + ) + payload = {k: v for k, v in fields.items() if k in valid_keys} + return self.request( + Route('PATCH', '/guilds/{guild_id}/welcome-screen', guild_id=guild_id), json=payload, reason=reason + ) + def get_ban(self, user_id: Snowflake, guild_id: Snowflake) -> Response[guild.Ban]: return self.request(Route('GET', '/guilds/{guild_id}/bans/{user_id}', guild_id=guild_id, user_id=user_id)) diff --git a/discord/welcome_screen.py b/discord/welcome_screen.py new file mode 100644 index 000000000..abec0ff56 --- /dev/null +++ b/discord/welcome_screen.py @@ -0,0 +1,214 @@ +# -*- coding: utf-8 -*- + +""" +The MIT License (MIT) + +Copyright (c) 2015-present Rapptz + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS +OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. +""" + +from __future__ import annotations + +from typing import List, Optional, TYPE_CHECKING, Union +from .utils import _get_as_snowflake, get, MISSING +from .partial_emoji import _EmojiTag + +__all__ = ( + 'WelcomeChannel', + 'WelcomeScreen', +) + +if TYPE_CHECKING: + from typing_extensions import Self + from .types.welcome_screen import ( + WelcomeScreen as WelcomeScreenPayload, + WelcomeScreenChannel as WelcomeScreenChannelPayload, + ) + from .abc import Snowflake + from .guild import Guild + from .partial_emoji import PartialEmoji + from .emoji import Emoji + + +class WelcomeChannel: + """Represents a :class:`WelcomeScreen` welcome channel. + + .. versionadded:: 2.0 + + Attributes + ----------- + channel: :class:`abc.Snowflake` + The guild channel that is being referenced. + description: :class:`str` + The description shown of the channel. + emoji: Optional[:class:`PartialEmoji`, :class:`Emoji`, :class:`str`] + The emoji used beside the channel description. + """ + + def __init__(self, *, channel: Snowflake, description: str, emoji: Optional[Union[PartialEmoji, Emoji, str]] = None): + self.channel = channel + self.description = description + self.emoji = emoji + + def __repr__(self) -> str: + return f'' + + @classmethod + def _from_dict(cls, *, data: WelcomeScreenChannelPayload, guild: Guild) -> WelcomeChannel: + channel_id = int(data['channel_id']) + + channel = guild.get_channel(channel_id) + + description = data['description'] + _emoji_id = _get_as_snowflake(data, 'emoji_id') + _emoji_name = data['emoji_name'] + + if _emoji_id: + # custom + emoji = get(guild.emojis, id=_emoji_id) + else: + # unicode or None + emoji = _emoji_name + + return cls(channel=channel, description=description, emoji=emoji) # type: ignore + + def to_dict(self) -> WelcomeScreenChannelPayload: + ret: WelcomeScreenChannelPayload = { + 'channel_id': self.channel.id, + 'description': self.description, + 'emoji_id': None, + 'emoji_name': None, + } + + if isinstance(self.emoji, _EmojiTag): + ret['emoji_id'] = self.emoji.id + ret['emoji_name'] = self.emoji.name + else: + # unicode or None + ret['emoji_name'] = self.emoji + + return ret + + +class WelcomeScreen: + """Represents a :class:`Guild` welcome screen. + + .. versionadded:: 2.0 + + Attributes + ----------- + description: :class:`str` + The description shown on the welcome screen. + welcome_channels: List[:class:`WelcomeChannel`] + The channels shown on the welcome screen. + """ + + def __init__(self, *, data: WelcomeScreenPayload, guild: Guild): + self._state = guild._state + self._guild = guild + self._store(data) + + def _store(self, data: WelcomeScreenPayload) -> None: + self.description = data['description'] + welcome_channels = data.get('welcome_channels', []) + self.welcome_channels = [WelcomeChannel._from_dict(data=wc, guild=self._guild) for wc in welcome_channels] + + def __repr__(self) -> str: + return f'' + + @property + def enabled(self) -> bool: + """:class:`bool`: Whether the welcome screen is displayed. + + This is equivalent to checking if ``WELCOME_SCREEN_ENABLED`` + is present in :attr:`Guild.features`. + """ + return 'WELCOME_SCREEN_ENABLED' in self._guild.features + + async def edit( + self, + *, + description: str = MISSING, + welcome_channels: List[WelcomeChannel] = MISSING, + enabled: bool = MISSING, + reason: Optional[str] = None, + ) -> Self: + """|coro| + + Edit the welcome screen. + + Welcome channels can only accept custom emojis if :attr:`Guild.premium_tier` is level 2 or above. + + You must have the :attr:`~Permissions.manage_guild` permission in the + guild to do this. + + Usage: :: + + rules_channel = guild.get_channel(12345678) + announcements_channel = guild.get_channel(87654321) + + custom_emoji = utils.get(guild.emojis, name='loudspeaker') + + await welcome_screen.edit( + description='This is a very cool community server!', + welcome_channels=[ + WelcomeChannel(channel=rules_channel, description='Read the rules!', emoji='👨‍🏫'), + WelcomeChannel(channel=announcements_channel, description='Watch out for announcements!', emoji=custom_emoji), + ] + ) + + Parameters + ------------ + description: Optional[:class:`str`] + The template's description. + welcome_channels: Optional[List[:class:`WelcomeChannel`]] + The welcome channels, in their respective order. + enabled: Optional[:class:`bool`] + Whether the welcome screen should be displayed. + reason: Optional[:class:`str`] + The reason for editing the welcome screen. Shows up on the audit log. + + Raises + ------- + HTTPException + Editing the welcome screen failed failed. + Forbidden + You don't have permissions to edit the welcome screen. + NotFound + This welcome screen does not exist. + """ + fields = {} + + if welcome_channels is not MISSING: + welcome_channels_serialised = [] + for wc in welcome_channels: + if not isinstance(wc, WelcomeChannel): + raise TypeError('welcome_channels parameter must be a list of WelcomeChannel') + welcome_channels_serialised.append(wc.to_dict()) + fields['welcome_channels'] = welcome_channels_serialised + + if description is not MISSING: + fields['description'] = description + + if enabled is not MISSING: + fields['enabled'] = enabled + + data = await self._state.http.edit_welcome_screen(self._guild.id, reason=reason, **fields) + return WelcomeScreen(data=data, guild=self._guild) diff --git a/docs/api.rst b/docs/api.rst index 61ea91404..cfbaedfe9 100644 --- a/docs/api.rst +++ b/docs/api.rst @@ -3982,6 +3982,22 @@ Template .. autoclass:: Template() :members: +WelcomeScreen +~~~~~~~~~~~~~~~ + +.. attributetable:: WelcomeScreen + +.. autoclass:: WelcomeScreen() + :members: + +WelcomeChannel +~~~~~~~~~~~~~~~ + +.. attributetable:: WelcomeChannel + +.. autoclass:: WelcomeChannel() + :members: + WidgetChannel ~~~~~~~~~~~~~~~