2 Commits

Author SHA1 Message Date
0b689c90eb Add new Schematic reader 2025-01-08 12:49:17 +01:00
da4aa0aef5 Add new Schematic reader 2025-01-07 16:59:36 +01:00
2 changed files with 334 additions and 2 deletions

View File

@ -0,0 +1,333 @@
/*
* This file is a part of the SteamWar software.
*
* Copyright (C) 2025 SteamWar.de-Serverteam
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package de.steamwar.core;
import com.sk89q.worldedit.WorldEdit;
import com.sk89q.worldedit.extension.factory.BlockFactory;
import com.sk89q.worldedit.extension.input.ParserContext;
import com.sk89q.worldedit.extension.platform.Capability;
import com.sk89q.worldedit.extension.platform.Platform;
import com.sk89q.worldedit.extent.clipboard.BlockArrayClipboard;
import com.sk89q.worldedit.extent.clipboard.Clipboard;
import com.sk89q.worldedit.extent.clipboard.io.ClipboardReader;
import com.sk89q.worldedit.math.BlockVector3;
import com.sk89q.worldedit.regions.CuboidRegion;
import com.sk89q.worldedit.regions.Region;
import com.sk89q.worldedit.world.DataFixer;
import com.sk89q.worldedit.world.block.BlockState;
import org.enginehub.linbus.common.LinTagId;
import org.enginehub.linbus.tree.*;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class ChaosSchemReader implements ClipboardReader {
private final DataInputStream stream;
public ChaosSchemReader(InputStream stream) {
if (stream instanceof DataInputStream dis) {
this.stream = dis;
} else {
this.stream = new DataInputStream(stream);
}
}
@Override
public Clipboard read() throws IOException {
byte tag = stream.readByte();
if (tag != 0x0A) error("Invalid NBT");
String name = stream.readUTF();
if (name.isEmpty()) {
tag = stream.readByte();
if (tag != 0x0A) error("Invalid NBT");
stream.readUTF();
readSchemTag();
stream.readByte();
} else {
readSchemTag();
}
return constructSchematic();
}
private int version;
private int dataVersion;
private DataFixer fixer;
private int width;
private int height;
private int length;
private int[] offset = new int[3];
private Map<Integer, BlockState> palette = new HashMap<>();
private int[] blocks;
private LinCompoundTag metadata;
private Map<BlockVector3, LinCompoundTag> tileEntityMap = new HashMap<>();
private Clipboard constructSchematic() {
BlockVector3 min = BlockVector3.at(offset[0], offset[1], offset[2]);
Region region = new CuboidRegion(min, min.add(width, height, length).subtract(BlockVector3.ONE));
BlockArrayClipboard clipboard = new BlockArrayClipboard(region);
clipboard.setOrigin(min);
int index = 0;
for (int block : blocks) {
int y = index / (width * length);
int z = (index % (width * length)) / width;
int x = (index++ % (width * length)) % width;
BlockState state = palette.get(block);
BlockVector3 pt = BlockVector3.at(x, y, z);
if (tileEntityMap.containsKey(pt)) {
clipboard.setBlock(clipboard.getMinimumPoint().add(pt), state.toBaseBlock(tileEntityMap.get(pt)));
} else {
clipboard.setBlock(clipboard.getMinimumPoint().add(pt), state);
}
}
return clipboard;
}
private void readSchemTag() throws IOException {
parseCompoundTag((type, name) -> {
switch (name) {
case "Version" -> version = stream.readInt();
case "DataVersion" -> {
dataVersion = stream.readInt();
Platform platform = WorldEdit.getInstance().getPlatformManager()
.queryCapability(Capability.WORLD_EDITING);
if (dataVersion < platform.getDataVersion()) {
this.fixer = platform.getDataFixer();
}
}
case "Metadata" -> metadata = readCompound();
case "Width" -> width = stream.readUnsignedShort();
case "Height" -> height = stream.readUnsignedShort();
case "Length" -> length = stream.readUnsignedShort();
case "Offset" -> offset = readIntArray().value();
case "PaletteMax" -> stream.skipNBytes(4);
case "Palette" -> readPalette();
case "BlockData" -> readBlockData();
case "Blocks" -> readBlockContainer();
case "BlockEntities" -> readTileEntities();
default -> skipByType(type);
}
});
}
private void readBlockContainer() throws IOException {
parseCompoundTag((type, name) -> {
switch (name) {
case "Palette" -> readPalette();
case "Data" -> readBlockData();
case "BlockEntities" -> readTileEntities();
default -> skipByType(type);
}
});
}
private void readTileEntities() throws IOException {
LinListTag<LinCompoundTag> list = (LinListTag<LinCompoundTag>) readArray();
list.value().forEach(linCompoundTag -> {
List<LinIntTag> pos = linCompoundTag.getListTag("Pos", LinTagType.intTag()).value();
BlockVector3 pt = BlockVector3.at(pos.get(0).value(), pos.get(1).value(), pos.get(2).value());
if (fixer != null) {
linCompoundTag = fixer.fixUp(DataFixer.FixTypes.BLOCK_ENTITY, LinCompoundTag.of(linCompoundTag.value()), dataVersion);
}
tileEntityMap.put(pt, linCompoundTag);
});
}
private void readPalette() throws IOException {
BlockFactory bf = WorldEdit.getInstance().getBlockFactory();
ParserContext context = new ParserContext();
context.setRestricted(true);
context.setTryLegacy(false);
context.setPreferringWildcard(false);
parseCompoundTag((aByte, string) -> palette.put(stream.readInt(), bf.parseFromInput(string, context).toImmutableState()));
}
private static final int SEGMENT_BITS = 0x7F;
private static final int CONTINUE_BIT = 0x80;
private void readBlockData() throws IOException {
byte[] data = readByteArray();
int[] blocks = new int[data.length];
int x = 0;
int position = 0;
for (byte current : data) {
blocks[x] |= (current & SEGMENT_BITS) << position;
if ((current & CONTINUE_BIT) == 0) {
x++;
position = 0;
} else {
position += 7;
if (position >= 32) throw new RuntimeException("VarInt is too big");
}
}
this.blocks = blocks;
}
private void parseCompoundTag(CompoundReader tag) throws IOException {
byte type;
while ((type = stream.readByte()) != 0x00) {
String name = stream.readUTF();
tag.read(type, name);
}
}
private byte[] readByteArray() throws IOException {
return stream.readNBytes(stream.readInt());
}
private void skipByType(byte type, String str) throws IOException {
skipByType(type);
}
private void skipByType(byte type) throws IOException {
switch (type) {
case 0x00 -> stream.skipNBytes(0);
case 0x01 -> stream.skipNBytes(1);
case 0x02 -> stream.skipNBytes(2);
case 0x03, 0x05 -> stream.skipNBytes(4);
case 0x04, 0x06 -> stream.skipNBytes(8);
case 0x07 -> stream.skipNBytes(stream.readInt());
case 0x08 -> stream.readUTF();
case 0x09 -> {
byte t = stream.readByte();
int length = stream.readInt();
for (int i = 0; i < length; i++) { skipByType(t); }
}
case 0x0A -> parseCompoundTag(this::skipByType);
case 0x0B -> stream.skipNBytes(4L * stream.readInt());
case 0x0C -> stream.skipNBytes(8L * stream.readInt());
default -> error("Invalid Type");
}
}
private LinTag<?> readByType(byte type) throws IOException {
return switch (type) {
case 0x00 -> null;
case 0x01 -> LinByteTag.of(stream.readByte());
case 0x02 -> LinShortTag.of(stream.readShort());
case 0x03 -> LinIntTag.of(stream.readInt());
case 0x04 -> LinLongTag.of(stream.readLong());
case 0x05 -> LinFloatTag.of(stream.readFloat());
case 0x06 -> LinDoubleTag.of(stream.readDouble());
case 0x07 -> LinByteArrayTag.of(stream.readNBytes(stream.readInt()));
case 0x08 -> LinStringTag.of(stream.readUTF());
case 0x09 -> readArray();
case 0x0A -> readCompound();
case 0x0B -> readIntArray();
case 0x0C -> readLongArray();
default -> error("Invalid Type");
};
}
private LinIntArrayTag readIntArray() throws IOException {
int length = stream.readInt();
int[] ints = new int[length];
for (int i = 0; i < length; i++) {
ints[i] = stream.readInt();
}
return LinIntArrayTag.of(ints);
}
private LinLongArrayTag readLongArray() throws IOException {
int length = stream.readInt();
long[] longs = new long[length];
for (int i = 0; i < length; i++) {
longs[i] = stream.readInt();
}
return LinLongArrayTag.of(longs);
}
private LinListTag<?> readArray() throws IOException {
byte t = stream.readByte();
int length = stream.readInt();
List<LinTag<?>> obj = new ArrayList<>(length);
for (int i = 0; i < length; i++) {
obj.add(readByType(t));
}
return LinListTag.of(LinTagType.fromId(LinTagId.fromId(t)), obj);
}
private LinCompoundTag readCompound() throws IOException {
Map<String, LinTag<?>> entries = new HashMap<>();
parseCompoundTag((type, name) -> entries.put(name, readByType(type)));
return LinCompoundTag.of(entries);
}
private <T> T error(String message) {
throw new IllegalStateException("Could not read Schem: " + message);
}
@Override
public void close() throws IOException {
stream.close();
}
private interface CompoundReader {
void read(byte type, String name) throws IOException;
}
}

View File

@ -71,8 +71,7 @@ public class WorldEditWrapper21 implements WorldEditWrapper {
public Clipboard getClipboard(InputStream is, NodeData.SchematicFormat schemFormat) throws IOException {
return switch (schemFormat) {
case MCEDIT -> new MCEditSchematicReader(new NBTInputStream(is)).read();
case SPONGE_V2 -> new SpongeSchematicV2Reader(LinBinaryIO.read(new DataInputStream(is))).read();
case SPONGE_V3 -> new SpongeSchematicV3Reader(LinBinaryIO.read(new DataInputStream(is))).read();
case SPONGE_V2, SPONGE_V3 -> new ChaosSchemReader(is).read();
};
}