// $Id$ /* * WorldEdit * Copyright (C) 2010 sk89q and contributors * * 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 . */ package com.sk89q.worldedit; import com.sk89q.worldedit.blocks.BaseBlock; import com.sk89q.worldedit.patterns.Pattern; import com.sk89q.worldedit.regions.Region; /** * Generates solid and hollow shapes according to materials returned by the * {@link #getMaterial} method. * * @author TomyLobo */ public abstract class ArbitraryShape { private final Region extent; private int cacheOffsetX; private int cacheOffsetY; private int cacheOffsetZ; private int cacheSizeX; private int cacheSizeY; private int cacheSizeZ; public ArbitraryShape(Region extent) { this.extent = extent; Vector min = extent.getMinimumPoint(); Vector max = extent.getMaximumPoint(); cacheOffsetX = min.getBlockX() - 1; cacheOffsetY = min.getBlockY() - 1; cacheOffsetZ = min.getBlockZ() - 1; cacheSizeX = (int) (max.getX() - cacheOffsetX + 2); cacheSizeY = (int) (max.getY() - cacheOffsetY + 2); cacheSizeZ = (int) (max.getZ() - cacheOffsetZ + 2); cache = new short[cacheSizeX * cacheSizeY * cacheSizeZ]; } protected Region getExtent() { return extent; } /** * Cache entries: * 0 = unknown * -1 = outside * -2 = inside but type and data 0 * > 0 = inside, value = (type | (data << 8)), not handling data < 0 */ private final short[] cache; /** * Override this function to specify the shape to generate. * * @param x * @param y * @param z * @param defaultMaterial The material returned by the pattern for the current block. * @return material to place or null to not place anything. */ protected abstract BaseBlock getMaterial(int x, int y, int z, BaseBlock defaultMaterial); private BaseBlock getMaterialCached(int x, int y, int z, Pattern pattern) { final int index = (y - cacheOffsetY) + (z - cacheOffsetZ) * cacheSizeY + (x - cacheOffsetX) * cacheSizeY * cacheSizeZ; final short cacheEntry = cache[index]; switch (cacheEntry) { case 0: // unknown, fetch material final BaseBlock material = getMaterial(x, y, z, pattern.next(new BlockVector(x, y, z))); if (material == null) { // outside cache[index] = -1; return null; } short newCacheEntry = (short) (material.getType() | ((material.getData() + 1) << 8)); if (newCacheEntry == 0) { // type and data 0 newCacheEntry = -2; } cache[index] = newCacheEntry; return material; case -1: // outside return null; case -2: // type and data 0 return new BaseBlock(0, 0); } return new BaseBlock(cacheEntry & 255, ((cacheEntry >> 8) - 1) & 15); } private boolean isInsideCached(int x, int y, int z, Pattern pattern) { final int index = (y - cacheOffsetY) + (z - cacheOffsetZ) * cacheSizeY + (x - cacheOffsetX) * cacheSizeY * cacheSizeZ; switch (cache[index]) { case 0: // unknown block, meaning they must be outside the extent at this stage, but might still be inside the shape return getMaterialCached(x, y, z, pattern) != null; case -1: // outside return false; default: // inside return true; } } /** * Generates the shape. * * @param editSession * @param pattern The pattern to generate default materials from. * @param hollow Specifies whether to generate a hollow shape. * @return number of affected blocks. * @throws MaxChangedBlocksException */ public int generate(EditSession editSession, Pattern pattern, boolean hollow) throws MaxChangedBlocksException { int affected = 0; for (BlockVector position : getExtent()) { int x = position.getBlockX(); int y = position.getBlockY(); int z = position.getBlockZ(); if (!hollow) { final BaseBlock material = getMaterial(x, y, z, pattern.next(position)); if (material != null && editSession.setBlock(position, material)) { ++affected; } continue; } final BaseBlock material = getMaterialCached(x, y, z, pattern); if (material == null) { continue; } if (hollow) { boolean draw = false; do { if (!isInsideCached(x + 1, y, z, pattern)) { draw = true; break; } if (!isInsideCached(x - 1, y, z, pattern)) { draw = true; break; } if (!isInsideCached(x, y + 1, z, pattern)) { draw = true; break; } if (!isInsideCached(x, y - 1, z, pattern)) { draw = true; break; } if (!isInsideCached(x, y, z + 1, pattern)) { draw = true; break; } if (!isInsideCached(x, y, z - 1, pattern)) { draw = true; break; } } while (false); if (!draw) { continue; } } if (editSession.setBlock(position, material)) { ++affected; } } return affected; } }