194 lines
8.5 KiB
Java
194 lines
8.5 KiB
Java
package net.minecraft.world.level.levelgen;
|
|
|
|
import com.google.common.collect.ImmutableMap;
|
|
import com.google.common.collect.ImmutableSet;
|
|
import com.google.common.collect.ImmutableMap.Builder;
|
|
import com.mojang.serialization.Codec;
|
|
import com.mojang.serialization.Lifecycle;
|
|
import com.mojang.serialization.MapCodec;
|
|
import com.mojang.serialization.codecs.RecordCodecBuilder;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Optional;
|
|
import java.util.Set;
|
|
import java.util.stream.Collectors;
|
|
import java.util.stream.Stream;
|
|
import net.minecraft.core.Holder;
|
|
import net.minecraft.core.HolderLookup;
|
|
import net.minecraft.core.MappedRegistry;
|
|
import net.minecraft.core.RegistrationInfo;
|
|
import net.minecraft.core.Registry;
|
|
import net.minecraft.core.RegistryAccess;
|
|
import net.minecraft.core.WritableRegistry;
|
|
import net.minecraft.core.Holder.Reference;
|
|
import net.minecraft.core.registries.Registries;
|
|
import net.minecraft.resources.ResourceKey;
|
|
import net.minecraft.world.level.Level;
|
|
import net.minecraft.world.level.biome.MultiNoiseBiomeSource;
|
|
import net.minecraft.world.level.biome.MultiNoiseBiomeSourceParameterLists;
|
|
import net.minecraft.world.level.biome.TheEndBiomeSource;
|
|
import net.minecraft.world.level.chunk.ChunkGenerator;
|
|
import net.minecraft.world.level.dimension.BuiltinDimensionTypes;
|
|
import net.minecraft.world.level.dimension.DimensionType;
|
|
import net.minecraft.world.level.dimension.LevelStem;
|
|
import net.minecraft.world.level.storage.PrimaryLevelData.SpecialWorldProperty;
|
|
|
|
public record WorldDimensions(Map<ResourceKey<LevelStem>, LevelStem> dimensions) {
|
|
public static final MapCodec<WorldDimensions> CODEC = RecordCodecBuilder.mapCodec(
|
|
instance -> instance.group(
|
|
Codec.unboundedMap(ResourceKey.codec(Registries.LEVEL_STEM), LevelStem.CODEC).fieldOf("dimensions").forGetter(WorldDimensions::dimensions)
|
|
)
|
|
.apply(instance, instance.stable(WorldDimensions::new))
|
|
);
|
|
private static final Set<ResourceKey<LevelStem>> BUILTIN_ORDER = ImmutableSet.of(LevelStem.OVERWORLD, LevelStem.NETHER, LevelStem.END);
|
|
private static final int VANILLA_DIMENSION_COUNT = BUILTIN_ORDER.size();
|
|
|
|
public WorldDimensions(Map<ResourceKey<LevelStem>, LevelStem> dimensions) {
|
|
LevelStem levelStem = (LevelStem)dimensions.get(LevelStem.OVERWORLD);
|
|
if (levelStem == null) {
|
|
throw new IllegalStateException("Overworld settings missing");
|
|
} else {
|
|
this.dimensions = dimensions;
|
|
}
|
|
}
|
|
|
|
public WorldDimensions(Registry<LevelStem> stemRegistry) {
|
|
this((Map<ResourceKey<LevelStem>, LevelStem>)stemRegistry.listElements().collect(Collectors.toMap(Reference::key, Reference::value)));
|
|
}
|
|
|
|
public static Stream<ResourceKey<LevelStem>> keysInOrder(Stream<ResourceKey<LevelStem>> stemKeys) {
|
|
return Stream.concat(BUILTIN_ORDER.stream(), stemKeys.filter(resourceKey -> !BUILTIN_ORDER.contains(resourceKey)));
|
|
}
|
|
|
|
public WorldDimensions replaceOverworldGenerator(HolderLookup.Provider registries, ChunkGenerator chunkGenerator) {
|
|
HolderLookup<DimensionType> holderLookup = registries.lookupOrThrow(Registries.DIMENSION_TYPE);
|
|
Map<ResourceKey<LevelStem>, LevelStem> map = withOverworld(holderLookup, this.dimensions, chunkGenerator);
|
|
return new WorldDimensions(map);
|
|
}
|
|
|
|
public static Map<ResourceKey<LevelStem>, LevelStem> withOverworld(
|
|
HolderLookup<DimensionType> dimensionTypeRegistry, Map<ResourceKey<LevelStem>, LevelStem> dimensions, ChunkGenerator chunkGenerator
|
|
) {
|
|
LevelStem levelStem = (LevelStem)dimensions.get(LevelStem.OVERWORLD);
|
|
Holder<DimensionType> holder = (Holder<DimensionType>)(levelStem == null
|
|
? dimensionTypeRegistry.getOrThrow(BuiltinDimensionTypes.OVERWORLD)
|
|
: levelStem.type());
|
|
return withOverworld(dimensions, holder, chunkGenerator);
|
|
}
|
|
|
|
public static Map<ResourceKey<LevelStem>, LevelStem> withOverworld(
|
|
Map<ResourceKey<LevelStem>, LevelStem> stemMap, Holder<DimensionType> dimensionType, ChunkGenerator chunkGenerator
|
|
) {
|
|
Builder<ResourceKey<LevelStem>, LevelStem> builder = ImmutableMap.builder();
|
|
builder.putAll(stemMap);
|
|
builder.put(LevelStem.OVERWORLD, new LevelStem(dimensionType, chunkGenerator));
|
|
return builder.buildKeepingLast();
|
|
}
|
|
|
|
public ChunkGenerator overworld() {
|
|
LevelStem levelStem = (LevelStem)this.dimensions.get(LevelStem.OVERWORLD);
|
|
if (levelStem == null) {
|
|
throw new IllegalStateException("Overworld settings missing");
|
|
} else {
|
|
return levelStem.generator();
|
|
}
|
|
}
|
|
|
|
public Optional<LevelStem> get(ResourceKey<LevelStem> stemKey) {
|
|
return Optional.ofNullable((LevelStem)this.dimensions.get(stemKey));
|
|
}
|
|
|
|
public ImmutableSet<ResourceKey<Level>> levels() {
|
|
return (ImmutableSet<ResourceKey<Level>>)this.dimensions().keySet().stream().map(Registries::levelStemToLevel).collect(ImmutableSet.toImmutableSet());
|
|
}
|
|
|
|
public boolean isDebug() {
|
|
return this.overworld() instanceof DebugLevelSource;
|
|
}
|
|
|
|
private static SpecialWorldProperty specialWorldProperty(Registry<LevelStem> stemRegistry) {
|
|
return (SpecialWorldProperty)stemRegistry.getOptional(LevelStem.OVERWORLD).map(levelStem -> {
|
|
ChunkGenerator chunkGenerator = levelStem.generator();
|
|
if (chunkGenerator instanceof DebugLevelSource) {
|
|
return SpecialWorldProperty.DEBUG;
|
|
} else {
|
|
return chunkGenerator instanceof FlatLevelSource ? SpecialWorldProperty.FLAT : SpecialWorldProperty.NONE;
|
|
}
|
|
}).orElse(SpecialWorldProperty.NONE);
|
|
}
|
|
|
|
static Lifecycle checkStability(ResourceKey<LevelStem> key, LevelStem stem) {
|
|
return isVanillaLike(key, stem) ? Lifecycle.stable() : Lifecycle.experimental();
|
|
}
|
|
|
|
private static boolean isVanillaLike(ResourceKey<LevelStem> key, LevelStem stem) {
|
|
if (key == LevelStem.OVERWORLD) {
|
|
return isStableOverworld(stem);
|
|
} else if (key == LevelStem.NETHER) {
|
|
return isStableNether(stem);
|
|
} else {
|
|
return key == LevelStem.END ? isStableEnd(stem) : false;
|
|
}
|
|
}
|
|
|
|
private static boolean isStableOverworld(LevelStem levelStem) {
|
|
Holder<DimensionType> holder = levelStem.type();
|
|
return !holder.is(BuiltinDimensionTypes.OVERWORLD) && !holder.is(BuiltinDimensionTypes.OVERWORLD_CAVES)
|
|
? false
|
|
: !(
|
|
levelStem.generator().getBiomeSource() instanceof MultiNoiseBiomeSource multiNoiseBiomeSource
|
|
&& !multiNoiseBiomeSource.stable(MultiNoiseBiomeSourceParameterLists.OVERWORLD)
|
|
);
|
|
}
|
|
|
|
private static boolean isStableNether(LevelStem levelStem) {
|
|
return levelStem.type().is(BuiltinDimensionTypes.NETHER)
|
|
&& levelStem.generator() instanceof NoiseBasedChunkGenerator noiseBasedChunkGenerator
|
|
&& noiseBasedChunkGenerator.stable(NoiseGeneratorSettings.NETHER)
|
|
&& noiseBasedChunkGenerator.getBiomeSource() instanceof MultiNoiseBiomeSource multiNoiseBiomeSource
|
|
&& multiNoiseBiomeSource.stable(MultiNoiseBiomeSourceParameterLists.NETHER);
|
|
}
|
|
|
|
private static boolean isStableEnd(LevelStem levelStem) {
|
|
return levelStem.type().is(BuiltinDimensionTypes.END)
|
|
&& levelStem.generator() instanceof NoiseBasedChunkGenerator noiseBasedChunkGenerator
|
|
&& noiseBasedChunkGenerator.stable(NoiseGeneratorSettings.END)
|
|
&& noiseBasedChunkGenerator.getBiomeSource() instanceof TheEndBiomeSource;
|
|
}
|
|
|
|
public WorldDimensions.Complete bake(Registry<LevelStem> stemRegistry) {
|
|
Stream<ResourceKey<LevelStem>> stream = Stream.concat(stemRegistry.registryKeySet().stream(), this.dimensions.keySet().stream()).distinct();
|
|
|
|
record Entry(ResourceKey<LevelStem> key, LevelStem value) {
|
|
|
|
RegistrationInfo registrationInfo() {
|
|
return new RegistrationInfo(Optional.empty(), WorldDimensions.checkStability(this.key, this.value));
|
|
}
|
|
}
|
|
|
|
List<Entry> list = new ArrayList();
|
|
keysInOrder(stream)
|
|
.forEach(
|
|
resourceKey -> stemRegistry.getOptional(resourceKey)
|
|
.or(() -> Optional.ofNullable((LevelStem)this.dimensions.get(resourceKey)))
|
|
.ifPresent(levelStem -> list.add(new Entry(resourceKey, levelStem)))
|
|
);
|
|
Lifecycle lifecycle = list.size() == VANILLA_DIMENSION_COUNT ? Lifecycle.stable() : Lifecycle.experimental();
|
|
WritableRegistry<LevelStem> writableRegistry = new MappedRegistry<>(Registries.LEVEL_STEM, lifecycle);
|
|
list.forEach(arg -> writableRegistry.register(arg.key, arg.value, arg.registrationInfo()));
|
|
Registry<LevelStem> registry = writableRegistry.freeze();
|
|
SpecialWorldProperty specialWorldProperty = specialWorldProperty(registry);
|
|
return new WorldDimensions.Complete(registry.freeze(), specialWorldProperty);
|
|
}
|
|
|
|
public record Complete(Registry<LevelStem> dimensions, SpecialWorldProperty specialWorldProperty) {
|
|
public Lifecycle lifecycle() {
|
|
return this.dimensions.registryLifecycle();
|
|
}
|
|
|
|
public RegistryAccess.Frozen dimensionsRegistryAccess() {
|
|
return new RegistryAccess.ImmutableRegistryAccess(List.of(this.dimensions)).freeze();
|
|
}
|
|
}
|
|
}
|