/* * * MIT License * * Copyright (c) 2020 TerraForged * * 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 com.terraforged.core.world.geology; import com.terraforged.core.util.Seed; import me.dags.noise.Module; import me.dags.noise.Source; import me.dags.noise.util.NoiseUtil; import java.util.ArrayList; import java.util.LinkedList; import java.util.List; public class Strata { private final Module heightMod; private final List> strata; private Strata(Module heightMod, List> strata) { this.strata = strata; this.heightMod = heightMod; } public boolean downwards(final int x, final int y, final int z, final Stratum.Visitor visitor) { DepthBuffer buffer = new DepthBuffer(); initBuffer(buffer, x, z); return downwards(x, y, z, buffer, visitor); } public boolean downwards(final int x, final int y, final int z, final DepthBuffer buffer, Stratum.Visitor visitor) { initBuffer(buffer, x, z); int py = y; T last = null; for (int i = 0; i < strata.size(); i++) { float depth = buffer.getDepth(i); int height = NoiseUtil.round(depth * y); T value = strata.get(i).getValue(); last = value; for (int dy = 0; dy < height; dy++) { if (py <= y) { boolean cont = visitor.visit(py, value); if (!cont) { return false; } } if (--py < 0) { return false; } } } if (last != null) { while (py > 0) { visitor.visit(py, last); py--; } } return true; } private int getYOffset(int x, int z) { return (int) (64 * heightMod.getValue(x, z)); } private void initBuffer(DepthBuffer buffer, int x, int z) { buffer.init(strata.size()); for (int i = 0; i < strata.size(); i++) { float depth = strata.get(i).getDepth(x, z); buffer.set(i, depth); } } public static Builder builder(int seed, me.dags.noise.source.Builder noise) { return new Builder<>(seed, noise); } public static class Builder { private final Seed seed; private final me.dags.noise.source.Builder noise; private final List> strata = new LinkedList<>(); public Builder(int seed, me.dags.noise.source.Builder noise) { this.seed = new Seed(seed); this.noise = noise; } public Builder add(T material, double depth) { Module module = noise.seed(seed.next()).perlin().scale(depth); strata.add(Stratum.of(material, module)); return this; } public Builder add(Source type, T material, double depth) { Module module = noise.seed(seed.next()).build(type).scale(depth); strata.add(Stratum.of(material, module)); return this; } public Strata build() { Module height = Source.cell(seed.next(), 100); return new Strata<>(height, new ArrayList<>(strata)); } } }