All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.dv8tion.jda.api.entities.MessageReaction Maven / Gradle / Ivy

Go to download

Java wrapper for the popular chat & VOIP service: Discord https://discord.com

The newest version!
/*
 * Copyright 2015 Austin Keener, Michael Ritter, Florian Spieß, and the JDA contributors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package net.dv8tion.jda.api.entities;

import net.dv8tion.jda.api.JDA;
import net.dv8tion.jda.api.Permission;
import net.dv8tion.jda.api.entities.channel.ChannelType;
import net.dv8tion.jda.api.entities.channel.concrete.PrivateChannel;
import net.dv8tion.jda.api.entities.channel.middleman.GuildChannel;
import net.dv8tion.jda.api.entities.channel.middleman.GuildMessageChannel;
import net.dv8tion.jda.api.entities.channel.middleman.MessageChannel;
import net.dv8tion.jda.api.entities.channel.unions.GuildMessageChannelUnion;
import net.dv8tion.jda.api.entities.channel.unions.MessageChannelUnion;
import net.dv8tion.jda.api.entities.emoji.Emoji;
import net.dv8tion.jda.api.entities.emoji.EmojiUnion;
import net.dv8tion.jda.api.exceptions.InsufficientPermissionException;
import net.dv8tion.jda.api.exceptions.PermissionException;
import net.dv8tion.jda.api.requests.RestAction;
import net.dv8tion.jda.api.requests.Route;
import net.dv8tion.jda.api.requests.restaction.pagination.ReactionPaginationAction;
import net.dv8tion.jda.internal.requests.RestActionImpl;
import net.dv8tion.jda.internal.requests.restaction.pagination.ReactionPaginationActionImpl;
import net.dv8tion.jda.internal.utils.Checks;
import net.dv8tion.jda.internal.utils.EntityString;

import javax.annotation.CheckReturnValue;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.Objects;

/**
 * An object representing a single MessageReaction from Discord.
 * This is an immutable object and is not updated by method calls or changes in Discord. A new snapshot instance
 * built from Discord is needed to see changes.
 *
 * @see    Message#getReactions()
 * @see    Message#getReaction(Emoji)
 */
public class MessageReaction
{
    private final JDA jda;
    private final EmojiUnion emoji;
    private final MessageChannel channel;
    private final long channelId;
    private final long messageId;
    private final boolean[] self;
    private final int[] counts;

    /**
     * Creates a new MessageReaction instance
     *
     * @param  channel
     *         The {@link MessageChannel} this Reaction was used in
     * @param  emoji
     *         The {@link Emoji} that was used
     * @param  channelId
     *         The channel id for this reaction
     * @param  messageId
     *         The message id this reaction is attached to
     * @param  self
     *         Whether we already reacted with this Reaction,
     *         as an array of {@code [normal, super]}
     * @param  counts
     *         The amount of people that reacted with this Reaction,
     *         as an array of {@code [total, normal, super]}
     */
    public MessageReaction(@Nonnull JDA jda, @Nullable MessageChannel channel, @Nonnull EmojiUnion emoji, long channelId, long messageId, boolean[] self, int[] counts)
    {
        this.jda = jda;
        this.emoji = emoji;
        this.channel = channel;
        this.channelId = channelId;
        this.messageId = messageId;
        this.self = self;
        this.counts = counts;
    }

    /**
     * The JDA instance of this Reaction
     *
     * @return The JDA instance of this Reaction
     */
    @Nonnull
    public JDA getJDA()
    {
        return jda;
    }

