HyCodeYourTale
classpublicPriority 3

MultiMixDensity

com.hypixel.hytale.builtin.hytalegenerator.density.nodes.MultiMixDensity

extends Density

7

Methods

7

Public Methods

6

Fields

1

Constructors

Constants

MultiMixDensity.Segment.GaugeSegmentComparatorINSTANCE= new MultiMixDensity.Segment.GaugeSegmentComparator()

Constructors

public
MultiMixDensity(List<MultiMixDensity.Key> keys, Density influenceDensity)

Methods

Public Methods (7)

publicstatic
record Key(double value, Density density)
public
Integer apply(Double gauge, MultiMixDensity.Segment segment)
@Nonnull
public
boolean contains(double gauge)
public
double getValue(Density.Context context, double gauge)
publicstatic
boolean isKeysUnique(List<MultiMixDensity.Key> keys)
public
double process(Density.Context context)
@Override
public
void setInputs(Density[] inputs)
@Override

Fields

Private/Package Fields (6)

privateDensity firstDensity
privateDensity influenceDensity
privateDensity lastDensity
privatedouble max
privatedouble min
privateList<MultiMixDensity.Segment> segments

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.density.nodes;

import com.hypixel.hytale.builtin.hytalegenerator.ArrayUtil;
import com.hypixel.hytale.builtin.hytalegenerator.density.Density;
import com.hypixel.hytale.builtin.hytalegenerator.framework.math.Interpolation;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.function.BiFunction;
import javax.annotation.Nonnull;

public class MultiMixDensity extends Density {
   @Nonnull
   private final List<MultiMixDensity.Segment> segments;
   private final double min;
   private final double max;
   private final Density firstDensity;
   private final Density lastDensity;
   private Density influenceDensity;

   public MultiMixDensity(@Nonnull List<MultiMixDensity.Key> keys, @Nonnull Density influenceDensity) {
      if (keys.size() < 2) {
         throw new IllegalArgumentException("must have at least two keys");
      } else {
         keys.sort(Comparator.comparingDouble(element -> element.value));
         if (!isKeysUnique(keys)) {
            throw new IllegalArgumentException("Duplicate keys provided.");
         } else {
            this.segments = new ArrayList<>(keys.size() - 1);

            for (int i = 1; i < keys.size(); i++) {
               MultiMixDensity.Key key0 = keys.get(i - 1);
               MultiMixDensity.Key key1 = keys.get(i);
               this.segments.add(new MultiMixDensity.Segment(key0, key1));
            }

            this.min = keys.getFirst().value;
            this.max = keys.getLast().value;
            this.firstDensity = keys.getFirst().density;
            this.lastDensity = keys.getLast().density;
            this.influenceDensity = influenceDensity;
         }
      }
   }

   @Override
   public double process(@Nonnull Density.Context context) {
      double influence = this.influenceDensity.process(context);
      if (influence <= this.min) {
         return this.firstDensity.process(context);
      } else if (influence >= this.max) {
         return this.lastDensity.process(context);
      } else {
         int index = ArrayUtil.sortedSearch(this.segments, influence, MultiMixDensity.Segment.GaugeSegmentComparator.INSTANCE);
         if (index == -1) {
            assert false : "should never get here";

            return 0.0;
         } else {
            return this.segments.get(index).getValue(context, influence);
         }
      }
   }

   @Override
   public void setInputs(@Nonnull Density[] inputs) {
      if (inputs.length != 1) {
         throw new IllegalArgumentException("inputs.length != 1");
      } else {
         this.influenceDensity = inputs[0];
      }
   }

   public static boolean isKeysUnique(@Nonnull List<MultiMixDensity.Key> keys) {
      for (int i = 1; i < keys.size(); i++) {
         if (keys.get(i).value == keys.get(i - 1).value) {
            return false;
         }
      }

      return true;
   }

   public static record Key(double value, Density density) {
   }

   private static class Segment {
      @Nonnull
      private final MultiMixDensity.Key key0;
      @Nonnull
      private final MultiMixDensity.Key key1;
      private final double magnitude;

      public Segment(@Nonnull MultiMixDensity.Key key0, @Nonnull MultiMixDensity.Key key1) {
         assert key0.value < key1.value : "key0 should be smaller than key1";

         this.key0 = key0;
         this.key1 = key1;
         this.magnitude = key1.value - key0.value;
      }

      public boolean contains(double gauge) {
         return gauge >= this.key0.value && gauge <= this.key1.value;
      }

      public double getValue(@Nonnull Density.Context context, double gauge) {
         assert gauge >= this.key0.value && gauge <= this.key1.value : "mix outside range";

         double THRESHOLD_INPUT_0 = 0.0;
         double THRESHOLD_INPUT_1 = 1.0;
         double weight = (gauge - this.key0.value) / this.magnitude;
         if (weight == 0.0) {
            return this.key0.density.process(context);
         } else if (weight == 1.0) {
            return this.key1.density.process(context);
         } else if (this.key0.density == this.key1.density) {
            return this.key0.density.process(context);
         } else {
            double value0 = this.key0.density.process(context);
            double value1 = this.key1.density.process(context);
            return Interpolation.linear(value0, value1, weight);
         }
      }

      public static class GaugeSegmentComparator implements BiFunction<Double, MultiMixDensity.Segment, Integer> {
         public static final MultiMixDensity.Segment.GaugeSegmentComparator INSTANCE = new MultiMixDensity.Segment.GaugeSegmentComparator();

         public GaugeSegmentComparator() {
         }

         @Nonnull
         public Integer apply(Double gauge, @Nonnull MultiMixDensity.Segment segment) {
            if (gauge < segment.key0.value) {
               return -1;
            } else {
               return gauge >= segment.key1.value ? 1 : 0;
            }
         }
      }
   }
}