diff --git a/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IPartialUser.cs b/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IPartialUser.cs
index 81871241fd..9e9b2f5c7e 100644
--- a/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IPartialUser.cs
+++ b/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IPartialUser.cs
@@ -44,6 +44,9 @@ public interface IPartialUser
///
Optional Avatar { get; }
+ ///
+ Optional AvatarDecoration { get; }
+
///
Optional IsBot { get; }
diff --git a/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IUser.cs b/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IUser.cs
index ed75ee73a7..76e490c8ed 100644
--- a/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IUser.cs
+++ b/Backend/Remora.Discord.API.Abstractions/API/Objects/Users/IUser.cs
@@ -52,6 +52,11 @@ public interface IUser : IPartialUser
///
new IImageHash? Avatar { get; }
+ ///
+ /// Gets the user's avatar decoration.
+ ///
+ new Optional AvatarDecoration { get; }
+
///
/// Gets a value indicating whether the user is a bot, belonging to an OAuth2 application.
///
@@ -120,6 +125,9 @@ public interface IUser : IPartialUser
///
Optional IPartialUser.Avatar => new(this.Avatar);
+ ///
+ Optional IPartialUser.AvatarDecoration => this.AvatarDecoration;
+
///
Optional IPartialUser.IsBot => this.IsBot;
diff --git a/Backend/Remora.Discord.API/API/CDN.cs b/Backend/Remora.Discord.API/API/CDN.cs
index e22385cac2..59f06d0e5b 100644
--- a/Backend/Remora.Discord.API/API/CDN.cs
+++ b/Backend/Remora.Discord.API/API/CDN.cs
@@ -409,6 +409,75 @@ public static Result GetGuildBannerUrl
return ub.Uri;
}
+ ///
+ /// Gets the CDN URI of the given user's avatar decoration.
+ ///
+ /// The user.
+ /// The requested image format.
+ /// The requested image size. May be any power of two between 16 and 4096.
+ /// A result which may or may not have succeeded.
+ public static Result GetUserAvatarDecorationUrl
+ (
+ IUser user,
+ Optional imageFormat = default,
+ Optional imageSize = default
+ )
+ {
+ return user.AvatarDecoration.IsDefined(out var decoration)
+ ? GetUserAvatarDecorationUrl(user.ID, decoration, imageFormat, imageSize)
+ : new ImageNotFoundError();
+ }
+
+ ///
+ /// Gets the CDN URI of the given user's avatar decoration.
+ ///
+ /// The ID of the user.
+ /// The image hash of the user's avatar decoration.
+ /// The requested image format.
+ /// The requested image size. May be any power of two between 16 and 4096.
+ /// A result which may or may not have succeeded.
+ public static Result GetUserAvatarDecorationUrl
+ (
+ Snowflake userID,
+ IImageHash avatarDecorationHash,
+ Optional imageFormat = default,
+ Optional imageSize = default
+ )
+ {
+ var formatValidation = ValidateOrDefaultImageFormat
+ (
+ imageFormat,
+ CDNImageFormat.PNG,
+ CDNImageFormat.JPEG,
+ CDNImageFormat.WebP
+ );
+
+ if (!formatValidation.IsSuccess)
+ {
+ return Result.FromError(formatValidation);
+ }
+
+ var format = formatValidation.Entity;
+
+ var checkImageSize = CheckImageSize(imageSize);
+ if (!checkImageSize.IsSuccess)
+ {
+ return Result.FromError(checkImageSize);
+ }
+
+ var ub = new UriBuilder(Constants.CDNBaseURL)
+ {
+ Path = $"avatar-decorations/{userID}/{avatarDecorationHash.Value}.{format.ToFileExtension()}"
+ };
+
+ if (imageSize.HasValue)
+ {
+ ub.Query = $"size={imageSize.Value}";
+ }
+
+ return ub.Uri;
+ }
+
///
/// Gets the CDN URI of the given user's banner.
///
diff --git a/Backend/Remora.Discord.API/API/Gateway/Events/Users/UserUpdate.cs b/Backend/Remora.Discord.API/API/Gateway/Events/Users/UserUpdate.cs
index 02ae48edfe..291cdbc14a 100644
--- a/Backend/Remora.Discord.API/API/Gateway/Events/Users/UserUpdate.cs
+++ b/Backend/Remora.Discord.API/API/Gateway/Events/Users/UserUpdate.cs
@@ -36,6 +36,7 @@ public record UserUpdate
string Username,
ushort Discriminator,
IImageHash? Avatar,
+ Optional AvatarDecoration = default,
Optional IsBot = default,
Optional IsSystem = default,
Optional IsMFAEnabled = default,
diff --git a/Backend/Remora.Discord.API/API/Objects/Users/PartialUser.cs b/Backend/Remora.Discord.API/API/Objects/Users/PartialUser.cs
index cf19542db6..0387f7d461 100644
--- a/Backend/Remora.Discord.API/API/Objects/Users/PartialUser.cs
+++ b/Backend/Remora.Discord.API/API/Objects/Users/PartialUser.cs
@@ -37,6 +37,7 @@ public record PartialUser
Optional Username = default,
Optional Discriminator = default,
Optional Avatar = default,
+ Optional AvatarDecoration = default,
Optional IsBot = default,
Optional IsSystem = default,
Optional IsMFAEnabled = default,
diff --git a/Backend/Remora.Discord.API/API/Objects/Users/User.cs b/Backend/Remora.Discord.API/API/Objects/Users/User.cs
index d5440700ea..0e5630d11d 100644
--- a/Backend/Remora.Discord.API/API/Objects/Users/User.cs
+++ b/Backend/Remora.Discord.API/API/Objects/Users/User.cs
@@ -37,6 +37,7 @@ public record User
string Username,
ushort Discriminator,
IImageHash? Avatar,
+ Optional AvatarDecoration = default,
Optional IsBot = default,
Optional IsSystem = default,
Optional IsMFAEnabled = default,
diff --git a/Backend/Remora.Discord.API/API/Objects/Users/UserMention.cs b/Backend/Remora.Discord.API/API/Objects/Users/UserMention.cs
index ad7774ff6c..80ea9ebaf2 100644
--- a/Backend/Remora.Discord.API/API/Objects/Users/UserMention.cs
+++ b/Backend/Remora.Discord.API/API/Objects/Users/UserMention.cs
@@ -35,6 +35,7 @@ public record UserMention
string Username,
ushort Discriminator,
IImageHash? Avatar,
+ Optional AvatarDecoration = default,
Optional IsBot = default,
Optional IsSystem = default,
Optional IsMFAEnabled = default,
@@ -53,6 +54,7 @@ public record UserMention
Username,
Discriminator,
Avatar,
+ AvatarDecoration,
IsBot,
IsSystem,
IsMFAEnabled,
diff --git a/Tests/Remora.Discord.Tests/Samples/Objects/USER/USER.json b/Tests/Remora.Discord.Tests/Samples/Objects/USER/USER.json
index 88dc232b74..f67cc0322f 100644
--- a/Tests/Remora.Discord.Tests/Samples/Objects/USER/USER.json
+++ b/Tests/Remora.Discord.Tests/Samples/Objects/USER/USER.json
@@ -3,6 +3,7 @@
"username": "none",
"discriminator": "9999",
"avatar": "68b329da9893e34099c7d8ad5cb9c940",
+ "avatar_decoration": "68b329da9893e34099c7d8ad5cb9c940",
"bot": false,
"system": false,
"mfa_enabled": false,
diff --git a/Tests/Remora.Discord.Tests/Samples/Objects/USER_MENTION/USER_MENTION.json b/Tests/Remora.Discord.Tests/Samples/Objects/USER_MENTION/USER_MENTION.json
index bf212d578b..5dc21ddaa0 100644
--- a/Tests/Remora.Discord.Tests/Samples/Objects/USER_MENTION/USER_MENTION.json
+++ b/Tests/Remora.Discord.Tests/Samples/Objects/USER_MENTION/USER_MENTION.json
@@ -3,6 +3,7 @@
"username": "none",
"discriminator": "9999",
"avatar": "68b329da9893e34099c7d8ad5cb9c940",
+ "avatar_decoration": "68b329da9893e34099c7d8ad5cb9c940",
"bot": false,
"system": false,
"mfa_enabled": false,