mirror of
https://github.com/IntellectualSites/PlotSquared.git
synced 2025-04-04 18:46:24 +02:00
160 lines
7.2 KiB
Java
160 lines
7.2 KiB
Java
package com.plotsquared.sponge;
|
|
|
|
import org.spongepowered.api.block.BlockState;
|
|
import org.spongepowered.api.block.BlockTypes;
|
|
import org.spongepowered.api.world.World;
|
|
import org.spongepowered.api.world.extent.ImmutableBiomeArea;
|
|
import org.spongepowered.api.world.extent.MutableBlockVolume;
|
|
import org.spongepowered.api.world.gen.GeneratorPopulator;
|
|
|
|
import com.flowpowered.math.vector.Vector3i;
|
|
|
|
public class PlotGen implements GeneratorPopulator {
|
|
|
|
public String worldname;
|
|
public long seed;
|
|
|
|
|
|
public final int PLOT_HEIGHT = 64; // Plot height of 64
|
|
public final int PLOT_WIDTH = 42; // Plot width of 42
|
|
public final int ROAD_WIDTH = 7; // Road width of 7
|
|
|
|
public final BlockState ROAD_BLOCK = BlockTypes.QUARTZ_BLOCK.getDefaultState(); // Quartz
|
|
public final BlockState MAIN_BLOCK = BlockTypes.STONE.getDefaultState(); // Stone
|
|
public final BlockState WALL_BLOCK = BlockTypes.BEDROCK.getDefaultState(); // Bedrock
|
|
public final BlockState BORDER_BLOCK = BlockTypes.STONE_SLAB.getDefaultState(); // Stone slab
|
|
public final BlockState[] FLOOR_BLOCK = new BlockState[] {BlockTypes.GRASS.getDefaultState(), BlockTypes.SPONGE.getDefaultState()}; // Grass and sponge
|
|
|
|
public final int total_width;
|
|
public final int road_width_lower;
|
|
public final int road_width_upper;
|
|
|
|
/**
|
|
* I'm using my PseudoRandom class as it's more efficient and we don't need secure randomness
|
|
*/
|
|
public final PseudoRandom RANDOM = new PseudoRandom();
|
|
private SpongeMain main;
|
|
|
|
public PlotGen(SpongeMain main, String worldname, long seed) {
|
|
this.main = main;
|
|
this.worldname = worldname;
|
|
this.seed = seed;
|
|
|
|
total_width = PLOT_WIDTH + ROAD_WIDTH;
|
|
|
|
// Calculating the bottom and top road portions (this is for a PlotSquared compatible generator, but you can have any offset you want)
|
|
if ((ROAD_WIDTH % 2) == 0) {
|
|
road_width_lower = ROAD_WIDTH / 2 - 1;
|
|
} else {
|
|
road_width_lower = ROAD_WIDTH / 2;
|
|
}
|
|
road_width_upper = road_width_lower + PLOT_WIDTH + 1;
|
|
main.log("LOADED GEN FOR: " + worldname);
|
|
}
|
|
|
|
/**
|
|
* This simple pairing function is used for the seed for each chunk,
|
|
* - This is useful if you want generation to appear random, but be the same each time
|
|
* - You could also use a simple hash function like `return x + y * 31` - but this looks fancier
|
|
* @param x
|
|
* @param y
|
|
* @return
|
|
*/
|
|
public int pair(int x, int y) {
|
|
long hash;
|
|
if (x >= 0) {
|
|
if (y >= 0) {
|
|
hash = (x * x) + (3 * x) + (2 * x * y) + y + (y * y) + 2;
|
|
} else {
|
|
final int y1 = -y;
|
|
hash = (x * x) + (3 * x) + (2 * x * y1) + y1 + (y1 * y1) + 1;
|
|
}
|
|
} else {
|
|
final int x1 = -x;
|
|
if (y >= 0) {
|
|
hash = -((x1 * x1) + (3 * x1) + (2 * x1 * y) + y + (y * y));
|
|
} else {
|
|
final int y1 = -y;
|
|
hash = -((x1 * x1) + (3 * x1) + (2 * x1 * y1) + y1 + (y1 * y1) + 1);
|
|
}
|
|
}
|
|
return (int) (hash % Integer.MAX_VALUE);
|
|
}
|
|
|
|
@Override
|
|
public void populate(World world, MutableBlockVolume buffer, ImmutableBiomeArea biomeBase) {
|
|
try {
|
|
Vector3i min = buffer.getBlockMin();
|
|
int X = min.getX();
|
|
int Z = min.getZ();
|
|
int cx = X >> 4;
|
|
int cz = Z >> 4;
|
|
main.log("POPULATING " + worldname + " | " + cx + "," + cz);
|
|
// If you have any random elements to your generation, you will want to set the state of the random class
|
|
RANDOM.state = pair(cx, cz);
|
|
|
|
// TODO set bedrock
|
|
|
|
// We want all plots to be the same
|
|
// To do this we will need to reduce the coordinates to the same base location
|
|
// To get the world coord from a chunk coord we multiply by 16 `cx << 4`
|
|
// Then we find the remainder of that `(cx << 4) % total_width`
|
|
// We don't want negative numbers though so add the `total_width` if the remainder is less than 0
|
|
// We have to do this as the plot size will not necessarily have to be a multiple of 16, and so it won't always align to the chunk
|
|
// If the total width is a multiple of 16, you can in fact make some neat optimizations, see PlotSquaredMG source for more info
|
|
int bx = (cx << 4) % total_width + (cx < 0 ? total_width : 0);
|
|
int bz = (cz << 4) % total_width + (cz < 0 ? total_width : 0);
|
|
|
|
// This is our main loop where we go over all the columns in the chunk and set the blocks
|
|
for (int x = 0; x < 16; x++) {
|
|
for (int z = 0; z < 16; z++) {
|
|
// Getting the reduced coordinate
|
|
int xx = (x + bx);
|
|
int zz = (z + bz);
|
|
// If it's greater than the total width, we need to reduce it
|
|
// Although we reduced the chunk coordinates before, that only means the base coordinate of the chunk is reduced
|
|
// The top coordinate could still be outside our desired range
|
|
if (xx >= total_width) xx -= total_width;
|
|
if (zz >= total_width) zz -= total_width;
|
|
|
|
// ROAD
|
|
if (xx < road_width_lower || zz < road_width_lower || xx > road_width_upper || zz > road_width_upper) {
|
|
for (int y = 0; y < PLOT_HEIGHT; y++) setBlock(buffer, X + x, y, Z + z, ROAD_BLOCK);
|
|
}
|
|
// WALL
|
|
else if (xx == road_width_lower || zz == road_width_lower || xx == road_width_upper || zz == road_width_upper) {
|
|
// Set the wall block
|
|
for (int y = 0; y < PLOT_HEIGHT; y++) setBlock(buffer, X + x, y, Z + z, WALL_BLOCK);
|
|
// Set the border block (on top of the wall)
|
|
setBlock(buffer, X + x, PLOT_HEIGHT, Z + z, BORDER_BLOCK);
|
|
}
|
|
// PLOT
|
|
else {
|
|
// Set the main plot block
|
|
for (int y = 0; y < PLOT_HEIGHT - 1; y++) setBlock(buffer, X + x, y, Z + z, MAIN_BLOCK);
|
|
// Set the plot floor
|
|
setBlock(buffer, X + x, PLOT_HEIGHT - 1, Z + z, FLOOR_BLOCK);
|
|
}
|
|
}
|
|
}
|
|
main.log("SUCCESS " + worldname + " | " + cx + "," + cz);
|
|
}
|
|
catch (Exception e) {
|
|
// Normally if something goes wrong in your code it will fail silently with world generators
|
|
// Having this try/catch will help recover the exception
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
|
|
public void setBlock(MutableBlockVolume buffer, int x, int y, int z, BlockState...states) {
|
|
if (states.length == 1) {
|
|
setBlock(buffer, x, y, z, states[0]);
|
|
}
|
|
setBlock(buffer, x, y, z, states[RANDOM.random(states.length)]);
|
|
}
|
|
|
|
public void setBlock(MutableBlockVolume buffer, int x, int y, int z, BlockState state) {
|
|
buffer.setBlock(x, y, z, state);
|
|
}
|
|
}
|