diff --git a/src/main/java/com/sk89q/worldedit/EditSession.java b/src/main/java/com/sk89q/worldedit/EditSession.java
index 5735238e0..0abc7058f 100644
--- a/src/main/java/com/sk89q/worldedit/EditSession.java
+++ b/src/main/java/com/sk89q/worldedit/EditSession.java
@@ -16,7 +16,6 @@
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
-
package com.sk89q.worldedit;
import java.util.Map;
@@ -47,39 +46,38 @@
* @author sk89q
*/
public class EditSession {
+
/**
* Random number generator.
*/
private static Random prng = new Random();
-
/**
* World.
*/
protected LocalWorld world;
-
/**
* Stores the original blocks before modification.
*/
private DoubleArrayList original =
- new DoubleArrayList(
+ new DoubleArrayList(
true);
/**
* Stores the current blocks.
*/
private DoubleArrayList current =
- new DoubleArrayList(
+ new DoubleArrayList(
false);
/**
* Blocks that should be placed before last.
*/
private DoubleArrayList queueAfter =
- new DoubleArrayList(
+ new DoubleArrayList(
false);
/**
* Blocks that should be placed last.
*/
private DoubleArrayList queueLast =
- new DoubleArrayList(
+ new DoubleArrayList(
false);
/**
@@ -158,28 +156,28 @@ public EditSession(LocalWorld world, int maxBlocks, BlockBag blockBag) {
public boolean rawSetBlock(Vector pt, BaseBlock block) {
int y = pt.getBlockY();
int type = block.getType();
-
+
if (y < 0 || y > 127) {
return false;
}
-
+
// No invalid blocks
if (!world.isValidBlockType(type)) {
return false;
}
-
+
if (mask != null) {
if (!mask.matches(this, pt)) {
return false;
}
}
-
+
int existing = world.getBlockType(pt);
// Clear the container block so that it doesn't drop items
if (BlockType.isContainerBlock(existing) && blockBag == null) {
world.clearContainerBlockContents(pt);
- // Ice turns until water so this has to be done first
+ // Ice turns until water so this has to be done first
} else if (existing == BlockID.ICE) {
world.setBlockType(pt, 0);
}
@@ -205,7 +203,7 @@ public boolean rawSetBlock(Vector pt, BaseBlock block) {
}
}
}
-
+
boolean result;
if (BlockType.usesData(id)) {
@@ -222,30 +220,30 @@ public boolean rawSetBlock(Vector pt, BaseBlock block) {
}
}
//System.out.println(pt + "" +result);
-
+
if (id != 0) {
// Signs
if (block instanceof SignBlock) {
SignBlock signBlock = (SignBlock) block;
world.copyToWorld(pt, signBlock);
- // Chests
+ // Chests
} else if (block instanceof ChestBlock && blockBag == null) {
ChestBlock chestBlock = (ChestBlock) block;
world.copyToWorld(pt, chestBlock);
- // Furnaces
+ // Furnaces
} else if (block instanceof FurnaceBlock && blockBag == null) {
FurnaceBlock furnaceBlock = (FurnaceBlock) block;
world.copyToWorld(pt, furnaceBlock);
- // Dispenser
+ // Dispenser
} else if (block instanceof DispenserBlock && blockBag == null) {
DispenserBlock dispenserBlock = (DispenserBlock) block;
world.copyToWorld(pt, dispenserBlock);
- // Mob spawners
+ // Mob spawners
} else if (block instanceof MobSpawnerBlock) {
MobSpawnerBlock mobSpawnerblock = (MobSpawnerBlock) block;
world.copyToWorld(pt, mobSpawnerblock);
- // Note blocks
+ // Note blocks
} else if (block instanceof NoteBlock) {
NoteBlock noteBlock = (NoteBlock) block;
world.copyToWorld(pt, noteBlock);
@@ -337,14 +335,14 @@ public boolean smartSetBlock(Vector pt, BaseBlock block) {
// Place torches, etc. last
if (BlockType.shouldPlaceLast(block.getType())) {
queueLast.put(pt.toBlockVector(), block);
- return !(getBlockType(pt) == block.getType()
+ return !(getBlockType(pt) == block.getType()
&& getBlockData(pt) == block.getData());
// Destroy torches, etc. first
} else if (BlockType.shouldPlaceLast(getBlockType(pt))) {
rawSetBlock(pt, new BaseBlock(0));
} else {
queueAfter.put(pt.toBlockVector(), block);
- return !(getBlockType(pt) == block.getType()
+ return !(getBlockType(pt) == block.getType()
&& getBlockData(pt) == block.getData());
}
}
@@ -393,7 +391,7 @@ public int getBlockType(Vector pt) {
return world.getBlockType(pt);
}
-
+
public int getBlockData(Vector pt) {
// In the case of the queue, the block may have not actually been
// changed yet
@@ -423,27 +421,27 @@ public BaseBlock rawGetBlock(Vector pt) {
SignBlock block = new SignBlock(type, data);
world.copyFromWorld(pt, block);
return block;
- // Chest
+ // Chest
} else if (type == BlockID.CHEST) {
ChestBlock block = new ChestBlock(data);
world.copyFromWorld(pt, block);
return block;
- // Furnace
+ // Furnace
} else if (type == BlockID.FURNACE || type == BlockID.BURNING_FURNACE) {
FurnaceBlock block = new FurnaceBlock(type, data);
world.copyFromWorld(pt, block);
return block;
- // Dispenser
+ // Dispenser
} else if (type == BlockID.DISPENSER) {
DispenserBlock block = new DispenserBlock(data);
world.copyFromWorld(pt, block);
return block;
- // Mob spawner
+ // Mob spawner
} else if (type == BlockID.MOB_SPAWNER) {
MobSpawnerBlock block = new MobSpawnerBlock(data);
world.copyFromWorld(pt, block);
return block;
- // Note block
+ // Note block
} else if (type == BlockID.NOTE_BLOCK) {
NoteBlock block = new NoteBlock(data);
world.copyFromWorld(pt, block);
@@ -536,7 +534,7 @@ public void disableQueue() {
}
queued = false;
}
-
+
/**
* Set fast mode.
*
@@ -545,7 +543,7 @@ public void disableQueue() {
public void setFastMode(boolean fastMode) {
this.fastMode = fastMode;
}
-
+
/**
* Return fast mode status.
*
@@ -1603,7 +1601,7 @@ public int fixLiquid(Vector pos, int radius, int moving, int stationary)
if (setBlock(cur, stationaryBlock)){
++affected;
}
-
+
// Check radius
if (pos.distance(cur) > radius) {
continue;
@@ -1851,6 +1849,49 @@ public int makeSphere(Vector pos, Pattern block, int radius,
return affected;
}
+ /**
+ * Makes a pyramid.
+ *
+ * @param pos
+ * @param block
+ * @param size
+ * @param filled
+ * @return number of blocks changed
+ * @throws MaxChangedBlocksException
+ */
+ public int makePyramid(Vector pos, Pattern block, int size,
+ boolean filled) throws MaxChangedBlocksException {
+ int affected = 0;
+
+ int height = size;
+
+ for (int y = 0; y <= height; ++y) {
+ size--;
+ for (int x = 0; x <= size; ++x) {
+ for (int z = 0; z <= size; ++z) {
+
+ if ((filled && z <= size && x <= size) || z == size || x == size) {
+
+ 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;
+ }
+ }
+ }
+ }
+ }
+
+ return affected;
+ }
+
/**
* Thaw.
*
@@ -1944,7 +1985,7 @@ public int simulateSnow(Vector pos, int radius)
|| id == 55 // Redstone wire
|| id == 59 // Crops
|| (id >= 63 && id <= 72) || id == 75 // Redstone
- // torch
+ // torch
|| id == 76 // Redstone torch
|| id == 77 // Stone button
|| id == 78 // Snow
@@ -2026,9 +2067,9 @@ private void makePumpkinPatch(Vector basePos)
*/
private void makePumpkinPatchVine(Vector basePos, Vector pos)
throws MaxChangedBlocksException {
- if (pos.distance(basePos) > 4)
+ if (pos.distance(basePos) > 4)
return;
- if (getBlockType(pos) != 0)
+ if (getBlockType(pos) != 0)
return;
for (int i = -1; i > -3; --i) {
@@ -2046,27 +2087,27 @@ private void makePumpkinPatchVine(Vector basePos, Vector pos)
int h = prng.nextInt(3) - 1;
if (t == 0) {
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
makePumpkinPatchVine(basePos, pos.add(1, 0, 0));
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
setBlockIfAir(pos.add(1, h, -1), new BaseBlock(18));
setBlockIfAir(pos.add(0, 0, -1), new BaseBlock(86));
} else if (t == 1) {
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
makePumpkinPatchVine(basePos, pos.add(0, 0, 1));
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
setBlockIfAir(pos.add(1, h, 0), new BaseBlock(18));
setBlockIfAir(pos.add(1, 0, 1), new BaseBlock(86));
} else if (t == 2) {
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
makePumpkinPatchVine(basePos, pos.add(0, 0, -1));
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
setBlockIfAir(pos.add(-1, h, 0), new BaseBlock(18));
setBlockIfAir(pos.add(-1, 0, 1), new BaseBlock(86));
} else if (t == 3) {
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
makePumpkinPatchVine(basePos, pos.add(-1, 0, 0));
- if (prng.nextBoolean())
+ if (prng.nextBoolean())
setBlockIfAir(pos.add(-1, h, -1), new BaseBlock(18));
setBlockIfAir(pos.add(-1, 0, -1), new BaseBlock(86));
}
@@ -2089,7 +2130,7 @@ public int makePumpkinPatches(Vector basePos, int size)
for (int z = basePos.getBlockZ() - size; z <= basePos.getBlockZ()
+ size; ++z) {
// Don't want to be in the ground
- if (!getBlock(new Vector(x, basePos.getBlockY(), z)).isAir())
+ if (!getBlock(new Vector(x, basePos.getBlockY(), z)).isAir())
continue;
// The gods don't want a pumpkin patch here
if (Math.random() < 0.98) {
@@ -2270,7 +2311,6 @@ public List> getBlockDistribution(Region region) {
* maximal height
* @return height of highest block found or 'minY'
*/
-
public int getHighestTerrainBlock(int x, int z, int minY, int maxY) {
for (int y = maxY; y >= minY; --y) {
Vector pt = new Vector(x, y, z);
@@ -2327,7 +2367,7 @@ public BlockBag getBlockBag() {
public void setBlockBag(BlockBag blockBag) {
this.blockBag = blockBag;
}
-
+
/**
* Get the world.
*
@@ -2336,7 +2376,7 @@ public void setBlockBag(BlockBag blockBag) {
public LocalWorld getWorld() {
return world;
}
-
+
/**
* Get the number of blocks changed, including repeated block changes.
*
diff --git a/src/main/java/com/sk89q/worldedit/commands/GenerationCommands.java b/src/main/java/com/sk89q/worldedit/commands/GenerationCommands.java
index e6b44e0d1..2284c8d11 100644
--- a/src/main/java/com/sk89q/worldedit/commands/GenerationCommands.java
+++ b/src/main/java/com/sk89q/worldedit/commands/GenerationCommands.java
@@ -178,4 +178,48 @@ public static void pumpkins(CommandContext args, WorldEdit we,
int affected = editSession.makePumpkinPatches(player.getPosition(), size);
player.print(affected + " pumpkin patches created.");
}
+
+ @Command(
+ aliases = {"/pyramid"},
+ usage = " ",
+ desc = "Generate a filled pyramid",
+ min = 2,
+ max = 2
+ )
+ @CommandPermissions({"worldedit.generation.pyramid"})
+ public static void pyramid(CommandContext args, WorldEdit we,
+ LocalSession session, LocalPlayer player, EditSession editSession)
+ throws WorldEditException {
+
+ Pattern block = we.getBlockPattern(player, args.getString(0));
+ int size = Math.max(1, args.getInteger(1));
+ Vector pos = session.getPlacementPosition(player);
+
+ int affected = editSession.makePyramid(pos, block, size, true);
+
+ player.findFreePosition();
+ player.print(affected + " block(s) have been created.");
+ }
+
+ @Command(
+ aliases = {"/hpyramid"},
+ usage = " ",
+ desc = "Generate a hollow pyramid",
+ min = 2,
+ max = 2
+ )
+ @CommandPermissions({"worldedit.generation.pyramid"})
+ public static void hpyramid(CommandContext args, WorldEdit we,
+ LocalSession session, LocalPlayer player, EditSession editSession)
+ throws WorldEditException {
+
+ Pattern block = we.getBlockPattern(player, args.getString(0));
+ int size = Math.max(1, args.getInteger(1));
+ Vector pos = session.getPlacementPosition(player);
+
+ int affected = editSession.makePyramid(pos, block, size, false);
+
+ player.findFreePosition();
+ player.print(affected + " block(s) have been created.");
+ }
}
diff --git a/src/main/resources/plugin.yml b/src/main/resources/plugin.yml
index af7c466ae..7b51f05d9 100644
--- a/src/main/resources/plugin.yml
+++ b/src/main/resources/plugin.yml
@@ -50,10 +50,19 @@ commands:
description: Modify block change limit
usage: /
permissions: 'worldedit.limit'
+ /gmask:
+ description: Set the global mask
+ usage: / [mask]
+ aliases: ['gmask']
+ permissions: 'worldedit.global-mask'
we:
description: WorldEdit commands
usage: /
aliases: ['worldedit']
+ /fast:
+ description: Toggle fast mode
+ usage: /
+ permissions: 'worldedit.fast'
toggleplace:
description:
usage: /
@@ -85,6 +94,10 @@ commands:
description: Generate pumpkin patches
usage: / [size]
permissions: 'worldedit.generation.pumpkins'
+ /pyramid:
+ description: Generate a filled pyramid
+ usage: / [inverted]
+ permissions: 'worldedit.generation.sphere'
/undo:
description: Undoes the last action
usage: / [times]
@@ -105,11 +118,11 @@ commands:
permissions: 'worldedit.navigation.unstuck'
ascend:
description: Go up a floor
- usage: /
+ usage: / [# of levels]
permissions: 'worldedit.navigation.ascend'
descend:
description: Go down a floor
- usage: /
+ usage: / [# of floors]
permissions: 'worldedit.navigation.descend'
ceil:
description: Go to the celing
@@ -143,6 +156,10 @@ commands:
description: Set a block on top of blocks in the region
usage: /
permissions: 'worldedit.region.overlay'
+ /naturalize:
+ description: 3 layers of dirt on top then rock below
+ usage: /
+ permissions: 'worldedit.region.naturalize'
/walls:
description: Build the four sides of the selection
usage: /
@@ -270,6 +287,10 @@ commands:
usage: / [pattern]
aliases: ['material', 'fill']
permissions: 'worldedit.brush.options.material'
+ range:
+ description: Set the brush range
+ usage: / [pattern]
+ permissions: 'worldedit.brush.options.range'
info:
description: Block information tool
usage: /
@@ -289,6 +310,11 @@ commands:
description: Block data cycler tool
usage: /
permissions: 'worldedit.tool.data-cycler'
+ floodfill:
+ description: Flood fill tool
+ usage: /
+ aliases: ['flood']
+ permissions: 'worldedit.tool.flood-fill'
brush:
description: Brush tool
usage: /