forked from SteamWar/SteamWar
Add DynamicRegionVisualizer
This commit is contained in:
+190
-14
@@ -20,20 +20,25 @@
|
||||
package de.steamwar.bausystem.region;
|
||||
|
||||
import de.steamwar.bausystem.BauSystem;
|
||||
import de.steamwar.bausystem.region.dynamic.DynamicRegion;
|
||||
import de.steamwar.bausystem.region.dynamic.DynamicRegionRepository;
|
||||
import de.steamwar.bausystem.region.dynamic.RegionConstructorData;
|
||||
import de.steamwar.bausystem.region.dynamic.Tile;
|
||||
import de.steamwar.bausystem.utils.PasteBuilder;
|
||||
import de.steamwar.core.SWPlayer;
|
||||
import de.steamwar.entity.*;
|
||||
import de.steamwar.inventory.SWInventory;
|
||||
import de.steamwar.inventory.SWItem;
|
||||
import de.steamwar.inventory.SWListInv;
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.Location;
|
||||
import org.bukkit.Material;
|
||||
import org.bukkit.entity.Player;
|
||||
import org.bukkit.event.EventHandler;
|
||||
import org.bukkit.event.Listener;
|
||||
import org.bukkit.event.inventory.InventoryType;
|
||||
import org.bukkit.event.player.PlayerMoveEvent;
|
||||
|
||||
import java.util.Set;
|
||||
import java.util.*;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
public class DynamicRegionVisualizer implements SWPlayer.Component, Listener {
|
||||
@@ -43,6 +48,8 @@ public class DynamicRegionVisualizer implements SWPlayer.Component, Listener {
|
||||
private Location sourceLocation;
|
||||
private Tile sourceTile;
|
||||
|
||||
private Placement placement = null;
|
||||
|
||||
public DynamicRegionVisualizer() {
|
||||
this.entityServer = new REntityServer();
|
||||
}
|
||||
@@ -101,6 +108,21 @@ public class DynamicRegionVisualizer implements SWPlayer.Component, Listener {
|
||||
}
|
||||
}
|
||||
|
||||
private void resetTiles(Set<Tile> tiles) {
|
||||
entityServer.getEntitiesByType(CTile.class)
|
||||
.stream()
|
||||
.filter(cTile -> tiles.contains(cTile.tile))
|
||||
.peek(CTile::die)
|
||||
.map(cTile -> cTile.tile)
|
||||
.forEach(tile -> {
|
||||
new CTile(entityServer, tile);
|
||||
});
|
||||
|
||||
if (placement != null) {
|
||||
placement.check();
|
||||
}
|
||||
}
|
||||
|
||||
private class CTile extends CEntity {
|
||||
|
||||
private final Tile tile;
|
||||
@@ -109,7 +131,8 @@ public class DynamicRegionVisualizer implements SWPlayer.Component, Listener {
|
||||
super(server);
|
||||
this.tile = tile;
|
||||
|
||||
Material material = switch (DynamicRegionSystem.INSTANCE.get(tile).getType()) {
|
||||
RegionType regionType = DynamicRegionSystem.INSTANCE.get(tile).getType();
|
||||
Material material = switch (regionType) {
|
||||
case SPAWN, SPAWN_EXTENSION -> Material.LODESTONE;
|
||||
case SPAWN_PATH, PATH -> Material.DIRT_PATH;
|
||||
case DRY, DRY_SPECIAL -> Material.IRON_BLOCK;
|
||||
@@ -124,23 +147,176 @@ public class DynamicRegionVisualizer implements SWPlayer.Component, Listener {
|
||||
spawn.setShadowed(false);
|
||||
entities.add(spawn);
|
||||
} else if (tile.equals(sourceTile)) {
|
||||
CCubedTextDisplay spawn = new CCubedTextDisplay(entityServer, location);
|
||||
spawn.setText("§eORIGIN");
|
||||
spawn.setBackgroundColor(0);
|
||||
spawn.setShadowed(false);
|
||||
entities.add(spawn);
|
||||
CCubedTextDisplay origin = new CCubedTextDisplay(entityServer, location);
|
||||
origin.setText("§eORIGIN");
|
||||
origin.setBackgroundColor(0);
|
||||
origin.setShadowed(false);
|
||||
entities.add(origin);
|
||||
}
|
||||
|
||||
RBlockDisplay entity = new RBlockDisplay(entityServer, location);
|
||||
entity.setBlock(material.createBlockData());
|
||||
entities.add(entity);
|
||||
RBlockDisplay blockDisplay = new RBlockDisplay(entityServer, location);
|
||||
blockDisplay.setBlock(material.createBlockData());
|
||||
entities.add(blockDisplay);
|
||||
|
||||
RInteraction interaction = new RInteraction(entityServer, location.clone().add(0.5, 0, 0.5));
|
||||
interaction.setInteraction((player, entityAction) -> {
|
||||
SWInventory swInv = new SWInventory(player, () -> Bukkit.createInventory(null, InventoryType.DROPPER, tile.toString()));
|
||||
swInv.open();
|
||||
if (placement != null) {
|
||||
if (regionType.isGlobal()) placement.click(tile);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!regionType.isGlobal()) {
|
||||
SWInventory inv = new SWInventory(player, 9, "Delete Region: " + tile.display());
|
||||
inv.setItem(0, new SWItem(SWItem.getDye(10), "§8Cancel", click -> {
|
||||
player.closeInventory();
|
||||
}));
|
||||
inv.setItem(8, new SWItem(SWItem.getDye(1), "§cDelete", click -> {
|
||||
player.closeInventory();
|
||||
Region region = DynamicRegionSystem.INSTANCE.get(tile);
|
||||
Set<Tile> tiles = DynamicRegionSystem.INSTANCE.getTilesOfRegion(region);
|
||||
region.delete();
|
||||
|
||||
SWPlayer.allWithSingleComponent(DynamicRegionVisualizer.class)
|
||||
.forEach(pair -> {
|
||||
pair.getComponent().resetTiles(tiles);
|
||||
});
|
||||
}));
|
||||
inv.open();
|
||||
} else {
|
||||
List<SWListInv.SWListEntry<Map.Entry<Class<? extends DynamicRegion>, RegionConstructorData>>> entries = new ArrayList<>();
|
||||
DynamicRegionSystem.constructorDataMap.entrySet()
|
||||
.stream()
|
||||
.filter(entry -> entry.getValue().placeable())
|
||||
.sorted(Comparator.comparing(entry -> entry.getValue().name()))
|
||||
.forEach(entry -> {
|
||||
entries.add(new SWListInv.SWListEntry<>(new SWItem(entry.getValue().material(), entry.getValue().name()), entry));
|
||||
});
|
||||
|
||||
SWListInv<Map.Entry<Class<? extends DynamicRegion>, RegionConstructorData>> listInv = new SWListInv<>(player, "Select Region for: " + tile.display(), entries, (click, entry) -> {
|
||||
new Placement(entry.getKey(), entry.getValue(), tile);
|
||||
player.closeInventory();
|
||||
});
|
||||
listInv.open();
|
||||
}
|
||||
});
|
||||
entities.add(entity);
|
||||
entities.add(interaction);
|
||||
}
|
||||
}
|
||||
|
||||
private class Placement {
|
||||
private final Class<? extends DynamicRegion> regionType;
|
||||
private final RegionConstructorData constructorData;
|
||||
|
||||
private CWireframe wireframe;
|
||||
private Tile sourceTile;
|
||||
private int dx;
|
||||
private int dz;
|
||||
private boolean valid = false;
|
||||
|
||||
private Location getMinLocation() {
|
||||
Tile tile = sourceTile.add(-DynamicRegionVisualizer.this.sourceTile.getTileX(), -DynamicRegionVisualizer.this.sourceTile.getTileZ()).orElseThrow();
|
||||
return sourceLocation.clone().add(tile.getTileX(), 0, tile.getTileZ());
|
||||
}
|
||||
|
||||
private Location getMaxLocation() {
|
||||
Tile tile = sourceTile.add(-DynamicRegionVisualizer.this.sourceTile.getTileX(), -DynamicRegionVisualizer.this.sourceTile.getTileZ()).orElseThrow();
|
||||
return sourceLocation.clone().add(tile.getTileX(), 0, tile.getTileZ()).add(dx, 0, dz);
|
||||
}
|
||||
|
||||
public Placement(Class<? extends DynamicRegion> regionType, RegionConstructorData constructorData, Tile sourceTile) {
|
||||
this.regionType = regionType;
|
||||
this.constructorData = constructorData;
|
||||
this.sourceTile = sourceTile;
|
||||
dx = constructorData.widthX() / Tile.tileSize - 1;
|
||||
dz = constructorData.widthZ() / Tile.tileSize - 1;
|
||||
if (dx == 0 && dz == 0) {
|
||||
place();
|
||||
return;
|
||||
}
|
||||
|
||||
placement = this;
|
||||
wireframe = new CWireframe(entityServer);
|
||||
check();
|
||||
}
|
||||
|
||||
private void check() {
|
||||
wireframe.setPos1And2(getMinLocation(), getMaxLocation());
|
||||
for (int x = 0; x <= dx; x++) {
|
||||
for (int z = 0; z <= dz; z++) {
|
||||
if (!DynamicRegionSystem.INSTANCE.get(sourceTile.add(x, z).orElseThrow()).getType().isGlobal()) {
|
||||
wireframe.setBlock(Material.RED_CONCRETE.createBlockData());
|
||||
valid = false;
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
wireframe.setBlock(Material.LIME_CONCRETE.createBlockData());
|
||||
valid = true;
|
||||
}
|
||||
|
||||
public void click(Tile tile) {
|
||||
if (sourceTile.getTileX() >= tile.getTileX() && sourceTile.getTileX() + dx <= tile.getTileX()) {
|
||||
if (sourceTile.getTileZ() >= tile.getTileZ() && sourceTile.getTileZ() + dz <= tile.getTileZ()) {
|
||||
if (valid) {
|
||||
SWInventory inv = new SWInventory(player, 9, "Place Region: " + constructorData.name());
|
||||
inv.setItem(0, new SWItem(SWItem.getDye(1), "§cDeselect", click -> {
|
||||
placement = null;
|
||||
wireframe.die();
|
||||
player.closeInventory();
|
||||
}));
|
||||
inv.setItem(8, new SWItem(SWItem.getDye(10), "§aPlace", click -> {
|
||||
player.closeInventory();
|
||||
place();
|
||||
}));
|
||||
inv.open();
|
||||
}
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (dx == 0 && dz == 0) {
|
||||
sourceTile = tile;
|
||||
} else {
|
||||
Set<Tile> tiles = new HashSet<>();
|
||||
for (int x = 0; x <= dx; x++) {
|
||||
for (int z = 0; z <= dz; z++) {
|
||||
tiles.add(Tile.fromTile(x + sourceTile.getTileX(), z + sourceTile.getTileZ()).orElseThrow());
|
||||
}
|
||||
}
|
||||
Tile selected = tiles.stream().min(Comparator.comparing(current -> {
|
||||
int dx = current.getTileX() - tile.getTileX();
|
||||
int dz = current.getTileZ() - tile.getTileZ();
|
||||
return dx * dx + dz * dz;
|
||||
}))
|
||||
.orElseThrow();
|
||||
|
||||
int dx = tile.getTileX() - selected.getTileX();
|
||||
int dz = tile.getTileZ() - selected.getTileZ();
|
||||
sourceTile = sourceTile.add(dx, dz).orElseThrow();
|
||||
}
|
||||
check();
|
||||
}
|
||||
|
||||
private void place() {
|
||||
DynamicRegion dynamicRegion = DynamicRegionRepository.constructRegion(regionType, UUID.randomUUID(), sourceTile.getMinX(), sourceTile.getMinZ());
|
||||
if (dynamicRegion == null) {
|
||||
// TODO: Give error to user
|
||||
return;
|
||||
}
|
||||
|
||||
dynamicRegion.getArea().place(new PasteBuilder(new PasteBuilder.FileProvider(dynamicRegion.getArea().getResetFile())), false);
|
||||
dynamicRegion.updateNeighbours();
|
||||
|
||||
placement = null;
|
||||
if (wireframe != null) wireframe.die();
|
||||
Set<Tile> tiles = new HashSet<>();
|
||||
for (int x = 0; x <= dx; x++) {
|
||||
for (int z = 0; z <= dz; z++) {
|
||||
tiles.add(Tile.fromTile(x + sourceTile.getTileX(), z + sourceTile.getTileZ()).orElseThrow());
|
||||
}
|
||||
}
|
||||
SWPlayer.allWithSingleComponent(DynamicRegionVisualizer.class)
|
||||
.forEach(pair -> pair.getComponent().resetTiles(tiles));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user