org.spongepowered.api.data.Key Maven / Gradle / Ivy
Show all versions of spongeapi Show documentation
/*
* This file is part of SpongeAPI, licensed under the MIT License (MIT).
*
* Copyright (c) SpongePowered
* Copyright (c) contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package org.spongepowered.api.data;
import io.leangen.geantyref.TypeToken;
import org.spongepowered.api.ResourceKey;
import org.spongepowered.api.ResourceKeyed;
import org.spongepowered.api.Sponge;
import org.spongepowered.api.data.value.ListValue;
import org.spongepowered.api.data.value.MapValue;
import org.spongepowered.api.data.value.SetValue;
import org.spongepowered.api.data.value.Value;
import org.spongepowered.api.data.value.ValueContainer;
import org.spongepowered.api.data.value.WeightedCollectionValue;
import org.spongepowered.api.event.EventListener;
import org.spongepowered.api.event.data.ChangeDataHolderEvent;
import org.spongepowered.api.util.ResourceKeyedBuilder;
import org.spongepowered.api.util.annotation.CatalogedBy;
import org.spongepowered.api.util.weighted.WeightedTable;
import org.spongepowered.plugin.PluginContainer;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Collection;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.function.BiPredicate;
/**
* Represents a key to an underlying {@link Value} such that the underlying
* value can be retrieved from a {@link ValueContainer}. For the key to be used
* through retrieval of {@link DataHolder}s, it's required to use a
* {@link DataRegistration} if the data is needed to be serialized, or dynamically
* provided for through external mechanisms, through {@link DataProvider}s.
*
* If dynamic or persistent retention of the {@link Value Values} by
* {@link Key keys} is not desired, a registration with {@link DataRegistration}
* is optional. This would mean that any submitted {@link Value}s of a
* {@link Key} without an associated {@link DataRegistration} will be only
* stored on a
* {@link org.spongepowered.api.data.DataHolder.Mutable mutable DataHolder} for
* the duration that that holder exists. The value would not persist between
* reloads, restarts, etc.
*
* @param The type of {@link Value}
*/
@CatalogedBy(Keys.class)
public interface Key> extends ResourceKeyed {
/**
* Creates a {@link Key.Builder} which allows creation of a {@link Key}
* to later be registered for accessing values from
* a {@link ValueContainer}. It is the default policy that a
* custom created {@link Key} is NOT PERSISTENT by
* Sponge. If custom keys for {@link DataHolder}s is desired to be
* persisted, a {@link DataRegistration} is required.
*
* @see DataRegistration
* @return The key builder
*/
@SuppressWarnings("unchecked")
static Builder, ?> builder() {
return Sponge.game().builderProvider().provide(Builder.class);
}
static Key> from(final PluginContainer plugin, final String value, final Class type) {
return Key.from(ResourceKey.of(Objects.requireNonNull(plugin, "plugin"), value), type);
}
static Key> from(final ResourceKey resourceKey, final Class type) {
return Key.builder()
.key(Objects.requireNonNull(resourceKey, "resourceKey"))
.elementType(Objects.requireNonNull(type, "type"))
.build();
}
static Key> fromList(final PluginContainer plugin, final String value, final Class type) {
return Key.fromList(ResourceKey.of(Objects.requireNonNull(plugin, "plugin"), value), type);
}
static Key> fromList(final ResourceKey resourceKey, final Class type) {
return Key.builder()
.key(Objects.requireNonNull(resourceKey, "resourceKey"))
.listElementType(Objects.requireNonNull(type, "type"))
.build();
}
static Key> fromSet(final PluginContainer plugin, final String value, final Class type) {
return Key.fromSet(ResourceKey.of(Objects.requireNonNull(plugin, "plugin"), value), type);
}
static Key> fromSet(final ResourceKey resourceKey, final Class type) {
return Key.builder()
.key(Objects.requireNonNull(resourceKey, "resourceKey"))
.setElementType(Objects.requireNonNull(type, "type"))
.build();
}
static Key> fromMap(final PluginContainer plugin, final String value, final Class keyType, final Class valueType) {
return Key.fromMap(ResourceKey.of(Objects.requireNonNull(plugin, "plugin"), value), keyType, valueType);
}
static Key> fromMap(final ResourceKey resourceKey, final Class keyType, final Class valueType) {
return Key.builder()
.key(Objects.requireNonNull(resourceKey, "resourceKey"))
.mapElementType(Objects.requireNonNull(keyType, "keyType"), Objects.requireNonNull(valueType, "valueType"))
.build();
}
/**
* Gets the type of the {@link Value} this {@link Key} is representing.
*
* @return The value generic type
*/
Type valueType();
/**
* Gets the type of the element of the {@link Value} this {@link Key}
* is representing. On occasion, if the element is a {@link Collection} type,
* one can use {@link ParameterizedType#getActualTypeArguments()} to access
* type parameters, such as the element type parameter for {@link List} or
* {@link Map} values.
*
* @return The element generic type
*/
Type elementType();
/**
* Gets the {@link Comparator} to
* compare values of this key.
*
* @return The value comparator
*/
Comparator> elementComparator();
/**
* Gets the includes tester {@link BiPredicate}. This predicate should
* return {@code true} when the second parameter (the key value)
* is included in the first one (the matcher value).
*
* The default tester will always return {@code false}.
*
* @return The includes tester bi predicate
* @see KeyValueMatcher.Operator#INCLUDES
* @see KeyValueMatcher.Operator#EXCLUDES
*/
BiPredicate, ?> elementIncludesTester();
/**
* Register an event listener which listens to the value the key accesses
* changing.
*
* @param holderFilter The data holder to filter with
* @param listener The event listener
* @param The class type of the data holder
*/
void registerEvent(PluginContainer plugin, Class holderFilter, EventListener listener);
interface Builder> extends ResourceKeyedBuilder, Builder> {
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* For common cases, element types can be specified using one of the
* {@code *elementType(Class)} methods. If a new {@link TypeToken} is to
* be created, it is recommended to create an anonymous class instance
* of a token, as described in the GeAnTyRef documentation
*
*
* @param token The type token
* @param The element type of the Key
* @param The base value type of the key
* @return This builder, generified
*/
> Builder type(TypeToken token);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link Value} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder> elementType(Class type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link Value} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder> elementType(TypeToken type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link ListValue} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder, ListValue> listElementType(Class type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link ListValue} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder, ListValue> listElementType(TypeToken type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link SetValue} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder, SetValue> setElementType(Class type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link SetValue} is used.
*
* @param type The element type
* @param The element type of the Key
* @return This builder, generified
*/
Builder, SetValue> setElementType(TypeToken type);
/**
* Starter method for the builder, to be used immediately after
* {@link Key#builder()} is called. This defines the generics for the
* builder itself to provide the properly generified {@link Key}.
*
* This overload is provided for simple cases where a plain
* {@link MapValue} is used.
*
* @param keyType the key type
* @param valueType the value type
* @param The element type of the Key's key type
* @param The element type of the Key's value type
* @return This builder, generified
*/
Builder