From d2a55ea6323ae7f701900cf6c307b6697783a220 Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Thu, 18 Jul 2024 09:21:08 +0200 Subject: [PATCH 1/5] implement member banners --- changelog/1203.feature.rst | 1 + disnake/asset.py | 13 +++++++++++++ disnake/member.py | 13 +++++++++++++ 3 files changed, 27 insertions(+) create mode 100644 changelog/1203.feature.rst diff --git a/changelog/1203.feature.rst b/changelog/1203.feature.rst new file mode 100644 index 0000000000..d4412fdf31 --- /dev/null +++ b/changelog/1203.feature.rst @@ -0,0 +1 @@ +Implement new :attr:`.Member.guild_banner` property. diff --git a/disnake/asset.py b/disnake/asset.py index edb0d1c7a6..cf3cf5ac66 100644 --- a/disnake/asset.py +++ b/disnake/asset.py @@ -237,6 +237,19 @@ def _from_guild_avatar( animated=animated, ) + @classmethod + def _from_guild_banner( + cls, state: AnyState, guild_id: int, member_id: int, banner: str + ) -> Self: + animated = banner.startswith("a_") + format = "gif" if animated else "png" + return cls( + state, + url=f"{cls.BASE}/guilds/{guild_id}/users/{member_id}/banners/{banner}.{format}?size=1024", + key=banner, + animated=animated, + ) + @classmethod def _from_icon(cls, state: AnyState, object_id: int, icon_hash: str, path: str) -> Self: return cls( diff --git a/disnake/member.py b/disnake/member.py index 149fc97ecc..3ec71ee2f6 100644 --- a/disnake/member.py +++ b/disnake/member.py @@ -272,6 +272,7 @@ class Member(disnake.abc.Messageable, _UserTag): "_user", "_state", "_avatar", + "_banner", "_communication_disabled_until", "_flags", "_avatar_decoration_data", @@ -340,6 +341,7 @@ def __init__( self.nick: Optional[str] = data.get("nick") self.pending: bool = data.get("pending", False) self._avatar: Optional[str] = data.get("avatar") + self._banner: Optional[str] = data.get("banner") timeout_datetime = utils.parse_time(data.get("communication_disabled_until")) self._communication_disabled_until: Optional[datetime.datetime] = timeout_datetime self._flags: int = data.get("flags", 0) @@ -619,6 +621,17 @@ def guild_avatar(self) -> Optional[Asset]: return None return Asset._from_guild_avatar(self._state, self.guild.id, self.id, self._avatar) + @property + def guild_banner(self) -> Optional[Asset]: + """Optional[:class:`Asset`]: Returns an :class:`Asset` for the guild banner + the member has. If unavailable, ``None`` is returned. + + .. versionadded:: 2.10 + """ + if self._banner is None: + return None + return Asset._from_guild_banner(self._state, self.guild.id, self.id, self._banner) + @property def activity(self) -> Optional[ActivityTypes]: """Optional[Union[:class:`BaseActivity`, :class:`Spotify`]]: Returns the primary From c205924554b51620878cf2066737b9f485fb927d Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Sun, 21 Jul 2024 18:55:46 +0200 Subject: [PATCH 2/5] copy and update member banner in the relative methods --- disnake/member.py | 2 ++ disnake/types/gateway.py | 1 + 2 files changed, 3 insertions(+) diff --git a/disnake/member.py b/disnake/member.py index 3ec71ee2f6..1371f17975 100644 --- a/disnake/member.py +++ b/disnake/member.py @@ -411,6 +411,7 @@ def _copy(cls, member: Member) -> Self: self.activities = member.activities self._state = member._state self._avatar = member._avatar + self._banner = member._banner self._communication_disabled_until = member.current_timeout self._flags = member._flags @@ -439,6 +440,7 @@ def _update(self, data: GuildMemberUpdateEvent) -> None: self.premium_since = utils.parse_time(data.get("premium_since")) self._roles = utils.SnowflakeList(map(int, data["roles"])) self._avatar = data.get("avatar") + self._banner = data.get("banner") timeout_datetime = utils.parse_time(data.get("communication_disabled_until")) self._communication_disabled_until = timeout_datetime self._flags = data.get("flags", 0) diff --git a/disnake/types/gateway.py b/disnake/types/gateway.py index e2494848b7..d0ddc2f2a0 100644 --- a/disnake/types/gateway.py +++ b/disnake/types/gateway.py @@ -434,6 +434,7 @@ class GuildMemberUpdateEvent(TypedDict): user: User nick: NotRequired[Optional[str]] avatar: Optional[str] + banner: Optional[str] joined_at: Optional[str] premium_since: NotRequired[Optional[str]] deaf: NotRequired[bool] From bd2c32afabb352def5023db34c057f269f4e9639 Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Sun, 21 Jul 2024 19:02:06 +0200 Subject: [PATCH 3/5] implement display_banner --- disnake/member.py | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/disnake/member.py b/disnake/member.py index 1371f17975..c8bf001833 100644 --- a/disnake/member.py +++ b/disnake/member.py @@ -623,6 +623,18 @@ def guild_avatar(self) -> Optional[Asset]: return None return Asset._from_guild_avatar(self._state, self.guild.id, self.id, self._avatar) + @property + def display_banner(self) -> Optional[Asset]: + """Optional[:class:`Asset`]: Returns the member's display banner. + + For regular members this is just their banner, but + if they have a guild specific avatar then that + is returned instead. + + .. versionadded:: 2.10 + """ + return self.guild_banner or self._user.banner + @property def guild_banner(self) -> Optional[Asset]: """Optional[:class:`Asset`]: Returns an :class:`Asset` for the guild banner From 1054c4bb11533a08a88f02c56607219b26a41c7d Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Sun, 21 Jul 2024 19:30:00 +0200 Subject: [PATCH 4/5] Update disnake/member.py Co-authored-by: shiftinv <8530778+shiftinv@users.noreply.github.com> Signed-off-by: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> --- disnake/member.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/disnake/member.py b/disnake/member.py index c8bf001833..5499dc2f31 100644 --- a/disnake/member.py +++ b/disnake/member.py @@ -628,7 +628,7 @@ def display_banner(self) -> Optional[Asset]: """Optional[:class:`Asset`]: Returns the member's display banner. For regular members this is just their banner, but - if they have a guild specific avatar then that + if they have a guild specific banner then that is returned instead. .. versionadded:: 2.10 From 7b40755fd35938c208957817ec20cf323fd6c31a Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Fri, 15 Nov 2024 00:36:48 +0100 Subject: [PATCH 5/5] fix: remove display_avatar property and add TODO comment --- disnake/member.py | 16 ++++------------ 1 file changed, 4 insertions(+), 12 deletions(-) diff --git a/disnake/member.py b/disnake/member.py index 5499dc2f31..7ce63f8c9e 100644 --- a/disnake/member.py +++ b/disnake/member.py @@ -623,18 +623,10 @@ def guild_avatar(self) -> Optional[Asset]: return None return Asset._from_guild_avatar(self._state, self.guild.id, self.id, self._avatar) - @property - def display_banner(self) -> Optional[Asset]: - """Optional[:class:`Asset`]: Returns the member's display banner. - - For regular members this is just their banner, but - if they have a guild specific banner then that - is returned instead. - - .. versionadded:: 2.10 - """ - return self.guild_banner or self._user.banner - + # TODO + # implement a `display_banner` property + # for more info on why this wasn't implemented read this discussion + # https://github.com/DisnakeDev/disnake/pull/1204#discussion_r1685773429 @property def guild_banner(self) -> Optional[Asset]: """Optional[:class:`Asset`]: Returns an :class:`Asset` for the guild banner