Added support for stretched cylinders and brought their profile in line with that of the central layer of a sphere of the same x/z radius.

This commit is contained in:
TomyLobo 2011-10-28 20:16:19 +02:00
parent 318e81886c
commit 58be9d9be4
4 changed files with 96 additions and 163 deletions

View File

@ -1794,157 +1794,22 @@ public class EditSession {
} }
/** /**
* Helper method to draw the cylinder. * Makes a sphere or ellipsoid.
* *
* @param center * @param pos Center of the sphere or ellipsoid
* @param x * @param block The block pattern to use
* @param z * @param radiusX The cylinder's largest north/south extent
* @param height * @param radiusZ The cylinder's largest east/west extent
* @param block * @param height The cylinder's up/down extent. If negative, extend downward.
* @param filled If false, only a shell will be generated.
* @return number of blocks changed
* @throws MaxChangedBlocksException * @throws MaxChangedBlocksException
*/ */
private int makeHCylinderPoints(Vector center, int x, double z, int height, public int makeCylinder(Vector pos, Pattern block, double radiusX, double radiusZ, int height, boolean filled) throws MaxChangedBlocksException {
Pattern block) throws MaxChangedBlocksException {
int ceilZ = (int) Math.ceil(z);
int affected = 0; int affected = 0;
if (x == 0) { radiusX += 0.5;
for (int y = 0; y < height; ++y) { radiusZ += 0.5;
setBlock(center.add(0, y, ceilZ), block);
setBlock(center.add(0, y, -ceilZ), block);
setBlock(center.add(ceilZ, y, 0), block);
setBlock(center.add(-ceilZ, y, 0), block);
affected += 4;
}
} else if (x == z) {
for (int y = 0; y < height; ++y) {
setBlock(center.add(x, y, ceilZ), block);
setBlock(center.add(-x, y, ceilZ), block);
setBlock(center.add(x, y, -ceilZ), block);
setBlock(center.add(-x, y, -ceilZ), block);
affected += 4;
}
} else if (x < z) {
for (int y = 0; y < height; ++y) {
setBlock(center.add(x, y, ceilZ), block);
setBlock(center.add(-x, y, ceilZ), block);
setBlock(center.add(x, y, -ceilZ), block);
setBlock(center.add(-x, y, -ceilZ), block);
setBlock(center.add(ceilZ, y, x), block);
setBlock(center.add(-ceilZ, y, x), block);
setBlock(center.add(ceilZ, y, -x), block);
setBlock(center.add(-ceilZ, y, -x), block);
affected += 8;
}
}
return affected;
}
/**
* Draw a hollow cylinder.
*
* @param pos
* @param block
* @param radius
* @param height
* @return number of blocks set
* @throws MaxChangedBlocksException
*/
public int makeHollowCylinder(Vector pos, Pattern block, double radius,
int height) throws MaxChangedBlocksException {
int x = 0;
double z = radius;
double d = (5 - radius * 4) / 4;
int affected = 0;
if (height == 0) {
return 0;
} else if (height < 0) {
height = -height;
pos = pos.subtract(0, height, 0);
}
// Only do this check if height is negative --Elizabeth
if (height < 0 && pos.getBlockY() - height - 1 < 0) {
height = pos.getBlockY() + 1;
} else if (pos.getBlockY() + height - 1 > 127) {
height = 127 - pos.getBlockY() + 1;
}
affected += makeHCylinderPoints(pos, x, z, height, block);
while (x < z) {
++x;
if (d >= 0) {
d += 2 * (x - --z) + 1;
} else {
d += 2 * x + 1;
}
affected += makeHCylinderPoints(pos, x, z, height, block);
}
return affected;
}
/**
* Helper method to draw the cylinder.
*
* @param center
* @param x
* @param z
* @param height
* @param block
* @throws MaxChangedBlocksException
*/
private int makeCylinderPoints(Vector center, int x, double z, int height,
Pattern block) throws MaxChangedBlocksException {
int ceilZ = (int) Math.ceil(z);
int affected = 0;
if (x == z) {
for (int y = 0; y < height; ++y) {
for (int z2 = -ceilZ; z2 <= ceilZ; ++z2) {
setBlock(center.add(x, y, z2), block);
setBlock(center.add(-x, y, z2), block);
affected += 2;
}
}
} else if (x < z) {
for (int y = 0; y < height; ++y) {
for (int x2 = -x; x2 <= x; ++x2) {
for (int z2 = -ceilZ; z2 <= ceilZ; ++z2) {
setBlock(center.add(x2, y, z2), block);
++affected;
}
setBlock(center.add(ceilZ, y, x2), block);
setBlock(center.add(-ceilZ, y, x2), block);
affected += 2;
}
}
}
return affected;
}
/**
* Draw a filled cylinder.
*
* @param pos
* @param block
* @param radius
* @param height
* @return number of blocks set
* @throws MaxChangedBlocksException
*/
public int makeCylinder(Vector pos, Pattern block, double radius, int height)
throws MaxChangedBlocksException {
int x = 0;
double z = radius;
double d = (5 - radius * 4) / 4;
int affected = 0;
if (height == 0) { if (height == 0) {
return 0; return 0;
@ -1959,18 +1824,52 @@ public class EditSession {
height = 127 - pos.getBlockY() + 1; height = 127 - pos.getBlockY() + 1;
} }
affected += makeCylinderPoints(pos, x, z, height, block); final double invRadiusX = 1 / radiusX;
final double invRadiusZ = 1 / radiusZ;
while (x < z) { final int ceilRadiusX = (int) Math.ceil(radiusX);
++x; final int ceilRadiusZ = (int) Math.ceil(radiusZ);
if (d >= 0) { double nextXn = 0;
d += 2 * (x - --z) + 1; forX:
} else { for (int x = 0; x <= ceilRadiusX; ++x) {
d += 2 * x + 1; final double xn = nextXn;
nextXn = (x + 1) * invRadiusX;
double nextZn = 0;
forZ:
for (int z = 0; z <= ceilRadiusZ; ++z) {
final double zn = nextZn;
nextZn = (z + 1) * invRadiusZ;
double distanceSq = lengthSq(xn, zn);
if (distanceSq > 1) {
if (z == 0) {
break forX;
}
break forZ;
}
if (!filled) {
if (lengthSq(nextXn, zn) <= 1 && lengthSq(xn, nextZn) <= 1) {
continue;
}
}
for (int y = 0; y < height; ++y) {
if (setBlock(pos.add(x, y, z), block)) {
++affected;
}
if (setBlock(pos.add(-x, y, z), block)) {
++affected;
}
if (setBlock(pos.add(x, y, -z), block)) {
++affected;
}
if (setBlock(pos.add(-x, y, -z), block)) {
++affected;
}
}
} }
affected += makeCylinderPoints(pos, x, z, height, block);
} }
return affected; return affected;
@ -2071,6 +1970,10 @@ public class EditSession {
return (x * x) + (y * y) + (z * z); return (x * x) + (y * y) + (z * z);
} }
private static final double lengthSq(double x, double z) {
return (x * x) + (z * z);
}
/** /**
* Makes a pyramid. * Makes a pyramid.
* *

View File

@ -39,7 +39,7 @@ import com.sk89q.worldedit.util.TreeGenerator;
public class GenerationCommands { public class GenerationCommands {
@Command( @Command(
aliases = { "/hcyl" }, aliases = { "/hcyl" },
usage = "<block> <radius> [height]", usage = "<block> <radius>[,<radius>] [height]",
desc = "Generate a hollow cylinder", desc = "Generate a hollow cylinder",
min = 2, min = 2,
max = 3 max = 3
@ -51,17 +51,32 @@ public class GenerationCommands {
throws WorldEditException { throws WorldEditException {
Pattern block = we.getBlockPattern(player, args.getString(0)); Pattern block = we.getBlockPattern(player, args.getString(0));
double radius = Math.max(1, args.getDouble(1)); String[] radiuses = args.getString(1).split(",");
final double radiusX, radiusZ;
switch (radiuses.length) {
case 1:
radiusX = radiusZ = Math.max(1, Double.parseDouble(radiuses[0]));
break;
case 2:
radiusX = Math.max(1, Double.parseDouble(radiuses[0]));
radiusZ = Math.max(1, Double.parseDouble(radiuses[1]));
break;
default:
player.printError("You must either specify 1 or 2 radius values.");
return;
}
int height = args.argsLength() > 2 ? args.getInteger(2) : 1; int height = args.argsLength() > 2 ? args.getInteger(2) : 1;
Vector pos = session.getPlacementPosition(player); Vector pos = session.getPlacementPosition(player);
int affected = editSession.makeHollowCylinder(pos, block, radius, height); int affected = editSession.makeCylinder(pos, block, radiusX, radiusZ, height, false);
player.print(affected + " block(s) have been created."); player.print(affected + " block(s) have been created.");
} }
@Command( @Command(
aliases = { "/cyl" }, aliases = { "/cyl" },
usage = "<block> <radius> [height]", usage = "<block> <radius>[,<radius>] [height]",
desc = "Generate a cylinder", desc = "Generate a cylinder",
min = 2, min = 2,
max = 3 max = 3
@ -73,11 +88,26 @@ public class GenerationCommands {
throws WorldEditException { throws WorldEditException {
Pattern block = we.getBlockPattern(player, args.getString(0)); Pattern block = we.getBlockPattern(player, args.getString(0));
double radius = Math.max(1, args.getDouble(1)); String[] radiuses = args.getString(1).split(",");
final double radiusX, radiusZ;
switch (radiuses.length) {
case 1:
radiusX = radiusZ = Math.max(1, Double.parseDouble(radiuses[0]));
break;
case 2:
radiusX = Math.max(1, Double.parseDouble(radiuses[0]));
radiusZ = Math.max(1, Double.parseDouble(radiuses[1]));
break;
default:
player.printError("You must either specify 1 or 2 radius values.");
return;
}
int height = args.argsLength() > 2 ? args.getInteger(2) : 1; int height = args.argsLength() > 2 ? args.getInteger(2) : 1;
Vector pos = session.getPlacementPosition(player); Vector pos = session.getPlacementPosition(player);
int affected = editSession.makeCylinder(pos, block, radius, height); int affected = editSession.makeCylinder(pos, block, radiusX, radiusZ, height, true);
player.print(affected + " block(s) have been created."); player.print(affected + " block(s) have been created.");
} }

View File

@ -33,6 +33,6 @@ public class CylinderBrush implements Brush {
public void build(EditSession editSession, Vector pos, Pattern mat, double size) public void build(EditSession editSession, Vector pos, Pattern mat, double size)
throws MaxChangedBlocksException { throws MaxChangedBlocksException {
editSession.makeCylinder(pos, mat, size, height); editSession.makeCylinder(pos, mat, size, size, height, true);
} }
} }

View File

@ -33,6 +33,6 @@ public class HollowCylinderBrush implements Brush {
public void build(EditSession editSession, Vector pos, Pattern mat, double size) public void build(EditSession editSession, Vector pos, Pattern mat, double size)
throws MaxChangedBlocksException { throws MaxChangedBlocksException {
editSession.makeHollowCylinder(pos, mat, size, height); editSession.makeCylinder(pos, mat, size, size, height, false);
} }
} }