All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
net.dv8tion.jda.api.interactions.components.LayoutComponent Maven / Gradle / Ivy
Go to download
Java wrapper for the popular chat & VOIP service: Discord https://discord.com
/*
* 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.interactions.components;
import net.dv8tion.jda.api.interactions.components.buttons.Button;
import net.dv8tion.jda.api.interactions.components.buttons.ButtonStyle;
import net.dv8tion.jda.api.utils.data.SerializableData;
import net.dv8tion.jda.internal.utils.Checks;
import net.dv8tion.jda.internal.utils.ComponentsUtil;
import net.dv8tion.jda.internal.utils.Helpers;
import org.jetbrains.annotations.Unmodifiable;
import javax.annotation.CheckReturnValue;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.*;
import java.util.stream.Collectors;
/**
* Represents a top-level layout used for {@link ItemComponent ItemComponents} such as {@link Button Buttons}.
*
* Components must always be contained within such a layout.
*
* @see ActionRow
*/
public interface LayoutComponent extends SerializableData, Iterable, Component
{
/**
* List representation of this component layout.
* This list is modifiable. Note that empty layouts are not supported.
*
* @return {@link List} of components in this layout
*/
@Nonnull
List getComponents();
@Override
default boolean isMessageCompatible()
{
if (!getType().isMessageCompatible())
return false;
return getComponents().stream().allMatch(ItemComponent::isMessageCompatible);
}
@Override
default boolean isModalCompatible()
{
if (!getType().isModalCompatible())
return false;
return getComponents().stream().allMatch(ItemComponent::isModalCompatible);
}
/**
* Immutable filtered copy of {@link #getComponents()} elements which are {@link ActionComponent ActionComponents}.
*
* @return Immutable {@link List} copy of {@link ActionComponent ActionComponents} in this layout
*/
@Nonnull
@Unmodifiable
default List getActionComponents()
{
return getComponents().stream()
.filter(ActionComponent.class::isInstance)
.map(ActionComponent.class::cast)
.collect(Helpers.toUnmodifiableList());
}
/**
* List of buttons in this component layout.
*
* @return Immutable {@link List} of {@link Button Buttons}
*/
@Nonnull
@Unmodifiable
default List getButtons()
{
return getComponents().stream()
.filter(Button.class::isInstance)
.map(Button.class::cast)
.collect(Helpers.toUnmodifiableList());
}
/**
* Whether all components in this layout are {@link ActionComponent#isDisabled() disabled}.
* Note that this is a universal quantifier, which means false does not imply {@link #isEnabled()}!
*
* @return True, if all components are disabled
*/
default boolean isDisabled()
{
return getActionComponents().stream().allMatch(ActionComponent::isDisabled);
}
/**
* Whether all components in this layout are {@link ActionComponent#isDisabled() enabled}.
* Note that this is a universal quantifier, which means false does not imply {@link #isDisabled()}!
*
* @return True, if all components are enabled
*/
default boolean isEnabled()
{
return getActionComponents().stream().noneMatch(ActionComponent::isDisabled);
}
/**
* Returns a new instance of this LayoutComponent with all components set to disabled/enabled.
* This does not modify the layout this was called on. To do this in-place, you can use {@link #getComponents()}.
*
* @param disabled
* True if the components should be set to disabled, false if they should be enabled
*
* @return The new layout component with all components updated
*
* @see ActionComponent#withDisabled(boolean)
*/
@Nonnull
@CheckReturnValue
LayoutComponent withDisabled(boolean disabled);
/**
* Returns a new instance of this LayoutComponent with all components set to disabled.
* This does not modify the layout this was called on. To do this in-place, you can use {@link #getComponents()}.
*
* @return The new layout component with all components updated
*
* @see ActionComponent#asDisabled()
*/
@Nonnull
@CheckReturnValue
LayoutComponent asDisabled();
/**
* Returns a new instance of this LayoutComponent with all components set to enabled.
* This does not modify the layout this was called on. To do this in-place, you can use {@link #getComponents()}.
*
* @return The new layout component with all components updated
*
* @see ActionComponent#asEnabled()
*/
@Nonnull
@CheckReturnValue
LayoutComponent asEnabled();
/**
* Check whether this layout is empty.
* Identical to {@code getComponents().isEmpty()}
*
* @return True, if this layout has no components
*/
default boolean isEmpty()
{
return getComponents().isEmpty();
}
/**
* Check whether this is a valid layout configuration.
* This checks that there is at least one component in this layout and it does not violate {@link ItemComponent#getMaxPerRow()}.
*
* @return True, if this layout is valid
*/
default boolean isValid()
{
if (isEmpty())
return false;
List components = getComponents();
Map> groups = components.stream().collect(Collectors.groupingBy(Component::getType));
if (groups.size() > 1) // TODO: You can't mix components right now but maybe in the future, we need to check back on this when that happens
return false;
for (Map.Entry> entry : groups.entrySet())
{
Component.Type type = entry.getKey();
List list = entry.getValue();
if (list.size() > type.getMaxPerRow())
return false;
}
return true;
}
/**
* Creates a copy of this {@link LayoutComponent}.
* This does not create copies of the contained components.
*
* @return A copy of this {@link LayoutComponent}
*/
@Nonnull
LayoutComponent createCopy();
/**
* Find and replace a component in this layout.
* This will locate and replace the existing component with the specified ID. If you provide null it will be removed instead.
*
* @param id
* The custom id of this component, can also be a URL for a {@link Button} with {@link ButtonStyle#LINK},
* or an SKU id for {@link ButtonStyle#PREMIUM}
* @param newComponent
* The new component or null to remove it
*
* @throws IllegalArgumentException
* If the provided id is null
*
* @return The old {@link ItemComponent} that was replaced or removed
*/
@Nullable
default ItemComponent updateComponent(@Nonnull String id, @Nullable ItemComponent newComponent)
{
Checks.notNull(id, "ID");
List list = getComponents();
for (ListIterator it = list.listIterator(); it.hasNext();)
{
ItemComponent component = it.next();
if (!(component instanceof ActionComponent))
continue;
ActionComponent action = (ActionComponent) component;
if (ComponentsUtil.isSameIdentifier(action, id))
{
if (newComponent == null)
it.remove();
else
it.set(newComponent);
return component;
}
}
return null;
}
/**
* Find and replace a component in this list of layouts.
* This will locate and replace the existing component with the specified ID. If you provide null it will be removed instead.
*
* If one of the layouts is empty after removing the component, it will be removed from the list.
* This is an inplace operation and modifies the provided list directly.
*
* @param layouts
* The layouts to modify
* @param id
* The custom id of this component, can also be a URL for a {@link Button} with {@link ButtonStyle#LINK}
* @param newComponent
* The new component or null to remove it
*
* @throws UnsupportedOperationException
* If the list cannot be modified
* @throws IllegalArgumentException
* If the provided id or list is null or the replace operation results in an {@link #isValid() invalid} layout
*
* @return True, if any of the layouts was modified
*/
static boolean updateComponent(@Nonnull List extends LayoutComponent> layouts, @Nonnull String id, @Nullable ItemComponent newComponent)
{
Checks.notNull(layouts, "LayoutComponent");
Checks.notEmpty(id, "ID or URL");
for (Iterator extends LayoutComponent> it = layouts.iterator(); it.hasNext();)
{
LayoutComponent components = it.next();
ItemComponent oldComponent = components.updateComponent(id, newComponent);
if (oldComponent != null)
{
if (components.getComponents().isEmpty())
it.remove();
else if (!components.isValid() && newComponent != null)
throw new IllegalArgumentException("Cannot replace " + oldComponent.getType() + " with " + newComponent.getType() + " due to a violation of the layout maximum. The resulting LayoutComponent is invalid!");
return !Objects.equals(oldComponent, newComponent);
}
}
return false;
}
/**
* Find and replace a component in this layout.
* This will locate and replace the existing component by checking for {@link Object#equals(Object) equality}. If you provide null it will be removed instead.
*
*
Example
*
{@code
* public void disableButton(ActionRow row, Button button) {
* row.updateComponent(button, button.asDisabled());
* }
* }
*
* @param component
* The component that should be replaced
* @param newComponent
* The new component or null to remove it
*
* @throws IllegalArgumentException
* If the provided component is null
*
* @return The old {@link ItemComponent} that was replaced or removed
*/
@Nullable
default ItemComponent updateComponent(@Nonnull ItemComponent component, @Nullable ItemComponent newComponent)
{
Checks.notNull(component, "Component to replace");
List list = getComponents();
for (ListIterator it = list.listIterator(); it.hasNext();)
{
ItemComponent item = it.next();
if (component.equals(item))
{
if (newComponent == null)
it.remove();
else
it.set(newComponent);
return component;
}
}
return null;
}
/**
* Find and replace a component in this list of layouts.
* This will locate and replace the existing component by checking for {@link Object#equals(Object) equality}. If you provide null it will be removed instead.
*
* If one of the layouts is empty after removing the component, it will be removed from the list.
* This is an inplace operation and modifies the provided list directly.
*
* @param layouts
* The layouts to modify
* @param component
* The component that should be replaced
* @param newComponent
* The new component or null to remove it
*
* @throws UnsupportedOperationException
* If the list cannot be modified
* @throws IllegalArgumentException
* If the provided component or list is null or the replace operation results in an {@link #isValid() invalid} layout
*
* @return True, if any of the layouts was modified
*/
static boolean updateComponent(@Nonnull List extends LayoutComponent> layouts, @Nonnull ItemComponent component, @Nullable ItemComponent newComponent)
{
Checks.notNull(layouts, "LayoutComponent");
Checks.notNull(component, "Component to replace");
for (Iterator extends LayoutComponent> it = layouts.iterator(); it.hasNext();)
{
LayoutComponent components = it.next();
ItemComponent oldComponent = components.updateComponent(component, newComponent);
if (oldComponent != null)
{
if (components.getComponents().isEmpty())
it.remove();
else if (!components.isValid() && newComponent != null)
throw new IllegalArgumentException("Cannot replace " + oldComponent.getType() + " with " + newComponent.getType() + " due to a violation of the layout maximum. The resulting LayoutComponent is invalid!");
return !Objects.equals(oldComponent, newComponent);
}
}
return false;
}
}