Changed the clipboard so it doesn't store min/max anymore, just origin and offset. Added /editrotate.

This commit is contained in:
sk89q 2010-10-13 10:08:53 -07:00
parent bd9a90f01d
commit 598f4aa6cd
6 changed files with 219 additions and 129 deletions

View File

@ -25,30 +25,52 @@ import java.util.ArrayList;
import com.sk89q.worldedit.*; import com.sk89q.worldedit.*;
/** /**
* The clipboard remembers the state of a cuboid region.
* *
* @author Albert * @author sk89q
*/ */
public class CuboidClipboard { public class CuboidClipboard {
private int[][][] data; private int[][][] data;
private Vector min; private Vector offset;
private Vector max;
private Vector origin; private Vector origin;
private Vector size;
/** /**
* Constructs the region instance. The minimum and maximum points must be * Constructs the clipboard.
* the respective minimum and maximum numbers!
* *
* @param min * @param size
* @param max */
public CuboidClipboard(Vector size) {
this.size = size;
data = new int[size.getBlockX()][size.getBlockY()][size.getBlockZ()];
origin = new Vector();
offset = new Vector();
}
/**
* Constructs the clipboard.
*
* @param size
* @param origin * @param origin
*/ */
public CuboidClipboard(Vector min, Vector max, Vector origin) { public CuboidClipboard(Vector size, Vector origin) {
this.min = min; this.size = size;
this.max = max; data = new int[size.getBlockX()][size.getBlockY()][size.getBlockZ()];
this.origin = origin; this.origin = origin;
data = new int[(int)((max.getX()) - min.getX() + 1)] offset = new Vector();
[(int)(max.getY() - min.getY() + 1)] }
[(int)(max.getZ() - min.getZ() + 1)];
/**
* Constructs the clipboard.
*
* @param size
* @param origin
*/
public CuboidClipboard(Vector size, Vector origin, Vector offset) {
this.size = size;
data = new int[size.getBlockX()][size.getBlockY()][size.getBlockZ()];
this.origin = origin;
this.offset = offset;
} }
/** /**
@ -57,7 +79,7 @@ public class CuboidClipboard {
* @return width * @return width
*/ */
public int getWidth() { public int getWidth() {
return (int)(max.getX() - min.getX() + 1); return size.getBlockX();
} }
/** /**
@ -66,7 +88,7 @@ public class CuboidClipboard {
* @return length * @return length
*/ */
public int getLength() { public int getLength() {
return (int)(max.getZ() - min.getZ() + 1); return size.getBlockZ();
} }
/** /**
@ -75,7 +97,47 @@ public class CuboidClipboard {
* @return height * @return height
*/ */
public int getHeight() { public int getHeight() {
return (int)(max.getY() - min.getY() + 1); return size.getBlockY();
}
/**
* Rotate the clipboard in 2D. It can only rotate by angles divisible by 90.
*
* @param angle in degrees
*/
public void rotate2D(int angle) {
angle = angle % 360;
if (angle % 90 != 0) { // Can only rotate 90 degrees at the moment
return;
}
int width = getWidth();
int length = getLength();
int height = getHeight();
int newWidth = angle % 180 == 0 ? width : length;
int newLength = angle % 180 == 0 ? length : width;
Vector sizeRotated = size.transform2D(angle, 0, 0, 0, 0);
int shiftX = sizeRotated.getX() < 0 ? newWidth - 1 : 0;
int shiftZ = sizeRotated.getZ() < 0 ? newLength - 1: 0;
int newData[][][] = new int[newWidth][getHeight()][newLength];
for (int x = 0; x < width; x++) {
for (int z = 0; z < length; z++) {
int newX = (new Vector(x, 0, z)).transform2D(angle, 0, 0, 0, 0)
.getBlockX();
int newZ = (new Vector(x, 0, z)).transform2D(angle, 0, 0, 0, 0)
.getBlockZ();
for (int y = 0; y < height; y++) {
newData[shiftX + newX][y][shiftZ + newZ] = data[x][y][z];
}
}
}
data = newData;
size = new Vector(newWidth, getHeight(), newLength);
offset = offset.transform2D(angle, 0, 0, 0, 0)
.subtract(shiftX, 0, shiftZ);
} }
/** /**
@ -84,11 +146,11 @@ public class CuboidClipboard {
* @param editSession * @param editSession
*/ */
public void copy(EditSession editSession) { public void copy(EditSession editSession) {
for (int x = (int)min.getX(); x <= (int)max.getX(); x++) { for (int x = 0; x < size.getBlockX(); x++) {
for (int y = (int)min.getY(); y <= (int)max.getY(); y++) { for (int y = 0; y < size.getBlockY(); y++) {
for (int z = (int)min.getZ(); z <= (int)max.getZ(); z++) { for (int z = 0; z < size.getBlockZ(); z++) {
data[x - (int)min.getX()][y - (int)min.getY()][z - (int)min.getZ()] = data[x][y][z] =
editSession.getBlock(x, y, z); editSession.getBlock(new Vector(x, y, z).add(origin));
} }
} }
} }
@ -104,76 +166,31 @@ public class CuboidClipboard {
*/ */
public void paste(EditSession editSession, Vector newOrigin, boolean noAir) public void paste(EditSession editSession, Vector newOrigin, boolean noAir)
throws MaxChangedBlocksException { throws MaxChangedBlocksException {
int offsetX = (int)(min.getX() - origin.getX() + newOrigin.getX()); place(editSession, newOrigin.add(offset), noAir);
int offsetY = (int)(min.getY() - origin.getY() + newOrigin.getY());
int offsetZ = (int)(min.getZ() - origin.getZ() + newOrigin.getZ());
place(editSession, offsetX, offsetY, offsetZ, noAir);
} }
/** /**
* Places the blocks in a position from the minimum corner. * Places the blocks in a position from the minimum corner.
* *
* @param editSession * @param editSession
* @param offsetX * @param pos
* @param offsetY
* @param offsetZ
* @param noAir * @param noAir
* @throws MaxChangedBlocksException * @throws MaxChangedBlocksException
*/ */
public void place(EditSession editSession, int offsetX, public void place(EditSession editSession, Vector pos, boolean noAir)
int offsetY, int offsetZ, boolean noAir)
throws MaxChangedBlocksException { throws MaxChangedBlocksException {
int xs = getWidth(); for (int x = 0; x < size.getBlockX(); x++) {
int ys = getHeight(); for (int y = 0; y < size.getBlockY(); y++) {
int zs = getLength(); for (int z = 0; z < size.getBlockZ(); z++) {
if (noAir && data[x][y][z] == 0) continue;
for (int x = 0; x < xs; x++) { editSession.setBlock(new Vector(x, y, z).add(pos),
for (int y = 0; y < ys; y++) { data[x][y][z]);
for (int z = 0; z < zs; z++) {
if (noAir && data[x][y][z] == 0) { continue; }
editSession.setBlock(x + offsetX, y + offsetY, z + offsetZ,
data[x][y][z]);
} }
} }
} }
} }
/**
* Stack the clipboard in a certain direction a certain number of
* times.
*
* @param editSession
* @param xm
* @param ym
* @param zm
* @param count
* @param noAir
* @param moveOrigin move the origin
* @throws MaxChangedBlocksException
*/
public void stack(EditSession editSession, int xm, int ym, int zm, short count,
boolean noAir, boolean moveOrigin) throws MaxChangedBlocksException {
int xs = getWidth();
int ys = getHeight();
int zs = getLength();
int offsetX = (int)min.getX();
int offsetY = (int)min.getY();
int offsetZ = (int)min.getZ();
for (short i = 1; i <= count; i++) {
place(editSession, offsetX + xm * xs, offsetY + ym * ys,
offsetZ + zm * zs, noAir);
}
if (moveOrigin) {
min = new Vector((int)offsetX + xm * count,
(int)offsetY + ym * count,
(int)offsetZ + zm * count);
}
}
/** /**
* Saves the clipboard data to a .schematic-format file. * Saves the clipboard data to a .schematic-format file.
* *
@ -182,32 +199,32 @@ public class CuboidClipboard {
* @throws SchematicException * @throws SchematicException
*/ */
public void saveSchematic(String path) throws IOException, SchematicException { public void saveSchematic(String path) throws IOException, SchematicException {
int xs = getWidth(); int width = getWidth();
int ys = getHeight(); int height = getHeight();
int zs = getLength(); int length = getLength();
if (xs > 65535) { if (width > 65535) {
throw new SchematicException("Width of region too large for a .schematic"); throw new SchematicException("Width of region too large for a .schematic");
} }
if (ys > 65535) { if (height > 65535) {
throw new SchematicException("Height of region too large for a .schematic"); throw new SchematicException("Height of region too large for a .schematic");
} }
if (zs > 65535) { if (length > 65535) {
throw new SchematicException("Length of region too large for a .schematic"); throw new SchematicException("Length of region too large for a .schematic");
} }
HashMap<String,Tag> schematic = new HashMap<String,Tag>(); HashMap<String,Tag> schematic = new HashMap<String,Tag>();
schematic.put("Width", new ShortTag("Width", (short)xs)); schematic.put("Width", new ShortTag("Width", (short)width));
schematic.put("Length", new ShortTag("Length", (short)zs)); schematic.put("Length", new ShortTag("Length", (short)length));
schematic.put("Height", new ShortTag("Height", (short)ys)); schematic.put("Height", new ShortTag("Height", (short)height));
schematic.put("Materials", new StringTag("Materials", "Alpha")); schematic.put("Materials", new StringTag("Materials", "Alpha"));
// Copy blocks // Copy blocks
byte[] blocks = new byte[xs * ys * zs]; byte[] blocks = new byte[width * height * length];
for (int x = 0; x < xs; x++) { for (int x = 0; x < width; x++) {
for (int y = 0; y < ys; y++) { for (int y = 0; y < height; y++) {
for (int z = 0; z < zs; z++) { for (int z = 0; z < length; z++) {
int index = y * xs * zs + z * xs + x; int index = y * width * length + z * width + x;
blocks[index] = (byte)data[x][y][z]; blocks[index] = (byte)data[x][y][z];
} }
} }
@ -215,7 +232,7 @@ public class CuboidClipboard {
schematic.put("Blocks", new ByteArrayTag("Blocks", blocks)); schematic.put("Blocks", new ByteArrayTag("Blocks", blocks));
// Current data is not supported // Current data is not supported
byte[] data = new byte[xs * ys * zs]; byte[] data = new byte[width * height * length];
schematic.put("Data", new ByteArrayTag("Data", data)); schematic.put("Data", new ByteArrayTag("Data", data));
// These are not stored either // These are not stored either
@ -238,7 +255,7 @@ public class CuboidClipboard {
* @throws SchematicException * @throws SchematicException
* @throws IOException * @throws IOException
*/ */
public static CuboidClipboard loadSchematic(String path, Vector origin) public static CuboidClipboard loadSchematic(String path)
throws SchematicException, IOException { throws SchematicException, IOException {
FileInputStream stream = new FileInputStream(path); FileInputStream stream = new FileInputStream(path);
NBTInputStream nbtStream = new NBTInputStream(stream); NBTInputStream nbtStream = new NBTInputStream(stream);
@ -250,27 +267,23 @@ public class CuboidClipboard {
if (!schematic.containsKey("Blocks")) { if (!schematic.containsKey("Blocks")) {
throw new SchematicException("Schematic file is missing a \"Blocks\" tag"); throw new SchematicException("Schematic file is missing a \"Blocks\" tag");
} }
short xs = (Short)getChildTag(schematic, "Width", ShortTag.class).getValue(); short width = (Short)getChildTag(schematic, "Width", ShortTag.class).getValue();
short zs = (Short)getChildTag(schematic, "Length", ShortTag.class).getValue(); short length = (Short)getChildTag(schematic, "Length", ShortTag.class).getValue();
short ys = (Short)getChildTag(schematic, "Height", ShortTag.class).getValue(); short height = (Short)getChildTag(schematic, "Height", ShortTag.class).getValue();
String materials = (String)getChildTag(schematic, "Materials", StringTag.class).getValue(); String materials = (String)getChildTag(schematic, "Materials", StringTag.class).getValue();
if (!materials.equals("Alpha")) { if (!materials.equals("Alpha")) {
throw new SchematicException("Schematic file is not an Alpha schematic"); throw new SchematicException("Schematic file is not an Alpha schematic");
} }
byte[] blocks = (byte[])getChildTag(schematic, "Blocks", ByteArrayTag.class).getValue(); byte[] blocks = (byte[])getChildTag(schematic, "Blocks", ByteArrayTag.class).getValue();
Vector min = origin; Vector size = new Vector(width, height, length);
Vector max = new Vector(
origin.getX() + xs - 1,
origin.getY() + ys - 1,
origin.getZ() + zs - 1
);
CuboidClipboard clipboard = new CuboidClipboard(min, max, origin);
for (int x = 0; x < xs; x++) { CuboidClipboard clipboard = new CuboidClipboard(size);
for (int y = 0; y < ys; y++) {
for (int z = 0; z < zs; z++) { for (int x = 0; x < width; x++) {
int index = y * xs * zs + z * xs + x; for (int y = 0; y < height; y++) {
for (int z = 0; z < length; z++) {
int index = y * width * length + z * width + x;
clipboard.data[x][y][z] = blocks[index]; clipboard.data[x][y][z] = blocks[index];
} }
} }
@ -279,6 +292,15 @@ public class CuboidClipboard {
return clipboard; return clipboard;
} }
/**
* Get child tag of a NBT structure.
*
* @param items
* @param key
* @param expected
* @return child tag
* @throws SchematicException
*/
private static Tag getChildTag(Map<String,Tag> items, String key, Class expected) private static Tag getChildTag(Map<String,Tag> items, String key, Class expected)
throws SchematicException { throws SchematicException {
if (!items.containsKey(key)) { if (!items.containsKey(key)) {

View File

@ -94,6 +94,7 @@ public class WorldEdit {
commands.put("/editlimit", "[Num] - See documentation"); commands.put("/editlimit", "[Num] - See documentation");
commands.put("/editexpand", "<Dir> [Num] - Expands the selection"); commands.put("/editexpand", "<Dir> [Num] - Expands the selection");
commands.put("/editcontract", "<Dir> [Num] - Contracts the selection"); commands.put("/editcontract", "<Dir> [Num] - Contracts the selection");
commands.put("/editrotate", "[Angle] - Rotate the clipboard");
commands.put("/forestgen", "<Size> - Make an ugly pine tree forest"); commands.put("/forestgen", "<Size> - Make an ugly pine tree forest");
commands.put("/unstuck", "Go up to the first free spot"); commands.put("/unstuck", "Go up to the first free spot");
commands.put("/ascend", "Go up one level"); commands.put("/ascend", "Go up one level");
@ -303,15 +304,11 @@ public class WorldEdit {
// Paste // Paste
} else if (split[0].equalsIgnoreCase("/editpasteair") || } else if (split[0].equalsIgnoreCase("/editpasteair") ||
split[0].equalsIgnoreCase("/editpaste")) { split[0].equalsIgnoreCase("/editpaste")) {
if (session.getClipboard() == null) { Vector pos = player.getBlockIn();
player.printError("Nothing is in your clipboard."); session.getClipboard().paste(editSession, pos,
} else { split[0].equalsIgnoreCase("/editpaste"));
Vector pos = player.getBlockIn(); player.findFreePosition();
session.getClipboard().paste(editSession, pos, player.print("Pasted. Undo with /editundo");
split[0].equalsIgnoreCase("/editpaste"));
player.findFreePosition();
player.print("Pasted. Undo with /editundo");
}
return true; return true;
@ -363,8 +360,7 @@ public class WorldEdit {
if (!filePath.substring(0, dirPath.length()).equals(dirPath)) { if (!filePath.substring(0, dirPath.length()).equals(dirPath)) {
player.printError("Schematic could not read or it does not exist."); player.printError("Schematic could not read or it does not exist.");
} else { } else {
Vector origin = player.getBlockIn(); session.setClipboard(CuboidClipboard.loadSchematic(filePath));
session.setClipboard(CuboidClipboard.loadSchematic(filePath, origin));
logger.log(Level.INFO, player.getName() + " loaded " + filePath); logger.log(Level.INFO, player.getName() + " loaded " + filePath);
player.print(filename + " loaded."); player.print(filename + " loaded.");
} }
@ -378,11 +374,6 @@ public class WorldEdit {
// Save clipboard to .schematic // Save clipboard to .schematic
} else if (split[0].equalsIgnoreCase("/editsave")) { } else if (split[0].equalsIgnoreCase("/editsave")) {
if (session.getClipboard() == null) {
player.printError("Nothing is in your clipboard.");
return true;
}
checkArgs(split, 1, 1, split[0]); checkArgs(split, 1, 1, split[0]);
String filename = split[1].replace("\0", "") + ".schematic"; String filename = split[1].replace("\0", "") + ".schematic";
File dir = new File("schematics"); File dir = new File("schematics");
@ -488,7 +479,9 @@ public class WorldEdit {
Vector max = region.getMaximumPoint(); Vector max = region.getMaximumPoint();
Vector pos = player.getBlockIn(); Vector pos = player.getBlockIn();
CuboidClipboard clipboard = new CuboidClipboard(min, max, pos); CuboidClipboard clipboard = new CuboidClipboard(
max.subtract(min).add(new Vector(1, 1, 1)),
min, min.subtract(pos));
clipboard.copy(editSession); clipboard.copy(editSession);
session.setClipboard(clipboard); session.setClipboard(clipboard);
@ -543,7 +536,7 @@ public class WorldEdit {
return true; return true;
// Expand // Contract
} else if (split[0].equalsIgnoreCase("/editcontract")) { } else if (split[0].equalsIgnoreCase("/editcontract")) {
checkArgs(split, 1, 2, split[0]); checkArgs(split, 1, 2, split[0]);
Vector dir; Vector dir;
@ -563,6 +556,20 @@ public class WorldEdit {
int newSize = region.getSize(); int newSize = region.getSize();
player.print("Region contracted " + (oldSize - newSize) + " blocks."); player.print("Region contracted " + (oldSize - newSize) + " blocks.");
return true;
// Rotate
} else if (split[0].equalsIgnoreCase("/editrotate")) {
checkArgs(split, 1, 1, split[0]);
int angle = Integer.parseInt(split[1]);
if (angle % 90 == 0) {
CuboidClipboard clipboard = session.getClipboard();
clipboard.rotate2D(angle);
player.print("Clipboard rotated by " + angle + " degrees.");
} else {
player.printError("Angles must be divisible by 90 degrees.");
}
return true; return true;
} }

View File

@ -161,6 +161,8 @@ public class WorldEditSMListener extends PluginListener {
modPlayer.sendMessage(Colors.Rose + "Unknown direction: " + ue.getDirection()); modPlayer.sendMessage(Colors.Rose + "Unknown direction: " + ue.getDirection());
} catch (InsufficientArgumentsException e6) { } catch (InsufficientArgumentsException e6) {
modPlayer.sendMessage(Colors.Rose + e6.getMessage()); modPlayer.sendMessage(Colors.Rose + e6.getMessage());
} catch (EmptyClipboardException ec) {
modPlayer.sendMessage(Colors.Rose + "Your clipboard is empty.");
} catch (WorldEditException e7) { } catch (WorldEditException e7) {
modPlayer.sendMessage(Colors.Rose + e7.getMessage()); modPlayer.sendMessage(Colors.Rose + e7.getMessage());
} }

View File

@ -172,12 +172,16 @@ public class WorldEditSession {
} }
/** /**
* Get the region. May return null. If you change the region, you should * Get the region. If you change the region, you should
* call learnRegionChanges(). * call learnRegionChanges().
* *
* @return region * @return region
* @throws IncompleteRegionException
*/ */
public Region getRegion() { public Region getRegion() throws IncompleteRegionException {
if (region == null) {
throw new IncompleteRegionException();
}
return region; return region;
} }
@ -185,8 +189,12 @@ public class WorldEditSession {
* Gets the clipboard. * Gets the clipboard.
* *
* @return clipboard, may be null * @return clipboard, may be null
* @throws EmptyClipboardException
*/ */
public CuboidClipboard getClipboard() { public CuboidClipboard getClipboard() throws EmptyClipboardException {
if (clipboard == null) {
throw new EmptyClipboardException();
}
return clipboard; return clipboard;
} }

View File

@ -0,0 +1,28 @@
// $Id$
/*
* WorldEdit
* Copyright (C) 2010 sk89q <http://www.sk89q.com>
*
* 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
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package com.sk89q.worldedit;
/**
*
* @author Albert
*/
public class EmptyClipboardException extends WorldEditException {
}

View File

@ -356,6 +356,29 @@ public class Vector {
Math.pow(pt.z - z, 2)); Math.pow(pt.z - z, 2));
} }
/**
* 2D transformation.
*
* @param vec
* @param angle in degrees
* @param aboutX
* @param aboutY
* @param translateX
* @param translateY
* @return
*/
public Vector transform2D(double angle,
double aboutX, double aboutZ, double translateX, double translateZ) {
angle = Math.toRadians(angle);
double x = this.x;
double z = this.z;
double x2 = x * Math.cos(angle) - z * Math.sin(angle);
double z2 = x * Math.sin(angle) + z * Math.cos(angle);
return new Vector(x2 + aboutX + translateX,
y,
z2 + aboutZ + translateZ);
}
/** /**
* Get a block point from a point. * Get a block point from a point.
* *