# frozen_string_literal: true module Discorb # # Represents a member of a guild. # class Member < User # @return [Time] The time the member boosted the guild. attr_reader :premium_since # @return [String] The nickname of the member. # @return [nil] If the member has no nickname. attr_reader :nick # @return [Time] The time the member joined the guild. attr_reader :joined_at # @return [Discorb::Asset] The custom avatar of the member. # @return [nil] If the member has no custom avatar. attr_reader :custom_avatar # @return [Discorb::Asset] The display avatar of the member. attr_reader :display_avatar # @return [Boolean] Whether the member is muted. attr_reader :mute alias mute? mute # @return [Boolean] Whether the member is deafened. attr_reader :deaf alias deaf? deaf # @return [Boolean] Whether the member is pending (Not passed member screening). attr_reader :pending alias pending? pending # @!attribute [r] name # @return [String] The display name of the member. # @!attribute [r] mention # @return [String] The mention of the member. # @!attribute [r] voice_state # @return [Discorb::VoiceState] The voice state of the member. # @!attribute [r] roles # @macro client_cache # @return [Array] The roles of the member. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild the member is in. # @!attribute [r] hoisted_role # @macro client_cache # @return [Discorb::Role] The hoisted role of the member. # @return [nil] If the member has no hoisted role. # @!attribute [r] hoisted? # @return [Boolean] Whether the member has a hoisted role. # @!attribute [r] permissions # @return [Discorb::Permission] The permissions of the member. # @!attribute [r] presence # @macro client_cache # @return [Discorb::Presence] The presence of the member. # @!attribute [r] activity # @macro client_cache # @return [Discorb::Activity] The activity of the member. It's from the {#presence}. # @!attribute [r] activities # @macro client_cache # @return [Array] The activities of the member. It's from the {#presence}. # @!attribute [r] status # @macro client_cache # @return [Symbol] The status of the member. It's from the {#presence}. # @!visibility private def initialize(client, guild_id, user_data, member_data) @guild_id = guild_id @client = client @_member_data = {} @data = {} _set_data(user_data, member_data) end # # Format the member to `@name` style. # # @return [String] The formatted member. # def to_s "@#{name}" end # # Format the member to `Username#Discriminator` style. # # @return [String] The formatted member. # def to_s_user "#{username}##{discriminator}" end def name @nick || @username end def mention "<@#{@nick.nil? ? "" : "!"}#{@id}>" end def voice_state guild.voice_states[@id] end def guild @client.guilds[@guild_id] end def roles @role_ids.map { |r| guild.roles[r] } end def permissions roles.map(&:permissions).sum(Permission.new(0)) end def hoisted_role @hoisted_role_id && guild.roles[@hoisted_role_id] end def hoisted? !@hoisted_role_id.nil? end def presence guild.presences[@id] end def activity presence&.activity end def activities presence&.activities end def status presence&.status end def inspect "#<#{self.class} #{self} id=#{@id}>" end # # Add a role to the member. # @macro http # @macro async # # @param [Discorb::Role] role The role to add. # @param [String] reason The reason for the action. # def add_role(role, reason: nil) Async do @client.http.put("/guilds/#{@guild_id}/members/#{@id}/roles/#{role.is_a?(Role) ? role.id : role}", nil, audit_log_reason: reason).wait end end # # Remove a role to the member. # @macro http # @macro async # # @param [Discorb::Role] role The role to add. # @param [String] reason The reason for the action. # def remove_role(role, reason: nil) Async do @client.http.delete("/guilds/#{@guild_id}/members/#{@id}/roles/#{role.is_a?(Role) ? role.id : role}", audit_log_reason: reason).wait end end # # Edit the member. # @macro http # @macro async # @macro edit # # @param [String] nick The nickname of the member. # @param [Discorb::Role] role The roles of the member. # @param [Boolean] mute Whether the member is muted. # @param [Boolean] deaf Whether the member is deafened. # @param [Discorb::StageChannel] channel The channel the member is moved to. # @param [String] reason The reason for the action. # def edit(nick: :unset, role: :unset, mute: :unset, deaf: :unset, channel: :unset, reason: nil) Async do payload = {} payload[:nick] = nick if nick != :unset payload[:roles] = role if role != :unset payload[:mute] = mute if mute != :unset payload[:deaf] = deaf if deaf != :unset payload[:channel_id] = channel&.id if channel != :unset @client.http.patch("/guilds/#{@guild_id}/members/#{@id}", payload, audit_log_reason: reason).wait end end alias modify edit # # Kick the member. # # @param [String] reason The reason for the action. # def kick(reason: nil) Async do guild.kick_member(self, reason: reason).wait end end # # Ban the member. # # @param [Integer] delete_message_days The number of days to delete messages. # @param [String] reason The reason for the action. # # @return [Async::Task] The ban. # def ban(delete_message_days: 0, reason: nil) Async do guild.ban_member(self, delete_message_days: delete_message_days, reason: reason).wait end end private def _set_data(user_data, member_data) user_data ||= member_data[:user] @role_ids = member_data[:roles] @premium_since = member_data[:premium_since] && Time.iso8601(member_data[:premium_since]) @pending = member_data[:pending] @nick = member_data[:nick] @mute = member_data[:mute] @joined_at = member_data[:joined_at] && Time.iso8601(member_data[:joined_at]) @hoisted_role_id = member_data[:hoisted_role] @deaf = member_data[:deaf] @custom_avatar = member_data[:avatar] && Asset.new(member_data[:avatar]) super(user_data) @display_avatar = @avatar || @custom_avatar @client.guilds[@guild_id].members[@id] = self unless @guild_id.nil? @_member_data.update(member_data) end end end