    /**
     * Whether the currently logged in account has reacted with this reaction at all, including both super and normal.
     *
     * 

This will always be false for events. Discord does not provide this information for reaction events. * You can use {@link MessageChannel#retrieveMessageById(String)} to get this information on a complete message. * * @return True, if we reacted with this reaction * * @see #isSelf(ReactionType) */ public boolean isSelf() { return self[0] || self[1]; } /** * Whether the currently logged in account has reacted with this reaction as specifically a super or normal reaction. * *

This will always be false for events. Discord does not provide this information for reaction events. * You can use {@link MessageChannel#retrieveMessageById(String)} to get this information on a complete message. * * @param type * The specific type of reaction * * @return True, if we reacted with this reaction */ public boolean isSelf(@Nonnull ReactionType type) { Checks.notNull(type, "Type"); return self[type == ReactionType.NORMAL ? 0 : 1]; } /** * Whether this reaction can provide a count via {@link #getCount()}. *
This is usually not provided for reactions coming from {@link net.dv8tion.jda.api.events.message.react.MessageReactionAddEvent} * or similar. * * @return True, if a count is available * * @see #getCount() */ public boolean hasCount() { return counts != null; } /** * Whether this reaction instance has an available {@link #getChannel()}. * *

This can be {@code false} for messages sent via webhooks, or in the context of interactions. * * @return True, if {@link #getChannel()} is available */ public boolean hasChannel() { return channel != null; } /** * The total amount of users that already reacted with this Reaction. *
This is not updated, it is a {@code final int} per Reaction instance * *

This value is not available in events such as {@link net.dv8tion.jda.api.events.message.react.MessageReactionAddEvent MessageReactionAddEvent} * and {@link net.dv8tion.jda.api.events.message.react.MessageReactionRemoveEvent MessageReactionRemoveEvent} in which case an * {@link java.lang.IllegalStateException IllegalStateException} is thrown! * * @throws java.lang.IllegalStateException * If this MessageReaction is from an event which does not provide a count * * @return The amount of users that reacted with this Reaction * * @see #getCount(ReactionType) */ public int getCount() { if (!hasCount()) throw new IllegalStateException("Cannot retrieve count for this MessageReaction!"); return counts[0]; } /** * The specific amount of users that already reacted with this Reaction. *
This is not updated, it is a {@code final int} per Reaction instance * *

This value is not available in events such as {@link net.dv8tion.jda.api.events.message.react.MessageReactionAddEvent MessageReactionAddEvent} * and {@link net.dv8tion.jda.api.events.message.react.MessageReactionRemoveEvent MessageReactionRemoveEvent} in which case an * {@link java.lang.IllegalStateException IllegalStateException} is thrown! * * @param type * The specific type of reaction * * @throws java.lang.IllegalStateException * If this MessageReaction is from an event which does not provide a count * * @return The amount of users that reacted with this Reaction * * @see #getCount() */ public int getCount(@Nonnull ReactionType type) { if (!hasCount()) throw new IllegalStateException("Cannot retrieve count for this MessageReaction!"); Checks.notNull(type, "Type"); return counts[type == ReactionType.NORMAL ? 1 : 2]; } /** * The {@link ChannelType ChannelType} * this Reaction was used in. * * @return The ChannelType */ @Nonnull public ChannelType getChannelType() { return channel != null ? channel.getType() : ChannelType.UNKNOWN; } /** * Whether this Reaction was used in a {@link MessageChannel} * of the specified {@link ChannelType ChannelType}. * * @param type * The ChannelType to compare * * @return True, if this Reaction was used in a MessageChannel from the specified ChannelType */ public boolean isFromType(@Nonnull ChannelType type) { return getChannelType() == type; } /** * The {@link net.dv8tion.jda.api.entities.Guild Guild} this Reaction was used in. * * @throws IllegalStateException * If {@link #getChannel()} is not a guild channel or the channel is not provided * * @return {@link net.dv8tion.jda.api.entities.Guild Guild} this Reaction was used in */ @Nonnull public Guild getGuild() { return getGuildChannel().getGuild(); } /** * The {@link MessageChannel MessageChannel} * this Reaction was used in. * * @throws IllegalStateException * If no channel instance is provided, this might be missing for messages sent from webhooks. * * @return The channel this Reaction was used in */ @Nonnull public MessageChannelUnion getChannel() { if (channel != null) return (MessageChannelUnion) channel; throw new IllegalStateException("Cannot provide channel instance for this reaction! Use getChannelId() instead."); } /** * The {@link net.dv8tion.jda.api.entities.channel.middleman.GuildMessageChannel channel} this Reaction was used in. * * @throws IllegalStateException * If {@link #getChannel()} is not a guild channel or the channel is not provided * * @return The guild channel this Reaction was used in */ @Nonnull public GuildMessageChannelUnion getGuildChannel() { return (GuildMessageChannelUnion) getChannel().asGuildMessageChannel(); } /** * The ID for the channel this reaction happened in. * * @return The channel id */ public long getChannelIdLong() { return channelId; } /** * The ID for the channel this reaction happened in. * * @return The channel id */ @Nonnull public String getChannelId() { return Long.toUnsignedString(channelId); } /** * The {@link Emoji} of this Reaction. *
This includes both {@link Emoji.Type#CUSTOM custom emojis} and {@link Emoji.Type#UNICODE unicode emoji}. * * @return The final instance of this Reaction's Emoji */ @Nonnull public EmojiUnion getEmoji() { return emoji; } /** * The message id this reaction is attached to * * @return The message id this reaction is attached to */ @Nonnull public String getMessageId() { return Long.toUnsignedString(messageId); } /** * The message id this reaction is attached to * * @return The message id this reaction is attached to */ public long getMessageIdLong() { return messageId; } /** * Retrieves the {@link net.dv8tion.jda.api.entities.User Users} that * already reacted with this MessageReaction. * *

Possible ErrorResponses include: *

    *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_MESSAGE UNKNOWN_MESSAGE} *
    If the message this reaction was attached to got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_CHANNEL UNKNOWN_CHANNEL} *
    If the channel this reaction was used in got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#MISSING_ACCESS MISSING_ACCESS} *
    If we were removed from the channel/guild
  • *
* * @return {@link ReactionPaginationAction ReactionPaginationAction} */ @Nonnull @CheckReturnValue public ReactionPaginationAction retrieveUsers() { return new ReactionPaginationActionImpl(this); } /** * Removes this Reaction from the Message. *
This will remove our own reaction as an overload * of {@link #removeReaction(User)}. * *

Possible ErrorResponses include: *

    *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_MESSAGE UNKNOWN_MESSAGE} *
    If the message this reaction was attached to got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_CHANNEL UNKNOWN_CHANNEL} *
    If the channel this reaction was used in got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#MISSING_ACCESS MISSING_ACCESS} *
    If we were removed from the channel/guild
  • *
* * @return {@link net.dv8tion.jda.api.requests.RestAction RestAction} - Type: Void * Nothing is returned on success */ @Nonnull @CheckReturnValue public RestAction removeReaction() { return removeReaction(getJDA().getSelfUser()); } /** * Removes this Reaction from the Message. *
This will remove the reaction of the {@link net.dv8tion.jda.api.entities.User User} * provided. * *

If the provided User did not react with this Reaction this does nothing. * *

Possible ErrorResponses include: *

    *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_MESSAGE UNKNOWN_MESSAGE} *
    If the message this reaction was attached to got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_CHANNEL UNKNOWN_CHANNEL} *
    If the channel this reaction was used in got deleted.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#MISSING_ACCESS MISSING_ACCESS} *
    If we were removed from the channel/guild
  • *
* * @param user * The User of which to remove the reaction * * @throws java.lang.IllegalArgumentException * If the provided {@code user} is null. * @throws net.dv8tion.jda.api.exceptions.InsufficientPermissionException * If the provided User is not us and we do not have permission to * {@link net.dv8tion.jda.api.Permission#MESSAGE_MANAGE manage messages} * in the channel this reaction was used in * @throws net.dv8tion.jda.api.exceptions.PermissionException * If the message is from another user in a {@link PrivateChannel PrivateChannel} * * @return {@link net.dv8tion.jda.api.requests.RestAction RestAction} * Nothing is returned on success */ @Nonnull @CheckReturnValue public RestAction removeReaction(@Nonnull User user) { Checks.notNull(user, "User"); boolean self = user.equals(getJDA().getSelfUser()); if (!self && channel != null) { if (!channel.getType().isGuild()) throw new PermissionException("Unable to remove Reaction of other user in non-guild channels!"); GuildChannel guildChannel = (GuildChannel) channel; if (!guildChannel.getGuild().getSelfMember().hasPermission(guildChannel, Permission.MESSAGE_MANAGE)) throw new InsufficientPermissionException(guildChannel, Permission.MESSAGE_MANAGE); } String code = emoji.getAsReactionCode(); String target = self ? "@me" : user.getId(); Route.CompiledRoute route = Route.Messages.REMOVE_REACTION.compile(getChannelId(), getMessageId(), code, target); return new RestActionImpl<>(getJDA(), route); } /** * Removes this entire reaction from the message. *
Unlike {@link #removeReaction(User)}, which removes the reaction of a single user, this will remove the reaction * completely. * *

The following {@link net.dv8tion.jda.api.requests.ErrorResponse ErrorResponses} are possible: *

    *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#MISSING_ACCESS MISSING_ACCESS} *
    The currently logged in account lost access to the channel by either being removed from the guild * or losing the {@link net.dv8tion.jda.api.Permission#VIEW_CHANNEL VIEW_CHANNEL} permission
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_EMOJI UNKNOWN_EMOJI} *
    The provided unicode emoji doesn't exist. Try using one of the example formats.
  • * *
  • {@link net.dv8tion.jda.api.requests.ErrorResponse#UNKNOWN_MESSAGE UNKNOWN_MESSAGE} *
    The message was deleted.
  • *
* * @throws UnsupportedOperationException * If this reaction happened in a private channel * @throws InsufficientPermissionException * If the currently logged in account does not have {@link Permission#MESSAGE_MANAGE} in the channel * * @return {@link RestAction} * * @since 4.2.0 */ @Nonnull @CheckReturnValue public RestAction clearReactions() { if (channel == null) { Route.CompiledRoute route = Route.Messages.CLEAR_EMOJI_REACTIONS.compile(getChannelId(), getMessageId(), emoji.getAsReactionCode()); return new RestActionImpl<>(jda, route); } // Requires permission, only works in guilds if (!getChannelType().isGuild()) throw new UnsupportedOperationException("Cannot clear reactions on a message sent from a private channel"); GuildMessageChannel guildChannel = Objects.requireNonNull(getGuildChannel()); return guildChannel.clearReactionsById(getMessageId(), emoji); } @Override public boolean equals(Object obj) { if (obj == this) return true; if (!(obj instanceof MessageReaction)) return false; MessageReaction r = (MessageReaction) obj; return r.emoji.equals(emoji) && r.isSelf() == this.isSelf() && r.messageId == messageId; } @Override public String toString() { return new EntityString(this) .addMetadata("channelId", channelId) .addMetadata("messageId", messageId) .addMetadata("emoji", emoji) .toString(); } /** * Type of reaction. */ public enum ReactionType { NORMAL, SUPER } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy