PlotSquared/Bukkit/src/main/java/com/plotsquared/bukkit/queue/GenChunk.java

273 lines
9.1 KiB
Java
Raw Normal View History

/*
* _____ _ _ _____ _
* | __ \| | | | / ____| | |
* | |__) | | ___ | |_| (___ __ _ _ _ __ _ _ __ ___ __| |
* | ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` |
* | | | | (_) | |_ ____) | (_| | |_| | (_| | | | __/ (_| |
* |_| |_|\___/ \__|_____/ \__, |\__,_|\__,_|_| \___|\__,_|
* | |
* |_|
* PlotSquared plot management system for Minecraft
2022-01-02 22:22:19 +01:00
* Copyright (C) 2014 - 2022 IntellectualSites
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
2020-08-15 14:59:29 +02:00
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
2020-04-15 21:26:54 +02:00
package com.plotsquared.bukkit.queue;
2016-02-10 19:59:51 +01:00
2020-04-30 12:01:52 +02:00
import com.google.common.base.Preconditions;
2020-04-15 21:26:54 +02:00
import com.plotsquared.bukkit.util.BukkitBlockUtil;
2020-04-11 02:19:18 +02:00
import com.plotsquared.bukkit.util.BukkitUtil;
2020-04-15 21:26:54 +02:00
import com.plotsquared.core.location.ChunkWrapper;
import com.plotsquared.core.location.Location;
import com.plotsquared.core.queue.ScopedQueueCoordinator;
import com.plotsquared.core.util.ChunkUtil;
2020-04-15 21:26:54 +02:00
import com.plotsquared.core.util.PatternUtil;
import com.sk89q.worldedit.bukkit.BukkitAdapter;
import com.sk89q.worldedit.function.pattern.Pattern;
import com.sk89q.worldedit.world.biome.BiomeType;
2018-12-26 17:05:37 +01:00
import com.sk89q.worldedit.world.block.BaseBlock;
2019-11-04 20:58:24 +01:00
import com.sk89q.worldedit.world.block.BlockState;
import com.sk89q.worldedit.world.block.BlockTypes;
import org.bukkit.Bukkit;
2016-02-10 19:59:51 +01:00
import org.bukkit.Chunk;
import org.bukkit.World;
import org.bukkit.block.Biome;
import org.bukkit.generator.ChunkGenerator.BiomeGrid;
import org.bukkit.generator.ChunkGenerator.ChunkData;
import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
2018-08-10 17:01:10 +02:00
import java.util.Arrays;
public class GenChunk extends ScopedQueueCoordinator {
2016-02-10 19:59:51 +01:00
public final Biome[] biomes;
2019-11-04 20:55:55 +01:00
public BlockState[][] result;
public BiomeGrid biomeGrid;
2016-06-13 06:47:50 +02:00
public Chunk chunk;
public String world;
public int chunkX;
public int chunkZ;
2020-07-17 17:24:45 +02:00
private ChunkData chunkData = null;
2016-06-13 06:47:50 +02:00
public GenChunk() {
super(null, Location.at("", 0, 0, 0), Location.at("", 15, 255, 15));
2016-02-10 19:59:51 +01:00
this.biomes = Biome.values();
}
public @Nullable ChunkData getChunkData() {
return this.chunkData;
}
2020-07-24 18:00:08 +02:00
/**
* Set the internal Bukkit chunk data
2020-08-16 12:34:08 +02:00
*
* @param chunkData Bukkit ChunkData
2020-07-24 18:00:08 +02:00
*/
public void setChunkData(@NonNull ChunkData chunkData) {
this.chunkData = chunkData;
}
public @NonNull Chunk getChunk() {
2016-06-13 06:47:50 +02:00
if (chunk == null) {
World worldObj = BukkitUtil.getWorld(world);
if (worldObj != null) {
this.chunk = worldObj.getChunkAt(chunkX, chunkZ);
2016-06-13 06:47:50 +02:00
}
}
return chunk;
}
2020-07-24 18:00:08 +02:00
/**
* Set the chunk being represented
2020-08-16 12:34:08 +02:00
*
* @param chunk Bukkit Chunk
2020-07-24 18:00:08 +02:00
*/
public void setChunk(@NonNull Chunk chunk) {
2019-04-24 00:48:22 +02:00
this.chunk = chunk;
}
2020-07-24 18:00:08 +02:00
/**
* Set the world and XZ of the chunk being represented via {@link ChunkWrapper}
2020-08-16 12:34:08 +02:00
*
* @param wrap P2 ChunkWrapper
2020-07-24 18:00:08 +02:00
*/
public void setChunk(@NonNull ChunkWrapper wrap) {
2018-08-10 17:01:10 +02:00
chunk = null;
world = wrap.world;
chunkX = wrap.x;
chunkZ = wrap.z;
2018-08-10 17:01:10 +02:00
}
@Override
public void fillBiome(@NonNull BiomeType biomeType) {
if (biomeGrid == null) {
2016-06-13 06:47:50 +02:00
return;
}
Biome biome = BukkitAdapter.adapt(biomeType);
2020-07-24 18:00:08 +02:00
for (int y = 0; y < 256; y++) {
for (int x = 0; x < 16; x++) {
for (int z = 0; z < 16; z++) {
this.biomeGrid.setBiome(x, y, z, biome);
}
2016-06-13 06:47:50 +02:00
}
2016-02-10 19:59:51 +01:00
}
}
@Override
public void setCuboid(@NonNull Location pos1, @NonNull Location pos2, @NonNull BlockState block) {
2020-07-24 17:24:53 +02:00
if (result != null && pos1.getX() == 0 && pos1.getZ() == 0 && pos2.getX() == 15 && pos2.getZ() == 15) {
2017-09-25 08:57:42 +02:00
for (int y = pos1.getY(); y <= pos2.getY(); y++) {
int layer = y >> 4;
2019-11-04 20:55:55 +01:00
BlockState[] data = result[layer];
2017-09-25 08:57:42 +02:00
if (data == null) {
2019-11-04 20:55:55 +01:00
result[layer] = data = new BlockState[4096];
2017-09-25 08:57:42 +02:00
}
int start = y << 8;
int end = start + 256;
Arrays.fill(data, start, end, block);
2017-09-25 08:57:42 +02:00
}
}
int minX = Math.min(pos1.getX(), pos2.getX());
int minY = Math.min(pos1.getY(), pos2.getY());
int minZ = Math.min(pos1.getZ(), pos2.getZ());
int maxX = Math.max(pos1.getX(), pos2.getX());
int maxY = Math.max(pos1.getY(), pos2.getY());
int maxZ = Math.max(pos1.getZ(), pos2.getZ());
2020-07-24 17:24:53 +02:00
chunkData.setRegion(minX, minY, minZ, maxX + 1, maxY + 1, maxZ + 1, BukkitAdapter.adapt(block));
2017-09-25 08:57:42 +02:00
}
@Override
public boolean setBiome(int x, int z, @NonNull BiomeType biomeType) {
return setBiome(x, z, BukkitAdapter.adapt(biomeType));
2016-06-13 06:47:50 +02:00
}
2020-07-24 18:00:08 +02:00
/**
* Set the in the whole column of XZ
2020-08-16 12:34:08 +02:00
*
* @param x Relative x location within the chunk (0 - 15)
* @param z Relative z location within the chunk (0 - 15)
2020-08-16 12:34:08 +02:00
* @param biome Bukkit biome to set
* @return if successful
2020-07-24 18:00:08 +02:00
*/
public boolean setBiome(int x, int z, @NonNull Biome biome) {
if (this.biomeGrid != null) {
2020-07-24 18:00:08 +02:00
for (int y = 0; y < 256; y++) {
this.setBiome(x, y, z, biome);
}
return true;
}
return false;
}
public boolean setBiome(int x, int y, int z, @NonNull Biome biome) {
if (this.biomeGrid != null) {
2020-07-24 18:00:08 +02:00
this.biomeGrid.setBiome(x, y, z, biome);
2016-06-13 06:47:50 +02:00
return true;
2016-02-10 19:59:51 +01:00
}
2016-06-13 06:47:50 +02:00
return false;
2016-02-10 19:59:51 +01:00
}
@Override
public boolean setBlock(int x, int y, int z, @NonNull Pattern pattern) {
2020-07-24 17:24:53 +02:00
return setBlock(x, y, z, PatternUtil.apply(Preconditions.checkNotNull(pattern, "Pattern may not be null"), x, y, z));
}
@Override
public boolean setBlock(int x, int y, int z, @NonNull BlockState id) {
2016-03-23 02:41:37 +01:00
if (this.result == null) {
2019-11-04 20:55:55 +01:00
this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id));
2016-06-13 06:47:50 +02:00
return true;
2016-02-10 19:59:51 +01:00
}
2019-11-04 20:55:55 +01:00
this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id));
this.storeCache(x, y, z, id);
return true;
}
private void storeCache(final int x, final int y, final int z, final @NonNull BlockState id) {
int i = y >> 4;
2019-11-04 20:55:55 +01:00
BlockState[] v = this.result[i];
2016-02-10 19:59:51 +01:00
if (v == null) {
2019-11-04 20:55:55 +01:00
this.result[i] = v = new BlockState[4096];
2016-02-10 19:59:51 +01:00
}
int j = ChunkUtil.getJ(x, y, z);
v[j] = id;
2016-06-13 06:47:50 +02:00
}
@Override
public boolean setBlock(int x, int y, int z, @NonNull BaseBlock id) {
if (this.result == null) {
this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id));
return true;
}
this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id));
this.storeCache(x, y, z, id.toImmutableState());
return true;
2018-12-26 17:05:37 +01:00
}
@Override
public @Nullable BlockState getBlock(int x, int y, int z) {
int i = y >> 4;
2016-06-13 06:47:50 +02:00
if (result == null) {
2019-11-04 20:55:55 +01:00
return BukkitBlockUtil.get(chunkData.getType(x, y, z));
2016-06-13 06:47:50 +02:00
}
2019-11-04 20:55:55 +01:00
BlockState[] array = result[i];
2016-06-13 06:47:50 +02:00
if (array == null) {
return BlockTypes.AIR.getDefaultState();
2016-06-13 06:47:50 +02:00
}
int j = ChunkUtil.getJ(x, y, z);
return array[j];
2016-06-13 06:47:50 +02:00
}
public int getX() {
return chunk == null ? chunkX : chunk.getX();
2016-06-13 06:47:50 +02:00
}
public int getZ() {
return chunk == null ? chunkZ : chunk.getZ();
2016-02-10 19:59:51 +01:00
}
@Override
public com.sk89q.worldedit.world.@NonNull World getWorld() {
2020-07-24 17:24:53 +02:00
return chunk == null ? BukkitAdapter.adapt(Bukkit.getWorld(world)) : BukkitAdapter.adapt(chunk.getWorld());
2016-06-13 06:47:50 +02:00
}
@Override
public @NonNull Location getMax() {
return Location.at(getWorld().getName(), 15 + (getX() << 4), 255, 15 + (getZ() << 4));
2016-06-13 06:47:50 +02:00
}
@Override
public @NonNull Location getMin() {
return Location.at(getWorld().getName(), getX() << 4, 0, getZ() << 4);
2016-06-13 06:47:50 +02:00
}
public @NonNull GenChunk clone() {
GenChunk toReturn = new GenChunk();
2016-03-23 02:41:37 +01:00
if (this.result != null) {
for (int i = 0; i < this.result.length; i++) {
2019-11-04 20:55:55 +01:00
BlockState[] matrix = this.result[i];
2016-02-10 19:59:51 +01:00
if (matrix != null) {
2019-11-04 20:55:55 +01:00
toReturn.result[i] = new BlockState[matrix.length];
2016-02-10 19:59:51 +01:00
System.arraycopy(matrix, 0, toReturn.result[i], 0, matrix.length);
}
}
}
toReturn.chunkData = this.chunkData;
2016-02-10 19:59:51 +01:00
return toReturn;
}
2016-02-10 19:59:51 +01:00
}