geforkt von Mirrors/FastAsyncWorldEdit
8108d0a9
Dieser Commit ist enthalten in:
Ursprung
905fbf5a0b
Commit
ff94a1e5ed
@ -9,25 +9,14 @@ import com.boydti.fawe.object.number.MutableLong;
|
|||||||
import com.boydti.fawe.util.MainUtil;
|
import com.boydti.fawe.util.MainUtil;
|
||||||
import com.boydti.fawe.util.MathMan;
|
import com.boydti.fawe.util.MathMan;
|
||||||
import com.boydti.fawe.util.ReflectionUtils;
|
import com.boydti.fawe.util.ReflectionUtils;
|
||||||
import com.sk89q.jnbt.CompoundTag;
|
import com.sk89q.jnbt.*;
|
||||||
import com.sk89q.jnbt.IntTag;
|
|
||||||
import com.sk89q.jnbt.ListTag;
|
|
||||||
import com.sk89q.jnbt.NBTConstants;
|
|
||||||
import com.sk89q.jnbt.NBTInputStream;
|
|
||||||
import com.sk89q.jnbt.NBTOutputStream;
|
|
||||||
import com.sk89q.jnbt.Tag;
|
|
||||||
import com.sk89q.worldedit.world.biome.BiomeType;
|
import com.sk89q.worldedit.world.biome.BiomeType;
|
||||||
import com.sk89q.worldedit.world.biome.BiomeTypes;
|
import com.sk89q.worldedit.world.biome.BiomeTypes;
|
||||||
|
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.DataOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.*;
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.UUID;
|
|
||||||
import java.util.function.BiConsumer;
|
import java.util.function.BiConsumer;
|
||||||
|
|
||||||
public class MCAChunk extends FaweChunk<Void> {
|
public class MCAChunk extends FaweChunk<Void> {
|
||||||
@ -88,9 +77,6 @@ public class MCAChunk extends FaweChunk<Void> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void write(NBTOutputStream nbtOut) throws IOException {
|
public void write(NBTOutputStream nbtOut) throws IOException {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
nbtOut.writeNamedTagName("", NBTConstants.TYPE_COMPOUND);
|
nbtOut.writeNamedTagName("", NBTConstants.TYPE_COMPOUND);
|
||||||
nbtOut.writeLazyCompoundTag("Level", out -> {
|
nbtOut.writeLazyCompoundTag("Level", out -> {
|
||||||
out.writeNamedTag("V", (byte) 1);
|
out.writeNamedTag("V", (byte) 1);
|
||||||
@ -116,9 +102,12 @@ public class MCAChunk extends FaweChunk<Void> {
|
|||||||
}
|
}
|
||||||
out.writeNamedTag("HeightMap", heightMap);
|
out.writeNamedTag("HeightMap", heightMap);
|
||||||
out.writeNamedTagName("Sections", NBTConstants.TYPE_LIST);
|
out.writeNamedTagName("Sections", NBTConstants.TYPE_LIST);
|
||||||
nbtOut.writeByte(NBTConstants.TYPE_COMPOUND);
|
nbtOut.getOutputStream().writeByte(NBTConstants.TYPE_COMPOUND);
|
||||||
int len = (int) Arrays.stream(ids).filter(Objects::nonNull).count();
|
int len = 0;
|
||||||
nbtOut.writeInt(len);
|
for (int[] id : ids) {
|
||||||
|
if (id != null) len++;
|
||||||
|
}
|
||||||
|
nbtOut.getOutputStream().writeInt(len);
|
||||||
for (int layer = 0; layer < ids.length; layer++) {
|
for (int layer = 0; layer < ids.length; layer++) {
|
||||||
int[] idLayer = ids[layer];
|
int[] idLayer = ids[layer];
|
||||||
if (idLayer == null) {
|
if (idLayer == null) {
|
||||||
@ -610,14 +599,14 @@ public class MCAChunk extends FaweChunk<Void> {
|
|||||||
public BiomeType[] getBiomeArray() {
|
public BiomeType[] getBiomeArray() {
|
||||||
BiomeType[] arr = new BiomeType[256];
|
BiomeType[] arr = new BiomeType[256];
|
||||||
for (int i = 0; i < arr.length; i++) {
|
for (int i = 0; i < arr.length; i++) {
|
||||||
arr[i] = BiomeTypes.register(biomes[i]);
|
arr[i] = BiomeTypes.get(biomes[i]);
|
||||||
}
|
}
|
||||||
return arr;
|
return arr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public BiomeType getBiomeType(int x, int z) {
|
public BiomeType getBiomeType(int x, int z) {
|
||||||
return BiomeTypes.register(biomes[(x & 15) + ((z & 15) << 4)]);
|
return BiomeTypes.get(biomes[(x & 15) + ((z & 15) << 4)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -41,7 +41,8 @@ public class FaweTimer implements Runnable {
|
|||||||
if (tick < lastGetTPSTick + tickInterval) {
|
if (tick < lastGetTPSTick + tickInterval) {
|
||||||
return lastGetTPSValue;
|
return lastGetTPSValue;
|
||||||
}
|
}
|
||||||
double total = Arrays.stream(history).sum();
|
double total = 0;
|
||||||
|
for (double v : history) total += v;
|
||||||
lastGetTPSValue = total / history.length;
|
lastGetTPSValue = total / history.length;
|
||||||
lastGetTPSTick = tick;
|
lastGetTPSTick = tick;
|
||||||
return lastGetTPSValue;
|
return lastGetTPSValue;
|
||||||
|
@ -408,7 +408,6 @@ public class PlatformManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Tool tool = session.getTool(player.getItemInHand(HandSide.MAIN_HAND).getType());
|
|
||||||
switch (event.getInputType()) {
|
switch (event.getInputType()) {
|
||||||
case PRIMARY: {
|
case PRIMARY: {
|
||||||
if (getConfiguration().navigationWandMaxDistance > 0 && player.getItemInHand(HandSide.MAIN_HAND).getType().getId().equals(getConfiguration().navigationWand)) {
|
if (getConfiguration().navigationWandMaxDistance > 0 && player.getItemInHand(HandSide.MAIN_HAND).getType().getId().equals(getConfiguration().navigationWand)) {
|
||||||
@ -426,7 +425,7 @@ public class PlatformManager {
|
|||||||
event.setCancelled(true);
|
event.setCancelled(true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
Tool tool = session.getTool(player);
|
||||||
if (tool instanceof DoubleActionTraceTool && tool.canUse(player)) {
|
if (tool instanceof DoubleActionTraceTool && tool.canUse(player)) {
|
||||||
FawePlayer<?> fp = FawePlayer.wrap(player);
|
FawePlayer<?> fp = FawePlayer.wrap(player);
|
||||||
fp.runAsyncIfFree(() -> reset((DoubleActionTraceTool) tool).actSecondary(queryCapability(Capability.WORLD_EDITING), getConfiguration(), player, session));
|
fp.runAsyncIfFree(() -> reset((DoubleActionTraceTool) tool).actSecondary(queryCapability(Capability.WORLD_EDITING), getConfiguration(), player, session));
|
||||||
@ -450,7 +449,7 @@ public class PlatformManager {
|
|||||||
event.setCancelled(true);
|
event.setCancelled(true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
Tool tool = session.getTool(player);
|
||||||
if (tool instanceof TraceTool && tool.canUse(player)) {
|
if (tool instanceof TraceTool && tool.canUse(player)) {
|
||||||
FawePlayer<?> fp = FawePlayer.wrap(player);
|
FawePlayer<?> fp = FawePlayer.wrap(player);
|
||||||
//todo this needs to be fixed so the event is canceled after actPrimary is used and returns true
|
//todo this needs to be fixed so the event is canceled after actPrimary is used and returns true
|
||||||
|
@ -62,7 +62,9 @@ import java.util.Map;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Reads schematic files that are compatible with MCEdit and other editors.
|
* Reads schematic files that are compatible with MCEdit and other editors.
|
||||||
|
* @deprecated Use SchematicStreamer
|
||||||
*/
|
*/
|
||||||
|
@Deprecated
|
||||||
public class MCEditSchematicReader extends NBTSchematicReader {
|
public class MCEditSchematicReader extends NBTSchematicReader {
|
||||||
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(MCEditSchematicReader.class);
|
private static final Logger log = LoggerFactory.getLogger(MCEditSchematicReader.class);
|
||||||
@ -355,6 +357,10 @@ public class MCEditSchematicReader extends NBTSchematicReader {
|
|||||||
|
|
||||||
private String convertBlockEntityId(String id) {
|
private String convertBlockEntityId(String id) {
|
||||||
switch (id) {
|
switch (id) {
|
||||||
|
case "Chest":
|
||||||
|
return "chest";
|
||||||
|
case "Sign":
|
||||||
|
return "sign";
|
||||||
case "Cauldron":
|
case "Cauldron":
|
||||||
return "brewing_stand";
|
return "brewing_stand";
|
||||||
case "Control":
|
case "Control":
|
||||||
|
@ -125,49 +125,22 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private String fix(String palettePart) {
|
||||||
|
if (fixer == null || dataVersion == -1) return palettePart;
|
||||||
|
return fixer.fixUp(DataFixer.FixTypes.BLOCK_STATE, palettePart, dataVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
private CompoundTag fixBlockEntity(CompoundTag tag) {
|
||||||
|
if (fixer == null || dataVersion == -1) return tag;
|
||||||
|
return fixer.fixUp(DataFixer.FixTypes.BLOCK_ENTITY, tag, dataVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
private CompoundTag fixEntity(CompoundTag tag) {
|
||||||
|
if (fixer == null || dataVersion == -1) return tag;
|
||||||
|
return fixer.fixUp(DataFixer.FixTypes.ENTITY, tag, dataVersion);
|
||||||
|
}
|
||||||
|
|
||||||
private Clipboard reader(UUID uuid) throws IOException {
|
private Clipboard reader(UUID uuid) throws IOException {
|
||||||
NamedTag rootTag = inputStream.readNamedTag();
|
|
||||||
if (!rootTag.getName().equals("Schematic")) {
|
|
||||||
throw new IOException("Tag 'Schematic' does not exist or is not first");
|
|
||||||
}
|
|
||||||
CompoundTag schematicTag = (CompoundTag) rootTag.getTag();
|
|
||||||
|
|
||||||
// Check
|
|
||||||
Map<String, Tag> schematic = schematicTag.getValue();
|
|
||||||
|
|
||||||
int version = requireTag(schematic, "Version", IntTag.class).getValue();
|
|
||||||
final Platform platform = WorldEdit.getInstance().getPlatformManager()
|
|
||||||
.queryCapability(Capability.WORLD_EDITING);
|
|
||||||
int liveDataVersion = platform.getDataVersion();
|
|
||||||
|
|
||||||
if (version == 1) {
|
|
||||||
dataVersion = 1631; // this is a relatively safe assumption unless someone imports a schematic from 1.12, e.g. sponge 7.1-
|
|
||||||
fixer = platform.getDataFixer();
|
|
||||||
return readVersion1(uuid);
|
|
||||||
} else if (version == 2) {
|
|
||||||
dataVersion = requireTag(schematic, "DataVersion", IntTag.class).getValue();
|
|
||||||
if (dataVersion > liveDataVersion) {
|
|
||||||
log.warn("Schematic was made in a newer Minecraft version ({} > {}). Data may be incompatible.",
|
|
||||||
dataVersion, liveDataVersion);
|
|
||||||
} else if (dataVersion < liveDataVersion) {
|
|
||||||
fixer = platform.getDataFixer();
|
|
||||||
if (fixer != null) {
|
|
||||||
log.info("Schematic was made in an older Minecraft version ({} < {}), will attempt DFU.",
|
|
||||||
dataVersion, liveDataVersion);
|
|
||||||
} else {
|
|
||||||
log.info("Schematic was made in an older Minecraft version ({} < {}), but DFU is not available. Data may be incompatible.",
|
|
||||||
dataVersion, liveDataVersion);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
BlockArrayClipboard clip = readVersion1(uuid);
|
|
||||||
return readVersion2(clip, schematicTag);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new IOException("This schematic version is currently not supported");
|
|
||||||
}
|
|
||||||
|
|
||||||
private BlockArrayClipboard readVersion1(UUID uuid) throws IOException {
|
|
||||||
width = height = length = offsetX = offsetY = offsetZ = Integer.MIN_VALUE;
|
width = height = length = offsetX = offsetY = offsetZ = Integer.MIN_VALUE;
|
||||||
|
|
||||||
final BlockArrayClipboard clipboard = new BlockArrayClipboard(new CuboidRegion(BlockVector3.at(0, 0, 0), BlockVector3.at(0, 0, 0)), fc);
|
final BlockArrayClipboard clipboard = new BlockArrayClipboard(new CuboidRegion(BlockVector3.at(0, 0, 0), BlockVector3.at(0, 0, 0)), fc);
|
||||||
@ -176,6 +149,7 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
|
|
||||||
|
|
||||||
NBTStreamer streamer = new NBTStreamer(inputStream);
|
NBTStreamer streamer = new NBTStreamer(inputStream);
|
||||||
|
streamer.addReader("Schematic.DataVersion", (BiConsumer<Integer, Short>) (i, v) -> dataVersion = v);
|
||||||
streamer.addReader("Schematic.Width", (BiConsumer<Integer, Short>) (i, v) -> width = v);
|
streamer.addReader("Schematic.Width", (BiConsumer<Integer, Short>) (i, v) -> width = v);
|
||||||
streamer.addReader("Schematic.Height", (BiConsumer<Integer, Short>) (i, v) -> height = v);
|
streamer.addReader("Schematic.Height", (BiConsumer<Integer, Short>) (i, v) -> height = v);
|
||||||
streamer.addReader("Schematic.Length", (BiConsumer<Integer, Short>) (i, v) -> length = v);
|
streamer.addReader("Schematic.Length", (BiConsumer<Integer, Short>) (i, v) -> length = v);
|
||||||
@ -188,7 +162,8 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
for (Map.Entry<String, Tag> entry : v.entrySet()) {
|
for (Map.Entry<String, Tag> entry : v.entrySet()) {
|
||||||
BlockState state = null;
|
BlockState state = null;
|
||||||
try {
|
try {
|
||||||
state = BlockState.get(entry.getKey());
|
String palettePart = fix(entry.getKey());
|
||||||
|
state = BlockState.get(palettePart);
|
||||||
} catch (InputParseException e) {
|
} catch (InputParseException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
@ -196,6 +171,9 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
palette[index] = (char) state.getOrdinal();
|
palette[index] = (char) state.getOrdinal();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/// readBiomes
|
||||||
|
|
||||||
streamer.addReader("Schematic.BlockData.#", new NBTStreamer.LazyReader() {
|
streamer.addReader("Schematic.BlockData.#", new NBTStreamer.LazyReader() {
|
||||||
@Override
|
@Override
|
||||||
public void accept(Integer arrayLen, DataInputStream dis) {
|
public void accept(Integer arrayLen, DataInputStream dis) {
|
||||||
@ -224,21 +202,40 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
int x = pos[0];
|
int x = pos[0];
|
||||||
int y = pos[1];
|
int y = pos[1];
|
||||||
int z = pos[2];
|
int z = pos[2];
|
||||||
|
Map<String, Tag> values = value.getValue();
|
||||||
|
Tag id = values.get("Id");
|
||||||
|
if (id != null) {
|
||||||
|
values.put("x", new IntTag(x));
|
||||||
|
values.put("y", new IntTag(y));
|
||||||
|
values.put("z", new IntTag(z));
|
||||||
|
values.put("id", id);
|
||||||
|
}
|
||||||
|
values.remove("Id");
|
||||||
|
values.remove("Pos");
|
||||||
|
value = fixBlockEntity(value);
|
||||||
fc.setTile(x, y, z, value);
|
fc.setTile(x, y, z, value);
|
||||||
});
|
});
|
||||||
streamer.addReader("Schematic.Entities.#", (BiConsumer<Integer, CompoundTag>) (index, compound) -> {
|
streamer.addReader("Schematic.Entities.#", (BiConsumer<Integer, CompoundTag>) (index, compound) -> {
|
||||||
if (fc == null) {
|
if (fc == null) {
|
||||||
setupClipboard(0, uuid);
|
setupClipboard(0, uuid);
|
||||||
}
|
}
|
||||||
String id = compound.getString("id");
|
Map<String, Tag> value = compound.getValue();
|
||||||
if (id.isEmpty()) {
|
StringTag id = (StringTag) value.get("Id");
|
||||||
|
if (id == null) {
|
||||||
|
id = (StringTag) value.get("id");
|
||||||
|
if (id == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
value.put("id", id);
|
||||||
|
value.remove("Id");
|
||||||
|
}
|
||||||
|
|
||||||
ListTag positionTag = compound.getListTag("Pos");
|
ListTag positionTag = compound.getListTag("Pos");
|
||||||
ListTag directionTag = compound.getListTag("Rotation");
|
ListTag directionTag = compound.getListTag("Rotation");
|
||||||
EntityType type = EntityTypes.parse(id);
|
EntityType type = EntityTypes.parse(id.getValue());
|
||||||
if (type != null) {
|
if (type != null) {
|
||||||
compound.getValue().put("Id", new StringTag(type.getId()));
|
compound = fixEntity(compound);
|
||||||
BaseEntity state = new BaseEntity(type, compound);
|
BaseEntity state = new BaseEntity(type, compound);
|
||||||
fc.createEntity(clipboard, positionTag.asDouble(0), positionTag.asDouble(1), positionTag.asDouble(2), (float) directionTag.asDouble(0), (float) directionTag.asDouble(1), state);
|
fc.createEntity(clipboard, positionTag.asDouble(0), positionTag.asDouble(1), positionTag.asDouble(2), (float) directionTag.asDouble(0), (float) directionTag.asDouble(1), state);
|
||||||
} else {
|
} else {
|
||||||
@ -274,7 +271,7 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
try (FaweInputStream fis = new FaweInputStream(new LZ4BlockInputStream(new FastByteArraysInputStream(biomesOut.toByteArrays())))) {
|
try (FaweInputStream fis = new FaweInputStream(new LZ4BlockInputStream(new FastByteArraysInputStream(biomesOut.toByteArrays())))) {
|
||||||
int volume = width * length;
|
int volume = width * length;
|
||||||
for (int index = 0; index < volume; index++) {
|
for (int index = 0; index < volume; index++) {
|
||||||
fc.setBiome(index, BiomeTypes.register(fis.read()));
|
fc.setBiome(index, BiomeTypes.get(fis.read()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -283,17 +280,6 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
return clipboard;
|
return clipboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Clipboard readVersion2(BlockArrayClipboard version1, CompoundTag schematicTag) throws IOException {
|
|
||||||
Map<String, Tag> schematic = schematicTag.getValue();
|
|
||||||
if (schematic.containsKey("BiomeData")) {
|
|
||||||
readBiomes(version1, schematic);
|
|
||||||
}
|
|
||||||
if (schematic.containsKey("Entities")) {
|
|
||||||
readEntities(version1, schematic);
|
|
||||||
}
|
|
||||||
return version1;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void readBiomes(BlockArrayClipboard clipboard, Map<String, Tag> schematic) throws IOException {
|
private void readBiomes(BlockArrayClipboard clipboard, Map<String, Tag> schematic) throws IOException {
|
||||||
ByteArrayTag dataTag = requireTag(schematic, "BiomeData", ByteArrayTag.class);
|
ByteArrayTag dataTag = requireTag(schematic, "BiomeData", ByteArrayTag.class);
|
||||||
IntTag maxTag = requireTag(schematic, "BiomePaletteMax", IntTag.class);
|
IntTag maxTag = requireTag(schematic, "BiomePaletteMax", IntTag.class);
|
||||||
@ -309,7 +295,7 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
if (fixer != null) {
|
if (fixer != null) {
|
||||||
key = fixer.fixUp(DataFixer.FixTypes.BIOME, key, dataVersion);
|
key = fixer.fixUp(DataFixer.FixTypes.BIOME, key, dataVersion);
|
||||||
}
|
}
|
||||||
BiomeType biome = BiomeTypes.register(key);
|
BiomeType biome = BiomeTypes.get(key);
|
||||||
if (biome == null) {
|
if (biome == null) {
|
||||||
log.warn("Unknown biome type :" + key +
|
log.warn("Unknown biome type :" + key +
|
||||||
" in palette. Are you missing a mod or using a schematic made in a newer version of Minecraft?");
|
" in palette. Are you missing a mod or using a schematic made in a newer version of Minecraft?");
|
||||||
@ -352,37 +338,6 @@ public class SpongeSchematicReader extends NBTSchematicReader {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void readEntities(BlockArrayClipboard clipboard, Map<String, Tag> schematic) throws IOException {
|
|
||||||
List<Tag> entList = requireTag(schematic, "Entities", ListTag.class).getValue();
|
|
||||||
if (entList.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
for (Tag et : entList) {
|
|
||||||
if (!(et instanceof CompoundTag)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
CompoundTag entityTag = (CompoundTag) et;
|
|
||||||
Map<String, Tag> tags = entityTag.getValue();
|
|
||||||
String id = requireTag(tags, "Id", StringTag.class).getValue();
|
|
||||||
entityTag = entityTag.createBuilder().putString("id", id).remove("Id").build();
|
|
||||||
|
|
||||||
if (fixer != null) {
|
|
||||||
entityTag = fixer.fixUp(DataFixer.FixTypes.ENTITY, entityTag, dataVersion);
|
|
||||||
}
|
|
||||||
|
|
||||||
EntityType entityType = EntityTypes.get(id);
|
|
||||||
if (entityType != null) {
|
|
||||||
Location location = NBTConversions.toLocation(clipboard,
|
|
||||||
requireTag(tags, "Pos", ListTag.class),
|
|
||||||
requireTag(tags, "Rotation", ListTag.class));
|
|
||||||
BaseEntity state = new BaseEntity(entityType, entityTag);
|
|
||||||
clipboard.createEntity(location, state);
|
|
||||||
} else {
|
|
||||||
log.warn("Unknown entity when pasting schematic: " + id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
inputStream.close();
|
inputStream.close();
|
||||||
|
@ -25,8 +25,10 @@ import com.boydti.fawe.util.IOUtil;
|
|||||||
import static com.google.common.base.Preconditions.checkNotNull;
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import com.sk89q.jnbt.*;
|
import com.sk89q.jnbt.*;
|
||||||
|
import com.sk89q.worldedit.WorldEdit;
|
||||||
import com.sk89q.worldedit.entity.BaseEntity;
|
import com.sk89q.worldedit.entity.BaseEntity;
|
||||||
import com.sk89q.worldedit.entity.Entity;
|
import com.sk89q.worldedit.entity.Entity;
|
||||||
|
import com.sk89q.worldedit.extension.platform.Capability;
|
||||||
import com.sk89q.worldedit.extent.clipboard.BlockArrayClipboard;
|
import com.sk89q.worldedit.extent.clipboard.BlockArrayClipboard;
|
||||||
import com.sk89q.worldedit.extent.clipboard.Clipboard;
|
import com.sk89q.worldedit.extent.clipboard.Clipboard;
|
||||||
import com.sk89q.worldedit.math.BlockVector2;
|
import com.sk89q.worldedit.math.BlockVector2;
|
||||||
@ -107,7 +109,8 @@ public class SpongeSchematicWriter implements ClipboardWriter {
|
|||||||
|
|
||||||
final DataOutput rawStream = outputStream.getOutputStream();
|
final DataOutput rawStream = outputStream.getOutputStream();
|
||||||
outputStream.writeLazyCompoundTag("Schematic", out -> {
|
outputStream.writeLazyCompoundTag("Schematic", out -> {
|
||||||
out.writeNamedTag("Version", 1);
|
out.writeNamedTag("DataVersion", WorldEdit.getInstance().getPlatformManager().queryCapability(Capability.WORLD_EDITING).getDataVersion());
|
||||||
|
out.writeNamedTag("Version", CURRENT_VERSION);
|
||||||
out.writeNamedTag("Width", (short) width);
|
out.writeNamedTag("Width", (short) width);
|
||||||
out.writeNamedTag("Height", (short) height);
|
out.writeNamedTag("Height", (short) height);
|
||||||
out.writeNamedTag("Length", (short) length);
|
out.writeNamedTag("Length", (short) length);
|
||||||
@ -224,6 +227,11 @@ public class SpongeSchematicWriter implements ClipboardWriter {
|
|||||||
out.writeNamedEmptyList("TileEntities");
|
out.writeNamedEmptyList("TileEntities");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (clipboard.hasBiomes()) {
|
||||||
|
writeBiomes(clipboard, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
TODO optimize
|
||||||
List<Tag> entities = new ArrayList<>();
|
List<Tag> entities = new ArrayList<>();
|
||||||
for (Entity entity : clipboard.getEntities()) {
|
for (Entity entity : clipboard.getEntities()) {
|
||||||
BaseEntity state = entity.getState();
|
BaseEntity state = entity.getState();
|
||||||
@ -238,7 +246,8 @@ public class SpongeSchematicWriter implements ClipboardWriter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Store our location data, overwriting any
|
// Store our location data, overwriting any
|
||||||
values.put("id", new StringTag(state.getType().getId()));
|
values.remove("id");
|
||||||
|
values.put("Id", new StringTag(state.getType().getId()));
|
||||||
values.put("Pos", writeVector(entity.getLocation()));
|
values.put("Pos", writeVector(entity.getLocation()));
|
||||||
values.put("Rotation", writeRotation(entity.getLocation()));
|
values.put("Rotation", writeRotation(entity.getLocation()));
|
||||||
|
|
||||||
@ -251,20 +260,11 @@ public class SpongeSchematicWriter implements ClipboardWriter {
|
|||||||
} else {
|
} else {
|
||||||
out.writeNamedTag("Entities", new ListTag(CompoundTag.class, entities));
|
out.writeNamedTag("Entities", new ListTag(CompoundTag.class, entities));
|
||||||
}
|
}
|
||||||
|
|
||||||
// version 2 stuff
|
|
||||||
if (clipboard.hasBiomes()) {
|
|
||||||
writeBiomes(clipboard, out);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!clipboard.getEntities().isEmpty()) {
|
|
||||||
writeEntities(clipboard, out);
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void writeBiomes(Clipboard clipboard, NBTOutputStream schematic) throws IOException {
|
private void writeBiomes(Clipboard clipboard, NBTOutputStream schematic) throws IOException {
|
||||||
|
TODO optimize
|
||||||
BlockVector3 min = clipboard.getMinimumPoint();
|
BlockVector3 min = clipboard.getMinimumPoint();
|
||||||
int width = clipboard.getRegion().getWidth();
|
int width = clipboard.getRegion().getWidth();
|
||||||
int length = clipboard.getRegion().getLength();
|
int length = clipboard.getRegion().getLength();
|
||||||
|
@ -94,11 +94,19 @@ public class BlockTransformExtent extends ResettableExtent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private static long[] adapt(Direction... dirs) {
|
private static long[] adapt(Direction... dirs) {
|
||||||
return Arrays.stream(dirs).mapToLong(dir -> 1L << dir.ordinal()).toArray();
|
long[] arr = new long[dirs.length];
|
||||||
|
for (int i = 0; i < arr.length; i++) {
|
||||||
|
arr[i] = 1L << dirs[i].ordinal();
|
||||||
|
}
|
||||||
|
return arr;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static long[] adapt(Long... dirs) {
|
private static long[] adapt(Long... dirs) {
|
||||||
return Arrays.stream(dirs).mapToLong(dir -> dir).toArray();
|
long[] arr = new long[dirs.length];
|
||||||
|
for (int i = 0; i < arr.length; i++) {
|
||||||
|
arr[i] = dirs[i];
|
||||||
|
}
|
||||||
|
return arr;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static long[] getDirections(AbstractProperty property) {
|
private static long[] getDirections(AbstractProperty property) {
|
||||||
@ -506,37 +514,4 @@ public class BlockTransformExtent extends ResettableExtent {
|
|||||||
public boolean setBlock(int x, int y, int z, BlockStateHolder block) throws WorldEditException {
|
public boolean setBlock(int x, int y, int z, BlockStateHolder block) throws WorldEditException {
|
||||||
return super.setBlock(x, y, z, transformInverse(block));
|
return super.setBlock(x, y, z, transformInverse(block));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the new value with the transformed direction.
|
|
||||||
*
|
|
||||||
* @param allowedStates the allowed states
|
|
||||||
* @param transform the transform
|
|
||||||
* @param oldDirection the old direction to transform
|
|
||||||
* @return a new state or null if none could be found
|
|
||||||
*/
|
|
||||||
@Nullable
|
|
||||||
private static Vector3 getNewStateValue(List<Direction> allowedStates, Transform transform, Vector3 oldDirection) {
|
|
||||||
Vector3 newDirection = transform.apply(oldDirection).subtract(transform.apply(Vector3.ZERO)).normalize();
|
|
||||||
Vector3 newValue = null;
|
|
||||||
double closest = -2;
|
|
||||||
boolean found = false;
|
|
||||||
|
|
||||||
for (Direction v : allowedStates) {
|
|
||||||
double dot = v.toVector().normalize().dot(newDirection);
|
|
||||||
if (dot >= closest) {
|
|
||||||
closest = dot;
|
|
||||||
newValue = v.toVector();
|
|
||||||
found = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (found) {
|
|
||||||
return newValue;
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -34,7 +34,7 @@ import java.util.List;
|
|||||||
*/
|
*/
|
||||||
public class CombinedRegionFunction implements RegionFunction {
|
public class CombinedRegionFunction implements RegionFunction {
|
||||||
|
|
||||||
private final List<RegionFunction> functions = new ArrayList<>();
|
private RegionFunction[] functions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a combined region function.
|
* Create a combined region function.
|
||||||
@ -49,7 +49,7 @@ public class CombinedRegionFunction implements RegionFunction {
|
|||||||
*/
|
*/
|
||||||
public CombinedRegionFunction(Collection<RegionFunction> functions) {
|
public CombinedRegionFunction(Collection<RegionFunction> functions) {
|
||||||
checkNotNull(functions);
|
checkNotNull(functions);
|
||||||
this.functions.addAll(functions);
|
this.functions = functions.toArray(new RegionFunction[functions.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -58,7 +58,7 @@ public class CombinedRegionFunction implements RegionFunction {
|
|||||||
* @param function an array of functions to match
|
* @param function an array of functions to match
|
||||||
*/
|
*/
|
||||||
public CombinedRegionFunction(RegionFunction... function) {
|
public CombinedRegionFunction(RegionFunction... function) {
|
||||||
this(Arrays.asList(checkNotNull(function)));
|
this.functions = function;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static CombinedRegionFunction combine(RegionFunction function, RegionFunction add) {
|
public static CombinedRegionFunction combine(RegionFunction function, RegionFunction add) {
|
||||||
@ -82,7 +82,9 @@ public class CombinedRegionFunction implements RegionFunction {
|
|||||||
*/
|
*/
|
||||||
public void add(Collection<RegionFunction> functions) {
|
public void add(Collection<RegionFunction> functions) {
|
||||||
checkNotNull(functions);
|
checkNotNull(functions);
|
||||||
this.functions.addAll(functions);
|
ArrayList<RegionFunction> functionsList = new ArrayList<>(Arrays.asList(this.functions));
|
||||||
|
functionsList.addAll(functions);
|
||||||
|
this.functions = functionsList.toArray(new RegionFunction[functionsList.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -98,9 +100,7 @@ public class CombinedRegionFunction implements RegionFunction {
|
|||||||
public boolean apply(BlockVector3 position) throws WorldEditException {
|
public boolean apply(BlockVector3 position) throws WorldEditException {
|
||||||
boolean ret = false;
|
boolean ret = false;
|
||||||
for (RegionFunction function : functions) {
|
for (RegionFunction function : functions) {
|
||||||
if (function.apply(position)) {
|
ret |= (function.apply(position));
|
||||||
ret = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -80,14 +80,7 @@ public class Naturalizer implements LayerFunction {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private boolean naturalize(BlockVector3 position, int depth) throws WorldEditException {
|
private boolean naturalize(BlockVector3 position, int depth) throws WorldEditException {
|
||||||
BlockState block = editSession.getBlock(position);
|
return editSession.setBlock(position, getTargetBlock(depth));
|
||||||
BlockState targetBlock = getTargetBlock(depth);
|
|
||||||
|
|
||||||
if (block.equalsFuzzy(targetBlock)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return editSession.setBlock(position, targetBlock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -110,7 +110,7 @@ public class BlockTypeMask extends AbstractExtentMask {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean test(BlockVector3 vector) {
|
public boolean test(BlockVector3 vector) {
|
||||||
return types[getExtent().getBlock(vector).getInternalId()];
|
return types[getExtent().getBlock(vector).getBlockType().getInternalId()];
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@ -15,7 +15,7 @@ public class SingleBlockTypeMask extends AbstractExtentMask {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean test(BlockVector3 vector) {
|
public boolean test(BlockVector3 vector) {
|
||||||
return getExtent().getBlock(vector).getInternalId() == internalId;
|
return getExtent().getBlock(vector).getBlockType().getInternalId() == internalId;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -3,12 +3,15 @@ package com.sk89q.worldedit.function.operation;
|
|||||||
import com.sk89q.worldedit.WorldEditException;
|
import com.sk89q.worldedit.WorldEditException;
|
||||||
import com.sk89q.worldedit.function.RegionFunction;
|
import com.sk89q.worldedit.function.RegionFunction;
|
||||||
import com.sk89q.worldedit.math.BlockVector3;
|
import com.sk89q.worldedit.math.BlockVector3;
|
||||||
|
import com.sk89q.worldedit.math.MutableBlockVector3;
|
||||||
|
import com.sk89q.worldedit.math.MutableVector3;
|
||||||
import com.sk89q.worldedit.math.Vector3;
|
import com.sk89q.worldedit.math.Vector3;
|
||||||
import com.sk89q.worldedit.math.transform.Transform;
|
import com.sk89q.worldedit.math.transform.Transform;
|
||||||
import com.sk89q.worldedit.regions.CuboidRegion;
|
import com.sk89q.worldedit.regions.CuboidRegion;
|
||||||
import com.sk89q.worldedit.regions.Region;
|
import com.sk89q.worldedit.regions.Region;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Vector;
|
||||||
|
|
||||||
public class BackwardsExtentBlockCopy implements Operation {
|
public class BackwardsExtentBlockCopy implements Operation {
|
||||||
private final Region region;
|
private final Region region;
|
||||||
@ -16,7 +19,8 @@ public class BackwardsExtentBlockCopy implements Operation {
|
|||||||
private final RegionFunction function;
|
private final RegionFunction function;
|
||||||
private final BlockVector3 origin;
|
private final BlockVector3 origin;
|
||||||
|
|
||||||
// private Vector mutable = new MutableBlockVector3();
|
private MutableBlockVector3 mutBV3 = new MutableBlockVector3();
|
||||||
|
private MutableVector3 mutV3 = new MutableVector3();
|
||||||
|
|
||||||
BackwardsExtentBlockCopy(Region region, BlockVector3 origin, Transform transform, RegionFunction function) {
|
BackwardsExtentBlockCopy(Region region, BlockVector3 origin, Transform transform, RegionFunction function) {
|
||||||
this.region = region;
|
this.region = region;
|
||||||
@ -56,7 +60,14 @@ public class BackwardsExtentBlockCopy implements Operation {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private BlockVector3 transform(Transform transform, BlockVector3 pt) {
|
private BlockVector3 transform(Transform transform, BlockVector3 pt) {
|
||||||
return transform.apply(Vector3.at(pt.getBlockX() - origin.getBlockX(), pt.getBlockY() - origin.getBlockY(), pt.getBlockZ() - origin.getBlockZ())).toBlockPoint().add(origin.getBlockX(), origin.getBlockY(), origin.getBlockZ());
|
mutV3.mutX(((pt.getBlockX() - origin.getBlockX())));
|
||||||
|
mutV3.mutY(((pt.getBlockY() - origin.getBlockY())));
|
||||||
|
mutV3.mutZ(((pt.getBlockZ() - origin.getBlockZ())));
|
||||||
|
Vector3 tmp = transform.apply(mutV3);
|
||||||
|
mutBV3.mutX((tmp.getBlockX() + origin.getBlockX()));
|
||||||
|
mutBV3.mutY((tmp.getBlockY() + origin.getBlockY()));
|
||||||
|
mutBV3.mutZ((tmp.getBlockZ() + origin.getBlockZ()));
|
||||||
|
return mutBV3;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -75,7 +75,7 @@ public class ForwardExtentCopy implements Operation {
|
|||||||
private int repetitions = 1;
|
private int repetitions = 1;
|
||||||
private Mask sourceMask = Masks.alwaysTrue();
|
private Mask sourceMask = Masks.alwaysTrue();
|
||||||
private boolean removingEntities;
|
private boolean removingEntities;
|
||||||
private boolean copyingEntities = true; // default to true for backwards compatibility, sort of
|
private boolean copyingEntities = true; // default to true for backwards compatibility, sort of // No, it's not for compatibility, it makes sense for entities to be copied and people will get annoyed if it doesn't
|
||||||
private boolean copyingBiomes;
|
private boolean copyingBiomes;
|
||||||
private RegionFunction sourceFunction = null;
|
private RegionFunction sourceFunction = null;
|
||||||
private Transform transform = new Identity();
|
private Transform transform = new Identity();
|
||||||
|
@ -27,6 +27,8 @@ import com.sk89q.worldedit.WorldEditException;
|
|||||||
*/
|
*/
|
||||||
public final class Operations {
|
public final class Operations {
|
||||||
|
|
||||||
|
private static final RunContext context = new RunContext();
|
||||||
|
|
||||||
private Operations() {
|
private Operations() {
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -38,7 +40,7 @@ public final class Operations {
|
|||||||
*/
|
*/
|
||||||
public static void complete(Operation op) throws WorldEditException {
|
public static void complete(Operation op) throws WorldEditException {
|
||||||
while (op != null) {
|
while (op != null) {
|
||||||
op = op.resume(new RunContext());
|
op = op.resume(context);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,7 +54,7 @@ public final class Operations {
|
|||||||
public static void completeLegacy(Operation op) throws MaxChangedBlocksException {
|
public static void completeLegacy(Operation op) throws MaxChangedBlocksException {
|
||||||
while (op != null) {
|
while (op != null) {
|
||||||
try {
|
try {
|
||||||
op = op.resume(new RunContext());
|
op = op.resume(context);
|
||||||
} catch (MaxChangedBlocksException e) {
|
} catch (MaxChangedBlocksException e) {
|
||||||
throw e;
|
throw e;
|
||||||
} catch (WorldEditException e) {
|
} catch (WorldEditException e) {
|
||||||
@ -71,7 +73,7 @@ public final class Operations {
|
|||||||
public static void completeBlindly(Operation op) {
|
public static void completeBlindly(Operation op) {
|
||||||
while (op != null) {
|
while (op != null) {
|
||||||
try {
|
try {
|
||||||
op = op.resume(new RunContext());
|
op = op.resume(context);
|
||||||
} catch (WorldEditException e) {
|
} catch (WorldEditException e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,9 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.function.pattern;
|
package com.sk89q.worldedit.function.pattern;
|
||||||
|
|
||||||
|
import com.sk89q.minecraft.util.commands.Link;
|
||||||
import com.sk89q.worldedit.WorldEditException;
|
import com.sk89q.worldedit.WorldEditException;
|
||||||
|
import com.sk89q.worldedit.command.UtilityCommands;
|
||||||
import com.sk89q.worldedit.extent.Extent;
|
import com.sk89q.worldedit.extent.Extent;
|
||||||
import com.sk89q.worldedit.math.BlockVector3;
|
import com.sk89q.worldedit.math.BlockVector3;
|
||||||
import com.sk89q.worldedit.world.block.BaseBlock;
|
import com.sk89q.worldedit.world.block.BaseBlock;
|
||||||
@ -28,6 +30,7 @@ import com.sk89q.worldedit.world.block.BlockStateHolder;
|
|||||||
/**
|
/**
|
||||||
* Returns a {@link BlockStateHolder} for a given position.
|
* Returns a {@link BlockStateHolder} for a given position.
|
||||||
*/
|
*/
|
||||||
|
@Link(clazz = UtilityCommands.class, value = "patterns")
|
||||||
public interface Pattern {
|
public interface Pattern {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -20,27 +20,59 @@
|
|||||||
package com.sk89q.worldedit.function.pattern;
|
package com.sk89q.worldedit.function.pattern;
|
||||||
|
|
||||||
import com.sk89q.worldedit.extent.Extent;
|
import com.sk89q.worldedit.extent.Extent;
|
||||||
|
import com.sk89q.worldedit.function.mask.BlockMaskBuilder;
|
||||||
import com.sk89q.worldedit.math.BlockVector3;
|
import com.sk89q.worldedit.math.BlockVector3;
|
||||||
import com.sk89q.worldedit.registry.state.Property;
|
import com.sk89q.worldedit.registry.state.Property;
|
||||||
|
import com.sk89q.worldedit.registry.state.PropertyKey;
|
||||||
import com.sk89q.worldedit.world.block.BaseBlock;
|
import com.sk89q.worldedit.world.block.BaseBlock;
|
||||||
|
import com.sk89q.worldedit.world.block.BlockState;
|
||||||
|
import com.sk89q.worldedit.world.block.BlockType;
|
||||||
import com.sk89q.worldedit.world.block.BlockTypes;
|
import com.sk89q.worldedit.world.block.BlockTypes;
|
||||||
|
|
||||||
|
import java.lang.ref.SoftReference;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.function.BiPredicate;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes the waterlogged state from blocks if possible. If not possible, returns air.
|
* Removes the waterlogged state from blocks if possible. If not possible, returns air.
|
||||||
*/
|
*/
|
||||||
public class WaterloggedRemover extends AbstractExtentPattern {
|
public class WaterloggedRemover extends AbstractExtentPattern {
|
||||||
|
|
||||||
|
private static SoftReference<BlockState[]> cache = new SoftReference<>(null);
|
||||||
|
|
||||||
|
private synchronized BlockState[] getRemap() {
|
||||||
|
BlockState[] remap = this.cache.get();
|
||||||
|
if (remap != null) return remap;
|
||||||
|
this.cache = new SoftReference<>(remap = new BlockState[BlockTypes.states.length]);
|
||||||
|
|
||||||
|
// init
|
||||||
|
for (int i = 0; i < remap.length; i++) {
|
||||||
|
BlockState state = remap[i];
|
||||||
|
BlockType type = state.getBlockType();
|
||||||
|
if (!type.hasProperty(PropertyKey.WATERLOGGED)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (state.getState(PropertyKey.WATERLOGGED) == Boolean.TRUE) {
|
||||||
|
remap[i] = state.with(PropertyKey.WATERLOGGED, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return remap;
|
||||||
|
}
|
||||||
|
|
||||||
|
private final BlockState[] remap;
|
||||||
|
|
||||||
public WaterloggedRemover(Extent extent) {
|
public WaterloggedRemover(Extent extent) {
|
||||||
super(extent);
|
super(extent);
|
||||||
|
this.remap = getRemap();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public BaseBlock apply(BlockVector3 position) {
|
public BaseBlock apply(BlockVector3 position) {
|
||||||
BaseBlock block = getExtent().getFullBlock(position);
|
BaseBlock block = getExtent().getFullBlock(position);
|
||||||
@SuppressWarnings("unchecked")
|
BlockState newState = remap[block.getOrdinal()];
|
||||||
Property<Object> prop = (Property<Object>) block.getBlockType().getPropertyMap().getOrDefault("waterlogged", null);
|
if (newState != null) {
|
||||||
if (prop != null) {
|
return newState.toBaseBlock(block.getNbtData());
|
||||||
return block.with(prop, false);
|
|
||||||
}
|
}
|
||||||
return BlockTypes.AIR.getDefaultState().toBaseBlock();
|
return BlockTypes.AIR.getDefaultState().toBaseBlock();
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,9 @@ import java.util.List;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility class to apply region functions to {@link com.sk89q.worldedit.regions.Region}.
|
* Utility class to apply region functions to {@link com.sk89q.worldedit.regions.Region}.
|
||||||
|
* @deprecated let the queue iterate, not the region function which lacks any kind of optimizations / parallelism
|
||||||
*/
|
*/
|
||||||
|
@Deprecated
|
||||||
public class RegionVisitor implements Operation {
|
public class RegionVisitor implements Operation {
|
||||||
|
|
||||||
public final Region region;
|
public final Region region;
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.math;
|
package com.sk89q.worldedit.math;
|
||||||
|
|
||||||
|
import com.google.common.collect.ComparisonChain;
|
||||||
import com.sk89q.worldedit.math.transform.AffineTransform;
|
import com.sk89q.worldedit.math.transform.AffineTransform;
|
||||||
|
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
@ -47,26 +48,18 @@ public class BlockVector2 {
|
|||||||
* cdef
|
* cdef
|
||||||
* </pre>
|
* </pre>
|
||||||
*/
|
*/
|
||||||
public static final Comparator<BlockVector2> COMPARING_GRID_ARRANGEMENT =
|
public static final Comparator<BlockVector2> COMPARING_GRID_ARRANGEMENT = (a, b) -> {
|
||||||
Comparator.comparingInt(BlockVector2::getZ).thenComparingInt(BlockVector2::getX);
|
return ComparisonChain.start()
|
||||||
|
.compare(a.getBlockZ(), b.getBlockZ())
|
||||||
|
.compare(a.getBlockX(), b.getBlockX())
|
||||||
|
.result();
|
||||||
|
};
|
||||||
|
|
||||||
public static BlockVector2 at(double x, double z) {
|
public static BlockVector2 at(double x, double z) {
|
||||||
return at((int) Math.floor(x), (int) Math.floor(z));
|
return at((int) Math.floor(x), (int) Math.floor(z));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static BlockVector2 at(int x, int z) {
|
public static BlockVector2 at(int x, int z) {
|
||||||
switch (x) {
|
|
||||||
case 0:
|
|
||||||
if (z == 0) {
|
|
||||||
return ZERO;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
if (z == 1) {
|
|
||||||
return ONE;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return new BlockVector2(x, z);
|
return new BlockVector2(x, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -349,27 +342,6 @@ public class BlockVector2 {
|
|||||||
return divide(n, n);
|
return divide(n, n);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Shift all components right.
|
|
||||||
*
|
|
||||||
* @param x the value to shift x by
|
|
||||||
* @param z the value to shift z by
|
|
||||||
* @return a new vector
|
|
||||||
*/
|
|
||||||
public BlockVector2 shr(int x, int z) {
|
|
||||||
return at(this.x >> x, this.z >> z);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Shift all components right by {@code n}.
|
|
||||||
*
|
|
||||||
* @param n the value to shift by
|
|
||||||
* @return a new vector
|
|
||||||
*/
|
|
||||||
public BlockVector2 shr(int n) {
|
|
||||||
return shr(n, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the length of the vector.
|
* Get the length of the vector.
|
||||||
*
|
*
|
||||||
@ -599,4 +571,5 @@ public class BlockVector2 {
|
|||||||
public String toString() {
|
public String toString() {
|
||||||
return "(" + x + ", " + z + ")";
|
return "(" + x + ", " + z + ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -44,20 +44,6 @@ public class BlockVector3 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static BlockVector3 at(int x, int y, int z) {
|
public static BlockVector3 at(int x, int y, int z) {
|
||||||
// switch for efficiency on typical cases
|
|
||||||
// in MC y is rarely 0/1 on selections
|
|
||||||
switch (y) {
|
|
||||||
case 0:
|
|
||||||
if (x == 0 && z == 0) {
|
|
||||||
return ZERO;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
if (x == 1 && z == 1) {
|
|
||||||
return ONE;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return new BlockVector3(x, y, z);
|
return new BlockVector3(x, y, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,19 +33,6 @@ public class Vector2 {
|
|||||||
public static final Vector2 ONE = new Vector2(1, 1);
|
public static final Vector2 ONE = new Vector2(1, 1);
|
||||||
|
|
||||||
public static Vector2 at(double x, double z) {
|
public static Vector2 at(double x, double z) {
|
||||||
int xTrunc = (int) x;
|
|
||||||
switch (xTrunc) {
|
|
||||||
case 0:
|
|
||||||
if (x == 0 && z == 0) {
|
|
||||||
return ZERO;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
if (x == 1 && z == 1) {
|
|
||||||
return ONE;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return new Vector2(x, z);
|
return new Vector2(x, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,21 +39,6 @@ public class Vector3 {
|
|||||||
public static final Vector3 ONE = new Vector3(1, 1, 1);
|
public static final Vector3 ONE = new Vector3(1, 1, 1);
|
||||||
|
|
||||||
public static Vector3 at(double x, double y, double z) {
|
public static Vector3 at(double x, double y, double z) {
|
||||||
// switch for efficiency on typical cases
|
|
||||||
// in MC y is rarely 0/1 on selections
|
|
||||||
int yTrunc = (int) y;
|
|
||||||
switch (yTrunc) {
|
|
||||||
case 0:
|
|
||||||
if (x == 0 && y == 0 && z == 0) {
|
|
||||||
return ZERO;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
if (x == 1 && y == 1 && z == 1) {
|
|
||||||
return ONE;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return new Vector3(x, y, z);
|
return new Vector3(x, y, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -651,9 +636,9 @@ public class Vector3 {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
String x = "" + getX();
|
String x = (getX() == getBlockX() ? "" + getBlockX() : "" + getX());
|
||||||
String y = "" + getY();
|
String y = (getY() == getBlockY() ? "" + getBlockY() : "" + getY());
|
||||||
String z = "" + getZ();
|
String z = (getZ() == getBlockZ() ? "" + getBlockZ() : "" + getZ());
|
||||||
return "(" + x + ", " + y + ", " + z + ")";
|
return "(" + x + ", " + y + ", " + z + ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -65,7 +65,7 @@ public class BiomeType implements RegistryItem, Keyed {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return this.id.hashCode();
|
return this.internalId; // stop changing this
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -280,11 +280,7 @@ public class BaseBlock implements BlockStateHolder<BaseBlock>, TileEntityBlock {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
int ret = toImmutableState().hashCode() << 3;
|
return blockState.hashCode(); // stop changing this
|
||||||
if (hasNbtData()) {
|
|
||||||
ret += getNbtData().hashCode();
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -280,17 +280,16 @@ public class BlockState implements BlockStateHolder<BlockState>, FawePattern {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equalsFuzzy(BlockStateHolder<?> o) {
|
public boolean equalsFuzzy(BlockStateHolder<?> o) {
|
||||||
if (null == o) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (this == o) {
|
if (this == o) {
|
||||||
// Added a reference equality check for speediness
|
// Added a reference equality check for speediness
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (o.getClass() == BlockState.class) {
|
if (o.getClass() == BlockState.class) {
|
||||||
return o.getOrdinal() == this.getOrdinal();
|
return o.getOrdinal() == this.getOrdinal();
|
||||||
}
|
}
|
||||||
|
if (null == o) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
return o.equalsFuzzy(this);
|
return o.equalsFuzzy(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -99,6 +99,18 @@ public class BlockType implements FawePattern, Keyed {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public BlockState withProperties(String properties) { //
|
||||||
|
int id = getInternalId();
|
||||||
|
for (String keyPair : properties.split(",")) {
|
||||||
|
String[] split = keyPair.split("=");
|
||||||
|
String name = split[0];
|
||||||
|
String value = split[1];
|
||||||
|
AbstractProperty btp = settings.propertiesMap.get(name);
|
||||||
|
id = btp.modify(id, btp.getValueFor(value));
|
||||||
|
}
|
||||||
|
return withStateId(id);
|
||||||
|
}
|
||||||
|
|
||||||
@Deprecated
|
@Deprecated
|
||||||
public BlockState withPropertyId(int propertyId) {
|
public BlockState withPropertyId(int propertyId) {
|
||||||
if (settings.stateOrdinals == null) return settings.defaultState;
|
if (settings.stateOrdinals == null) return settings.defaultState;
|
||||||
@ -140,11 +152,7 @@ public class BlockType implements FawePattern, Keyed {
|
|||||||
* @return The property
|
* @return The property
|
||||||
*/
|
*/
|
||||||
public <V> Property<V> getProperty(String name) {
|
public <V> Property<V> getProperty(String name) {
|
||||||
// Assume it works, CCE later at runtime if not.
|
return (Property<V>) this.settings.propertiesMap.get(name); // stop changing this (performance)
|
||||||
@SuppressWarnings("unchecked")
|
|
||||||
Property<V> property = (Property<V>) getPropertyMap().get(name);
|
|
||||||
checkArgument(property != null, "%s has no property named %s", this, name);
|
|
||||||
return property;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean hasProperty(PropertyKey key) {
|
public boolean hasProperty(PropertyKey key) {
|
||||||
@ -269,12 +277,12 @@ public class BlockType implements FawePattern, Keyed {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return this.id.hashCode();
|
return settings.internalId; // stop changing this to WEs bad hashcode
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
return obj instanceof BlockType && this.id.equals(((BlockType) obj).id);
|
return obj == this; // stop changing this to a shitty string comparison
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
package com.sk89q.worldedit.world.entity;
|
package com.sk89q.worldedit.world.entity;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
public class EntityTypes {
|
public class EntityTypes {
|
||||||
|
|
||||||
@ -133,71 +134,113 @@ public class EntityTypes {
|
|||||||
return EntityType.REGISTRY.get(id);
|
return EntityType.REGISTRY.get(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static EntityType parse(String id) {
|
private static String convertEntityId(String id) {
|
||||||
if (id.startsWith("minecraft:")) id = id.substring(10);
|
if (id.startsWith("minecraft:")) id = id.substring(10);
|
||||||
switch (id) {
|
switch(id) {
|
||||||
case "FallingSand": return EntityTypes.FALLING_BLOCK;
|
case "AreaEffectCloud": return "area_effect_cloud";
|
||||||
case "FireworksRocketEntity": return EntityTypes.FIREWORK_ROCKET;
|
case "ArmorStand": return "armor_stand";
|
||||||
case "LavaSlime": return EntityTypes.MAGMA_CUBE;
|
case "CaveSpider": return "cave_spider";
|
||||||
case "MinecartChest": return EntityTypes.CHEST_MINECART;
|
case "MinecartChest": return "chest_minecart";
|
||||||
case "MinecartCommandBlock": return EntityTypes.COMMAND_BLOCK_MINECART;
|
case "DragonFireball": return "dragon_fireball";
|
||||||
case "MinecartFurnace": return EntityTypes.FURNACE_MINECART;
|
case "ThrownEgg": return "egg";
|
||||||
case "MinecartHopper": return EntityTypes.HOPPER_MINECART;
|
case "EnderDragon": return "ender_dragon";
|
||||||
case "MinecartRideable": return EntityTypes.MINECART;
|
case "ThrownEnderpearl": return "ender_pearl";
|
||||||
case "MinecartSpawner": return EntityTypes.SPAWNER_MINECART;
|
case "FallingSand": return "falling_block";
|
||||||
case "MinecartTNT": return EntityTypes.TNT_MINECART;
|
case "FireworksRocketEntity": return "fireworks_rocket";
|
||||||
case "MushroomCow": return EntityTypes.MOOSHROOM;
|
case "MinecartFurnace": return "furnace_minecart";
|
||||||
case "PigZombie": return EntityTypes.ZOMBIE_PIGMAN;
|
case "MinecartHopper": return "hopper_minecart";
|
||||||
case "PrimedTnt": return EntityTypes.TNT;
|
case "EntityHorse": return "horse";
|
||||||
case "SnowMan": return EntityTypes.SNOW_GOLEM;
|
case "ItemFrame": return "item_frame";
|
||||||
case "ThrownEgg": return EntityTypes.EGG;
|
case "LeashKnot": return "leash_knot";
|
||||||
case "ThrownEnderpearl": return EntityTypes.ENDER_PEARL;
|
case "LightningBolt": return "lightning_bolt";
|
||||||
case "ThrownExpBottle": return EntityTypes.EXPERIENCE_BOTTLE;
|
case "LavaSlime": return "magma_cube";
|
||||||
case "ThrownPotion": return EntityTypes.POTION;
|
case "MinecartRideable": return "minecart";
|
||||||
case "WitherBoss": return EntityTypes.WITHER;
|
case "MushroomCow": return "mooshroom";
|
||||||
case "XPOrb": return EntityTypes.EXPERIENCE_ORB;
|
case "Ozelot": return "ocelot";
|
||||||
default:
|
case "PolarBear": return "polar_bear";
|
||||||
if (Character.isUpperCase(id.charAt(0))) {
|
case "ThrownPotion": return "potion";
|
||||||
StringBuilder result = new StringBuilder();
|
case "ShulkerBullet": return "shulker_bullet";
|
||||||
for (int i = 0; i < id.length(); i++) {
|
case "SmallFireball": return "small_fireball";
|
||||||
char c = id.charAt(i);
|
case "MinecartSpawner": return "spawner_minecart";
|
||||||
if (Character.isUpperCase(c)) {
|
case "SpectralArrow": return "spectral_arrow";
|
||||||
c = Character.toLowerCase(c);
|
case "PrimedTnt": return "tnt";
|
||||||
if (i != 0) result.append('_');
|
case "MinecartTNT": return "tnt_minecart";
|
||||||
}
|
case "VillagerGolem": return "villager_golem";
|
||||||
result.append(c);
|
case "WitherBoss": return "wither";
|
||||||
}
|
case "WitherSkull": return "wither_skull";
|
||||||
return parse(result.toString());
|
case "PigZombie": return "zombie_pigman";
|
||||||
}
|
case "XPOrb": return "experience_orb";
|
||||||
switch (id.toLowerCase()) {
|
case "ThrownExpBottle": return "experience_bottle";
|
||||||
|
case "EyeOfEnderSignal": return "eye_of_ender";
|
||||||
|
case "EnderCrystal": return "end_crystal";
|
||||||
|
case "MinecartCommandBlock": return "command_block_minecart";
|
||||||
|
case "SnowMan": return "snow_golem";
|
||||||
|
case "areaeffectcloud": return "area_effect_cloud";
|
||||||
|
case "armorstand": return "armor_stand";
|
||||||
|
case "cavespider": return "cave_spider";
|
||||||
|
case "minecartchest": return "chest_minecart";
|
||||||
|
case "dragonfireball": return "dragon_fireball";
|
||||||
|
case "thrownegg": return "egg";
|
||||||
|
case "enderdragon": return "ender_dragon";
|
||||||
|
case "thrownenderpearl": return "ender_pearl";
|
||||||
|
case "fallingsand": return "falling_block";
|
||||||
|
case "fireworksrocketentity": return "fireworks_rocket";
|
||||||
|
case "minecartfurnace": return "furnace_minecart";
|
||||||
|
case "minecarthopper": return "hopper_minecart";
|
||||||
|
case "entityhorse": return "horse";
|
||||||
|
case "itemframe": return "item_frame";
|
||||||
|
case "leashknot": return "leash_knot";
|
||||||
|
case "lightningbolt": return "lightning_bolt";
|
||||||
|
case "lavaslime": return "magma_cube";
|
||||||
|
case "minecartrideable": return "minecart";
|
||||||
|
case "mushroomcow": return "mooshroom";
|
||||||
|
case "ozelot": return "ocelot";
|
||||||
|
case "polarbear": return "polar_bear";
|
||||||
|
case "thrownpotion": return "potion";
|
||||||
|
case "shulkerbullet": return "shulker_bullet";
|
||||||
|
case "smallfireball": return "small_fireball";
|
||||||
|
case "minecartspawner": return "spawner_minecart";
|
||||||
|
case "spectralarrow": return "spectral_arrow";
|
||||||
|
case "primedtnt": return "tnt";
|
||||||
|
case "minecarttnt": return "tnt_minecart";
|
||||||
|
case "villagergolem": return "villager_golem";
|
||||||
|
case "witherboss": return "wither";
|
||||||
|
case "witherskull": return "wither_skull";
|
||||||
|
case "pigzombie": return "zombie_pigman";
|
||||||
|
case "xporb":
|
||||||
case "xp_orb":
|
case "xp_orb":
|
||||||
return EntityTypes.EXPERIENCE_ORB;
|
return "experience_orb";
|
||||||
|
case "thrownexpbottle":
|
||||||
case "xp_bottle":
|
case "xp_bottle":
|
||||||
return EntityTypes.EXPERIENCE_BOTTLE;
|
return "experience_bottle";
|
||||||
|
case "eyeofendersignal":
|
||||||
case "eye_of_ender_signal":
|
case "eye_of_ender_signal":
|
||||||
return EntityTypes.EYE_OF_ENDER;
|
return "eye_of_ender";
|
||||||
|
case "endercrystal":
|
||||||
case "ender_crystal":
|
case "ender_crystal":
|
||||||
return EntityTypes.END_CRYSTAL;
|
return "end_crystal";
|
||||||
case "fireworks_rocket":
|
case "fireworks_rocket": return "firework_rocket";
|
||||||
return EntityTypes.FIREWORK_ROCKET;
|
case "minecartcommandblock":
|
||||||
case "commandblock_minecart":
|
case "commandblock_minecart":
|
||||||
return EntityTypes.COMMAND_BLOCK_MINECART;
|
return "command_block_minecart";
|
||||||
case "snowman":
|
case "snowman":
|
||||||
return EntityTypes.SNOW_GOLEM;
|
return "snow_golem";
|
||||||
case "villager_golem":
|
case "villager_golem": return "iron_golem";
|
||||||
return EntityTypes.IRON_GOLEM;
|
case "evocation_fangs": return "evoker_fangs";
|
||||||
case "evocation_fangs":
|
case "evocation_illager": return "evoker";
|
||||||
return EntityTypes.EVOKER_FANGS;
|
case "vindication_illager": return "vindicator";
|
||||||
case "evocation_illager":
|
case "illusion_illager": return "illusioner";
|
||||||
return EntityTypes.EVOKER;
|
default: {
|
||||||
case "vindication_illager":
|
if (Character.isUpperCase(id.charAt(0))) {
|
||||||
return EntityTypes.VINDICATOR;
|
return convertEntityId(id.toLowerCase(Locale.ROOT));
|
||||||
case "illusion_illager":
|
}
|
||||||
return EntityTypes.ILLUSIONER;
|
return id;
|
||||||
default:
|
|
||||||
return get(id);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static EntityType parse(String id) {
|
||||||
|
return get(convertEntityId(id));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Laden…
x
In neuem Issue referenzieren
Einen Benutzer sperren