HyCodeYourTale
classpublicPriority 3

NBiomeDistanceStage

com.hypixel.hytale.builtin.hytalegenerator.newsystem.stages.NBiomeDistanceStage

implements NStage

6

Methods

6

Public Methods

10

Fields

1

Constructors

Constants

doubleBUFFER_DIAGONAL_VOXEL_GRID= Math.sqrt( (double)(NPixelBuffer.SIZE.x * NPixelBuffer.SIZE.x + NPixelBuffer.SIZE.z * NPixe...
doubleDEFAULT_DISTANCE_TO_BIOME_EDGE= 1.7976931348623157E308
doubleORIGIN_REACH= 1.0

Constructors

public
NBiomeDistanceStage(String stageName, NParametrizedBufferType biomeInputBufferType, NParametrizedBufferType biomeDistanceOutputBufferType, double maxDistance_voxelGrid)

Methods

Public Methods (6)

publicstatic
double distanceToBuffer_voxelGrid(Vector3i position_voxelGrid, Vector3i position_bufferGrid)
public
double distanceToClosestOtherBiome(BiomeType thisBiome)
public
Map<NBufferType, Bounds3i> getInputTypesAndBounds_bufferGrid()
@Nonnull@Override
public
String getName()
@Nonnull@Override
public
List<NBufferType> getOutputTypes()
@Nonnull@Override
public
void run(NStage.Context context)
@Override

Fields

Public Fields (4)

publicstaticfinalClass<NCountedPixelBuffer> biomeBufferClass
publicstaticfinalClass<NSimplePixelBuffer> biomeDistanceBufferClass
publicstaticfinalClass<NBiomeDistanceStage.BiomeDistanceEntries> biomeDistanceClass
publicstaticfinalClass<BiomeType> biomeTypeClass

Private/Package Fields (6)

privateNParametrizedBufferType biomeDistanceOutputBufferType
privateNParametrizedBufferType biomeInputBufferType
privateBounds3i inputBounds_bufferGrid
privateint maxDistance_bufferGrid
privatedouble maxDistance_voxelGrid
privateString stageName

Inheritance

Parent
Current
Interface
Child

Use mouse wheel to zoom, drag to pan. Click nodes to navigate.

Related Classes

Source Code

package com.hypixel.hytale.builtin.hytalegenerator.newsystem.stages;

import com.hypixel.hytale.builtin.hytalegenerator.biome.BiomeType;
import com.hypixel.hytale.builtin.hytalegenerator.bounds.Bounds3i;
import com.hypixel.hytale.builtin.hytalegenerator.framework.math.Calculator;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.GridUtils;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.NBufferBundle;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.buffers.NCountedPixelBuffer;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.buffers.NPixelBuffer;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.buffers.NSimplePixelBuffer;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.buffers.type.NBufferType;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.bufferbundle.buffers.type.NParametrizedBufferType;
import com.hypixel.hytale.builtin.hytalegenerator.newsystem.views.NPixelBufferView;
import com.hypixel.hytale.math.vector.Vector3i;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;

public class NBiomeDistanceStage implements NStage {
   private static final double ORIGIN_REACH = 1.0;
   private static final double BUFFER_DIAGONAL_VOXEL_GRID = Math.sqrt(
      (double)(NPixelBuffer.SIZE.x * NPixelBuffer.SIZE.x + NPixelBuffer.SIZE.z * NPixelBuffer.SIZE.z)
   );
   public static final double DEFAULT_DISTANCE_TO_BIOME_EDGE = 1.7976931348623157E308;
   public static final Class<NCountedPixelBuffer> biomeBufferClass = NCountedPixelBuffer.class;
   public static final Class<BiomeType> biomeTypeClass = BiomeType.class;
   public static final Class<NSimplePixelBuffer> biomeDistanceBufferClass = NSimplePixelBuffer.class;
   public static final Class<NBiomeDistanceStage.BiomeDistanceEntries> biomeDistanceClass = NBiomeDistanceStage.BiomeDistanceEntries.class;
   private final NParametrizedBufferType biomeInputBufferType;
   private final NParametrizedBufferType biomeDistanceOutputBufferType;
   private final String stageName;
   private final double maxDistance_voxelGrid;
   private final int maxDistance_bufferGrid;
   private final Bounds3i inputBounds_bufferGrid;

   public NBiomeDistanceStage(
      @Nonnull String stageName,
      @Nonnull NParametrizedBufferType biomeInputBufferType,
      @Nonnull NParametrizedBufferType biomeDistanceOutputBufferType,
      double maxDistance_voxelGrid
   ) {
      assert maxDistance_voxelGrid >= 0.0;

      this.stageName = stageName;
      this.biomeInputBufferType = biomeInputBufferType;
      this.biomeDistanceOutputBufferType = biomeDistanceOutputBufferType;
      this.maxDistance_voxelGrid = maxDistance_voxelGrid;
      this.maxDistance_bufferGrid = GridUtils.toBufferDistanceInclusive_fromVoxelDistance((int)Math.ceil(maxDistance_voxelGrid));
      Bounds3i inputBounds_voxelGrid = GridUtils.createBounds_fromRadius_originVoxelInclusive((int)Math.ceil(maxDistance_voxelGrid));
      Bounds3i bufferColumnBounds_voxelGrid = GridUtils.createColumnBounds_voxelGrid(new Vector3i(), 0, 1);
      inputBounds_voxelGrid.stack(bufferColumnBounds_voxelGrid);
      this.inputBounds_bufferGrid = GridUtils.createBufferBoundsInclusive_fromVoxelBounds(inputBounds_voxelGrid);
      GridUtils.setBoundsYToWorldHeight_bufferGrid(this.inputBounds_bufferGrid);
   }

   @Override
   public void run(@Nonnull NStage.Context context) {
      NBufferBundle.Access.View biomeAccess = context.bufferAccess.get(this.biomeInputBufferType);
      NPixelBufferView<BiomeType> biomeSpace = new NPixelBufferView<>(biomeAccess, biomeTypeClass);
      NBufferBundle.Access.View biomeDistanceAccess = context.bufferAccess.get(this.biomeDistanceOutputBufferType);
      NPixelBufferView<NBiomeDistanceStage.BiomeDistanceEntries> biomeDistanceSpace = new NPixelBufferView<>(biomeDistanceAccess, biomeDistanceClass);
      Vector3i position_voxelGrid = new Vector3i();

      for (position_voxelGrid.x = biomeDistanceSpace.minX(); position_voxelGrid.x < biomeDistanceSpace.maxX(); position_voxelGrid.x++) {
         for (position_voxelGrid.z = biomeDistanceSpace.minZ(); position_voxelGrid.z < biomeDistanceSpace.maxZ(); position_voxelGrid.z++) {
            NBiomeDistanceStage.BiomeDistanceEntries distanceEntries = this.createDistanceTracker(biomeAccess, biomeSpace, position_voxelGrid);
            biomeDistanceSpace.set(distanceEntries, position_voxelGrid);
         }
      }
   }

   @Nonnull
   private NBiomeDistanceStage.BiomeDistanceEntries createDistanceTracker(
      @Nonnull NBufferBundle.Access.View biomeAccess, @Nonnull NPixelBufferView<BiomeType> biomeSpace, @Nonnull Vector3i targetPosition_voxelGrid
   ) {
      NBiomeDistanceStage.BiomeDistanceCounter counter = new NBiomeDistanceStage.BiomeDistanceCounter();
      Vector3i position_bufferGrid = new Vector3i();
      Bounds3i scanBounds_voxelGrid = GridUtils.createBounds_fromRadius_originVoxelInclusive((int)Math.ceil(this.maxDistance_voxelGrid));
      scanBounds_voxelGrid.offset(targetPosition_voxelGrid);
      Bounds3i scanBounds_bufferGrid = GridUtils.createBufferBoundsInclusive_fromVoxelBounds(scanBounds_voxelGrid);

      for (position_bufferGrid.x = scanBounds_bufferGrid.min.x; position_bufferGrid.x < scanBounds_bufferGrid.max.x; position_bufferGrid.x++) {
         for (position_bufferGrid.z = scanBounds_bufferGrid.min.z; position_bufferGrid.z < scanBounds_bufferGrid.max.z; position_bufferGrid.z++) {
            double distanceToBuffer_voxelGrid = distanceToBuffer_voxelGrid(targetPosition_voxelGrid, position_bufferGrid);
            distanceToBuffer_voxelGrid = Math.max(distanceToBuffer_voxelGrid - 1.0, 0.0);
            if (!(distanceToBuffer_voxelGrid > this.maxDistance_voxelGrid)) {
               NCountedPixelBuffer<BiomeType> biomeBuffer = (NCountedPixelBuffer<BiomeType>)biomeAccess.getBuffer(position_bufferGrid).buffer();
               List<BiomeType> uniqueBiomeTypes = biomeBuffer.getUniqueEntries();

               assert !uniqueBiomeTypes.isEmpty();

               if (!allBiomesAreCountedAndFarther(counter, uniqueBiomeTypes, distanceToBuffer_voxelGrid)) {
                  if (uniqueBiomeTypes.size() == 1) {
                     if (!(distanceToBuffer_voxelGrid > this.maxDistance_voxelGrid)) {
                        counter.accountFor(uniqueBiomeTypes.getFirst(), distanceToBuffer_voxelGrid);
                     }
                  } else {
                     Bounds3i bufferBounds_voxelGrid = GridUtils.createColumnBounds_voxelGrid(position_bufferGrid, 0, 1);
                     Vector3i columnPosition_voxelGrid = new Vector3i();

                     for (columnPosition_voxelGrid.x = bufferBounds_voxelGrid.min.x;
                        columnPosition_voxelGrid.x < bufferBounds_voxelGrid.max.x;
                        columnPosition_voxelGrid.x++
                     ) {
                        for (columnPosition_voxelGrid.z = bufferBounds_voxelGrid.min.z;
                           columnPosition_voxelGrid.z < bufferBounds_voxelGrid.max.z;
                           columnPosition_voxelGrid.z++
                        ) {
                           double distanceToColumn_voxelGrid = Calculator.distance(
                              (double)columnPosition_voxelGrid.x,
                              (double)columnPosition_voxelGrid.z,
                              (double)targetPosition_voxelGrid.x,
                              (double)targetPosition_voxelGrid.z
                           );
                           distanceToColumn_voxelGrid = Math.max(distanceToColumn_voxelGrid - 1.0, 0.0);
                           if (!(distanceToColumn_voxelGrid > this.maxDistance_voxelGrid)) {
                              BiomeType biomeType = biomeSpace.getContent(columnPosition_voxelGrid);

                              assert biomeType != null;

                              counter.accountFor(biomeType, distanceToColumn_voxelGrid);
                           }
                        }
                     }
                  }
               }
            }
         }
      }

      return new NBiomeDistanceStage.BiomeDistanceEntries(counter.entries);
   }

   @Nonnull
   @Override
   public Map<NBufferType, Bounds3i> getInputTypesAndBounds_bufferGrid() {
      return Map.of(this.biomeInputBufferType, this.inputBounds_bufferGrid);
   }

   @Nonnull
   @Override
   public List<NBufferType> getOutputTypes() {
      return List.of(this.biomeDistanceOutputBufferType);
   }

   @Nonnull
   @Override
   public String getName() {
      return this.stageName;
   }

   public static double distanceToBuffer_voxelGrid(@Nonnull Vector3i position_voxelGrid, @Nonnull Vector3i position_bufferGrid) {
      assert (double)position_voxelGrid.y == 0.0;

      assert (double)position_bufferGrid.y == 0.0;

      Vector3i bufferAtPosition_bufferGrid = position_voxelGrid.clone();
      GridUtils.toBufferGrid_fromVoxelGrid(bufferAtPosition_bufferGrid);
      if (bufferAtPosition_bufferGrid.x == position_bufferGrid.x && bufferAtPosition_bufferGrid.z == position_bufferGrid.z) {
         return 0.0;
      } else {
         int cornerShift = NCountedPixelBuffer.SIZE_VOXEL_GRID.x - 1;
         Vector3i corner00 = position_bufferGrid.clone();
         GridUtils.toVoxelGrid_fromBufferGrid(corner00);
         Vector3i corner01 = new Vector3i(corner00);
         corner01.z += cornerShift;
         if (position_voxelGrid.x >= corner00.x && position_voxelGrid.x <= corner00.x + cornerShift) {
            return (double)Math.min(Math.abs(position_voxelGrid.z - corner00.z), Math.abs(position_voxelGrid.z - corner01.z));
         } else {
            Vector3i corner10 = new Vector3i(corner00);
            corner10.x += cornerShift;
            if (position_voxelGrid.z >= corner00.z && position_voxelGrid.z <= corner00.z + cornerShift) {
               return (double)Math.min(Math.abs(position_voxelGrid.x - corner00.x), Math.abs(position_voxelGrid.x - corner10.x));
            } else if (position_voxelGrid.x < corner00.x && position_voxelGrid.z < corner00.z) {
               return position_voxelGrid.distanceTo(corner00);
            } else if (position_voxelGrid.x < corner01.x && position_voxelGrid.z > corner01.z) {
               return position_voxelGrid.distanceTo(corner01);
            } else if (position_voxelGrid.x > corner10.x && position_voxelGrid.z < corner10.z) {
               return position_voxelGrid.distanceTo(corner10);
            } else {
               Vector3i corner11 = new Vector3i(corner10.x, 0, corner01.z);
               return position_voxelGrid.distanceTo(corner11);
            }
         }
      }
   }

   private static boolean allBiomesAreCountedAndFarther(
      @Nonnull NBiomeDistanceStage.BiomeDistanceCounter counter, @Nonnull List<BiomeType> uniqueBiomes, double distanceToBuffer_voxelGrid
   ) {
      for (BiomeType biomeType : uniqueBiomes) {
         if (counter.isCloserThanCounted(biomeType, distanceToBuffer_voxelGrid)) {
            return false;
         }
      }

      return true;
   }

   private static class BiomeDistanceCounter {
      @Nonnull
      final List<NBiomeDistanceStage.BiomeDistanceEntry> entries = new ArrayList<>(3);
      @Nullable
      NBiomeDistanceStage.BiomeDistanceEntry cachedEntry = null;

      BiomeDistanceCounter() {
      }

      boolean isCloserThanCounted(@Nonnull BiomeType biomeType, double distance_voxelGrid) {
         for (NBiomeDistanceStage.BiomeDistanceEntry entry : this.entries) {
            if (entry.biomeType == biomeType) {
               return distance_voxelGrid < entry.distance_voxelGrid;
            }
         }

         return true;
      }

      void accountFor(@Nonnull BiomeType biomeType, double distance_voxelGrid) {
         if (this.cachedEntry != null && this.cachedEntry.biomeType == biomeType) {
            if (!(this.cachedEntry.distance_voxelGrid <= distance_voxelGrid)) {
               this.cachedEntry.distance_voxelGrid = distance_voxelGrid;
            }
         } else {
            for (NBiomeDistanceStage.BiomeDistanceEntry entry : this.entries) {
               if (entry.biomeType == biomeType) {
                  this.cachedEntry = entry;
                  if (entry.distance_voxelGrid <= distance_voxelGrid) {
                     return;
                  }

                  entry.distance_voxelGrid = distance_voxelGrid;
                  return;
               }
            }

            NBiomeDistanceStage.BiomeDistanceEntry entryx = new NBiomeDistanceStage.BiomeDistanceEntry();
            entryx.biomeType = biomeType;
            entryx.distance_voxelGrid = distance_voxelGrid;
            this.entries.add(entryx);
            this.cachedEntry = entryx;
         }
      }
   }

   public static class BiomeDistanceEntries {
      public final List<NBiomeDistanceStage.BiomeDistanceEntry> entries;

      public BiomeDistanceEntries(@Nonnull List<NBiomeDistanceStage.BiomeDistanceEntry> entries) {
         this.entries = entries;
      }

      public double distanceToClosestOtherBiome(@Nonnull BiomeType thisBiome) {
         double smallestDistance = 1.7976931348623157E308;

         for (NBiomeDistanceStage.BiomeDistanceEntry entry : this.entries) {
            if (entry.biomeType != thisBiome) {
               smallestDistance = Math.min(smallestDistance, entry.distance_voxelGrid);
            }
         }

         return smallestDistance;
      }
   }

   public static class BiomeDistanceEntry {
      public BiomeType biomeType;
      public double distance_voxelGrid;

      public BiomeDistanceEntry() {
      }
   }
}