216 lines
5 KiB
Java
216 lines
5 KiB
Java
package net.minecraft.core;
|
|
|
|
import com.mojang.datafixers.util.Either;
|
|
import java.util.Collection;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.Optional;
|
|
import java.util.Set;
|
|
import java.util.Spliterator;
|
|
import java.util.function.Function;
|
|
import java.util.stream.Stream;
|
|
import net.minecraft.Util;
|
|
import net.minecraft.tags.TagKey;
|
|
import net.minecraft.util.RandomSource;
|
|
import org.jetbrains.annotations.Nullable;
|
|
import org.jetbrains.annotations.VisibleForTesting;
|
|
|
|
public interface HolderSet<T> extends Iterable<Holder<T>> {
|
|
Stream<Holder<T>> stream();
|
|
|
|
int size();
|
|
|
|
boolean isBound();
|
|
|
|
Either<TagKey<T>, List<Holder<T>>> unwrap();
|
|
|
|
Optional<Holder<T>> getRandomElement(RandomSource random);
|
|
|
|
Holder<T> get(int index);
|
|
|
|
boolean contains(Holder<T> holder);
|
|
|
|
boolean canSerializeIn(HolderOwner<T> owner);
|
|
|
|
Optional<TagKey<T>> unwrapKey();
|
|
|
|
@Deprecated
|
|
@VisibleForTesting
|
|
static <T> HolderSet.Named<T> emptyNamed(HolderOwner<T> owner, TagKey<T> key) {
|
|
return new HolderSet.Named<T>(owner, key) {
|
|
@Override
|
|
protected List<Holder<T>> contents() {
|
|
throw new UnsupportedOperationException("Tag " + this.key() + " can't be dereferenced during construction");
|
|
}
|
|
};
|
|
}
|
|
|
|
static <T> HolderSet<T> empty() {
|
|
return (HolderSet<T>)HolderSet.Direct.EMPTY;
|
|
}
|
|
|
|
@SafeVarargs
|
|
static <T> HolderSet.Direct<T> direct(Holder<T>... contents) {
|
|
return new HolderSet.Direct<>(List.of(contents));
|
|
}
|
|
|
|
static <T> HolderSet.Direct<T> direct(List<? extends Holder<T>> contents) {
|
|
return new HolderSet.Direct<>(List.copyOf(contents));
|
|
}
|
|
|
|
@SafeVarargs
|
|
static <E, T> HolderSet.Direct<T> direct(Function<E, Holder<T>> holderFactory, E... values) {
|
|
return direct(Stream.of(values).map(holderFactory).toList());
|
|
}
|
|
|
|
static <E, T> HolderSet.Direct<T> direct(Function<E, Holder<T>> holderFactory, Collection<E> values) {
|
|
return direct(values.stream().map(holderFactory).toList());
|
|
}
|
|
|
|
public static final class Direct<T> extends HolderSet.ListBacked<T> {
|
|
static final HolderSet.Direct<?> EMPTY = new HolderSet.Direct(List.of());
|
|
private final List<Holder<T>> contents;
|
|
@Nullable
|
|
private Set<Holder<T>> contentsSet;
|
|
|
|
Direct(List<Holder<T>> contents) {
|
|
this.contents = contents;
|
|
}
|
|
|
|
@Override
|
|
protected List<Holder<T>> contents() {
|
|
return this.contents;
|
|
}
|
|
|
|
@Override
|
|
public boolean isBound() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public Either<TagKey<T>, List<Holder<T>>> unwrap() {
|
|
return Either.right(this.contents);
|
|
}
|
|
|
|
@Override
|
|
public Optional<TagKey<T>> unwrapKey() {
|
|
return Optional.empty();
|
|
}
|
|
|
|
@Override
|
|
public boolean contains(Holder<T> holder) {
|
|
if (this.contentsSet == null) {
|
|
this.contentsSet = Set.copyOf(this.contents);
|
|
}
|
|
|
|
return this.contentsSet.contains(holder);
|
|
}
|
|
|
|
public String toString() {
|
|
return "DirectSet[" + this.contents + "]";
|
|
}
|
|
|
|
public boolean equals(Object object) {
|
|
return this == object ? true : object instanceof HolderSet.Direct<?> direct && this.contents.equals(direct.contents);
|
|
}
|
|
|
|
public int hashCode() {
|
|
return this.contents.hashCode();
|
|
}
|
|
}
|
|
|
|
public abstract static class ListBacked<T> implements HolderSet<T> {
|
|
protected abstract List<Holder<T>> contents();
|
|
|
|
@Override
|
|
public int size() {
|
|
return this.contents().size();
|
|
}
|
|
|
|
public Spliterator<Holder<T>> spliterator() {
|
|
return this.contents().spliterator();
|
|
}
|
|
|
|
public Iterator<Holder<T>> iterator() {
|
|
return this.contents().iterator();
|
|
}
|
|
|
|
@Override
|
|
public Stream<Holder<T>> stream() {
|
|
return this.contents().stream();
|
|
}
|
|
|
|
@Override
|
|
public Optional<Holder<T>> getRandomElement(RandomSource random) {
|
|
return Util.getRandomSafe(this.contents(), random);
|
|
}
|
|
|
|
@Override
|
|
public Holder<T> get(int index) {
|
|
return (Holder<T>)this.contents().get(index);
|
|
}
|
|
|
|
@Override
|
|
public boolean canSerializeIn(HolderOwner<T> owner) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
public static class Named<T> extends HolderSet.ListBacked<T> {
|
|
private final HolderOwner<T> owner;
|
|
private final TagKey<T> key;
|
|
@Nullable
|
|
private List<Holder<T>> contents;
|
|
|
|
Named(HolderOwner<T> owner, TagKey<T> key) {
|
|
this.owner = owner;
|
|
this.key = key;
|
|
}
|
|
|
|
void bind(List<Holder<T>> contents) {
|
|
this.contents = List.copyOf(contents);
|
|
}
|
|
|
|
public TagKey<T> key() {
|
|
return this.key;
|
|
}
|
|
|
|
@Override
|
|
protected List<Holder<T>> contents() {
|
|
if (this.contents == null) {
|
|
throw new IllegalStateException("Trying to access unbound tag '" + this.key + "' from registry " + this.owner);
|
|
} else {
|
|
return this.contents;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isBound() {
|
|
return this.contents != null;
|
|
}
|
|
|
|
@Override
|
|
public Either<TagKey<T>, List<Holder<T>>> unwrap() {
|
|
return Either.left(this.key);
|
|
}
|
|
|
|
@Override
|
|
public Optional<TagKey<T>> unwrapKey() {
|
|
return Optional.of(this.key);
|
|
}
|
|
|
|
@Override
|
|
public boolean contains(Holder<T> holder) {
|
|
return holder.is(this.key);
|
|
}
|
|
|
|
public String toString() {
|
|
return "NamedSet(" + this.key + ")[" + this.contents + "]";
|
|
}
|
|
|
|
@Override
|
|
public boolean canSerializeIn(HolderOwner<T> owner) {
|
|
return this.owner.canSerializeIn(owner);
|
|
}
|
|
}
|
|
}
|