diff --git a/src/Discord/Parts/User/Member.php b/src/Discord/Parts/User/Member.php index 803e5f7d3..675a4566d 100644 --- a/src/Discord/Parts/User/Member.php +++ b/src/Discord/Parts/User/Member.php @@ -249,13 +249,14 @@ public function moveMember($channel, ?string $reason = null): ExtendedPromiseInt * * @param Role|string $role The role to add to the member. * @param string|null $reason Reason for Audit Log. + * @param bool|null $patch Whether to set the roles using PATCH instead of PUT and return the updated member part on resolved promise. * * @throws \RuntimeException * @throws NoPermissionsException Missing manage_roles permission. * - * @return ExtendedPromiseInterface + * @return ExtendedPromiseInterface|ExtendedPromiseInterface */ - public function addRole($role, ?string $reason = null): ExtendedPromiseInterface + public function addRole($role, ?string $reason = null, bool $patch = false): ExtendedPromiseInterface { if ($role instanceof Role) { $role = $role->id; @@ -279,12 +280,19 @@ public function addRole($role, ?string $reason = null): ExtendedPromiseInterface $headers['X-Audit-Log-Reason'] = $reason; } - return $this->http->put(Endpoint::bind(Endpoint::GUILD_MEMBER_ROLE, $this->guild_id, $this->id, $role), null, $headers) - ->then(function () use ($role) { - if (! in_array($role, $this->attributes['roles'])) { - $this->attributes['roles'][] = $role; - } - }); + return $patch + ? $this->http->patch(Endpoint::bind(Endpoint::GUILD_MEMBER, $this->guild_id, $this->id), ['roles' => array_unique(array_merge($this->attributes['roles'], [$role]))], $headers) + ->then(function ($response) { + $this->attributes['roles'] = $response->roles; + + return $this; + }) + : $this->http->put(Endpoint::bind(Endpoint::GUILD_MEMBER_ROLE, $this->guild_id, $this->id, $role), null, $headers) + ->then(function () use ($role) { + if (! in_array($role, $this->attributes['roles'])) { + $this->attributes['roles'][] = $role; + } + }); } /** @@ -294,12 +302,13 @@ public function addRole($role, ?string $reason = null): ExtendedPromiseInterface * * @param Role|string $role The role to remove from the member. * @param string|null $reason Reason for Audit Log. + * @param bool|null $patch Whether to set the roles using PATCH instead of DELETE and return the updated member part on resolved promise. * * @throws NoPermissionsException Missing manage_roles permission. * - * @return ExtendedPromiseInterface + * @return ExtendedPromiseInterface|ExtendedPromiseInterface */ - public function removeRole($role, ?string $reason = null): ExtendedPromiseInterface + public function removeRole($role, ?string $reason = null, bool $patch = false): ExtendedPromiseInterface { if ($role instanceof Role) { $role = $role->id; @@ -318,12 +327,19 @@ public function removeRole($role, ?string $reason = null): ExtendedPromiseInterf $headers['X-Audit-Log-Reason'] = $reason; } - return $this->http->delete(Endpoint::bind(Endpoint::GUILD_MEMBER_ROLE, $this->guild_id, $this->id, $role), null, $headers) - ->then(function () use ($role) { - if (($removeRole = array_search($role, $this->attributes['roles'])) !== false) { - unset($this->attributes['roles'][$removeRole]); - } - }); + return $patch + ? $this->http->patch(Endpoint::bind(Endpoint::GUILD_MEMBER, $this->guild_id, $this->id), ['roles' => array_diff($this->attributes['roles'], [$role])], $headers) + ->then(function ($response) { + $this->attributes['roles'] = $response->roles; + + return $this; + }) + : $this->http->delete(Endpoint::bind(Endpoint::GUILD_MEMBER_ROLE, $this->guild_id, $this->id, $role), null, $headers) + ->then(function () use ($role) { + if (($removeRole = array_search($role, $this->attributes['roles'])) !== false) { + unset($this->attributes['roles'][$removeRole]); + } + }); } /**