getMagicNumber(); $name = $treeType->getDisplayName(); self::register(new Planks(new BID(Block::PLANKS, $magicNumber), $name . " Planks")); self::register(new Sapling(new BID(Block::SAPLING, $magicNumber), $name . " Sapling", $treeType)); self::register(new WoodenFence(new BID(Block::FENCE, $magicNumber), $name . " Fence")); self::register(new WoodenSlab(new BlockIdentifierFlattened(Block::WOODEN_SLAB, Block::DOUBLE_WOODEN_SLAB, $treeType->getMagicNumber()), $treeType->getDisplayName())); //TODO: find a better way to deal with this split self::register(new Leaves(new BID($magicNumber >= 4 ? Block::LEAVES2 : Block::LEAVES, $magicNumber & 0x03), $name . " Leaves", $treeType)); self::register(new Log(new BID($magicNumber >= 4 ? Block::WOOD2 : Block::WOOD, $magicNumber & 0x03), $name . " Log", $treeType)); self::register(new Wood(new BID($magicNumber >= 4 ? Block::WOOD2 : Block::WOOD, ($magicNumber & 0x03) | 0b1100), $name . " Wood", $treeType)); self::register(new FenceGate(new BID($fenceGateIds[$treeType]), $treeType->getDisplayName() . " Fence Gate")); self::register(new WoodenStairs(new BID($woodenStairIds[$treeType]), $treeType->getDisplayName() . " Stairs")); self::register(new WoodenDoor($woodenDoorIds[$treeType], $treeType->getDisplayName() . " Door")); } static $sandstoneTypes = [ Sandstone::NORMAL => "", Sandstone::CHISELED => "Chiseled ", Sandstone::CUT => "Cut ", Sandstone::SMOOTH => "Smooth " ]; foreach($sandstoneTypes as $variant => $prefix){ self::register(new Sandstone(new BID(Block::SANDSTONE, $variant), $prefix . "Sandstone")); self::register(new Sandstone(new BID(Block::RED_SANDSTONE, $variant), $prefix . "Red Sandstone")); } /** @var int[]|\SplObjectStorage $glazedTerracottaIds */ $glazedTerracottaIds = new \SplObjectStorage(); $glazedTerracottaIds[DyeColor::WHITE()] = Block::WHITE_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::ORANGE()] = Block::ORANGE_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::MAGENTA()] = Block::MAGENTA_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::LIGHT_BLUE()] = Block::LIGHT_BLUE_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::YELLOW()] = Block::YELLOW_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::LIME()] = Block::LIME_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::PINK()] = Block::PINK_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::GRAY()] = Block::GRAY_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::LIGHT_GRAY()] = Block::SILVER_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::CYAN()] = Block::CYAN_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::PURPLE()] = Block::PURPLE_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::BLUE()] = Block::BLUE_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::BROWN()] = Block::BROWN_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::GREEN()] = Block::GREEN_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::RED()] = Block::RED_GLAZED_TERRACOTTA; $glazedTerracottaIds[DyeColor::BLACK()] = Block::BLACK_GLAZED_TERRACOTTA; foreach(DyeColor::getAll() as $color){ self::register(new Carpet(new BID(Block::CARPET, $color->getMagicNumber()), $color->getDisplayName() . " Carpet")); self::register(new Concrete(new BID(Block::CONCRETE, $color->getMagicNumber()), $color->getDisplayName() . " Concrete")); self::register(new ConcretePowder(new BID(Block::CONCRETE_POWDER, $color->getMagicNumber()), $color->getDisplayName() . " Concrete Powder")); self::register(new Glass(new BID(Block::STAINED_GLASS, $color->getMagicNumber()), $color->getDisplayName() . " Stained Glass")); self::register(new GlassPane(new BID(Block::STAINED_GLASS_PANE, $color->getMagicNumber()), $color->getDisplayName() . " Stained Glass Pane")); self::register(new GlazedTerracotta(new BID($glazedTerracottaIds[$color]), $color->getDisplayName() . " Glazed Terracotta")); self::register(new HardenedClay(new BID(Block::STAINED_CLAY, $color->getMagicNumber()), $color->getDisplayName() . " Stained Clay")); self::register(new HardenedGlass(new BID(Block::HARD_STAINED_GLASS, $color->getMagicNumber()), "Hardened " . $color->getDisplayName() . " Stained Glass")); self::register(new HardenedGlassPane(new BID(Block::HARD_STAINED_GLASS_PANE, $color->getMagicNumber()), "Hardened " . $color->getDisplayName() . " Stained Glass Pane")); self::register(new Wool(new BID(Block::WOOL, $color->getMagicNumber()), $color->getDisplayName() . " Wool")); } static $wallTypes = [ CobblestoneWall::ANDESITE_WALL => "Andesite", CobblestoneWall::BRICK_WALL => "Brick", CobblestoneWall::DIORITE_WALL => "Diorite", CobblestoneWall::END_STONE_BRICK_WALL => "End Stone Brick", CobblestoneWall::GRANITE_WALL => "Granite", CobblestoneWall::MOSSY_STONE_BRICK_WALL => "Mossy Stone Brick", CobblestoneWall::MOSSY_WALL => "Mossy Cobblestone", CobblestoneWall::NETHER_BRICK_WALL => "Nether Brick", CobblestoneWall::NONE_MOSSY_WALL => "Cobblestone", CobblestoneWall::PRISMARINE_WALL => "Prismarine", CobblestoneWall::RED_NETHER_BRICK_WALL => "Red Nether Brick", CobblestoneWall::RED_SANDSTONE_WALL => "Red Sandstone", CobblestoneWall::SANDSTONE_WALL => "Sandstone", CobblestoneWall::STONE_BRICK_WALL => "Stone Brick" ]; foreach($wallTypes as $magicNumber => $prefix){ self::register(new CobblestoneWall(new BID(Block::COBBLESTONE_WALL, $magicNumber), $prefix . " Wall")); } //TODO: minecraft:acacia_button //TODO: minecraft:acacia_pressure_plate //TODO: minecraft:acacia_standing_sign //TODO: minecraft:acacia_trapdoor //TODO: minecraft:acacia_wall_sign //TODO: minecraft:andesite_stairs //TODO: minecraft:bamboo //TODO: minecraft:bamboo_sapling //TODO: minecraft:barrel //TODO: minecraft:barrier //TODO: minecraft:beacon //TODO: minecraft:bell //TODO: minecraft:birch_button //TODO: minecraft:birch_pressure_plate //TODO: minecraft:birch_standing_sign //TODO: minecraft:birch_trapdoor //TODO: minecraft:birch_wall_sign //TODO: minecraft:blast_furnace //TODO: minecraft:blue_ice //TODO: minecraft:bubble_column //TODO: minecraft:cartography_table //TODO: minecraft:carved_pumpkin //TODO: minecraft:cauldron //TODO: minecraft:chain_command_block //TODO: minecraft:chemical_heat //TODO: minecraft:chemistry_table //TODO: minecraft:chorus_flower //TODO: minecraft:chorus_plant //TODO: minecraft:command_block //TODO: minecraft:conduit //TODO: minecraft:coral //TODO: minecraft:coral_block //TODO: minecraft:coral_fan //TODO: minecraft:coral_fan_dead //TODO: minecraft:coral_fan_hang //TODO: minecraft:coral_fan_hang2 //TODO: minecraft:coral_fan_hang3 //TODO: minecraft:dark_oak_button //TODO: minecraft:dark_oak_pressure_plate //TODO: minecraft:dark_oak_trapdoor //TODO: minecraft:dark_prismarine_stairs //TODO: minecraft:darkoak_standing_sign //TODO: minecraft:darkoak_wall_sign //TODO: minecraft:diorite_stairs //TODO: minecraft:dispenser //TODO: minecraft:double_stone_slab3 //TODO: minecraft:double_stone_slab4 //TODO: minecraft:dragon_egg //TODO: minecraft:dried_kelp_block //TODO: minecraft:dropper //TODO: minecraft:element_0 //TODO: minecraft:end_brick_stairs //TODO: minecraft:end_gateway //TODO: minecraft:end_portal //TODO: minecraft:fletching_table //TODO: minecraft:frosted_ice //TODO: minecraft:granite_stairs //TODO: minecraft:grindstone //TODO: minecraft:hopper //TODO: minecraft:jukebox //TODO: minecraft:jungle_button //TODO: minecraft:jungle_pressure_plate //TODO: minecraft:jungle_standing_sign //TODO: minecraft:jungle_trapdoor //TODO: minecraft:jungle_wall_sign //TODO: minecraft:kelp //TODO: minecraft:lantern //TODO: minecraft:lava_cauldron //TODO: minecraft:monster_egg //TODO: minecraft:mossy_cobblestone_stairs //TODO: minecraft:mossy_stone_brick_stairs //TODO: minecraft:movingBlock //TODO: minecraft:normal_stone_stairs //TODO: minecraft:observer //TODO: minecraft:piston //TODO: minecraft:pistonArmCollision //TODO: minecraft:polished_andesite_stairs //TODO: minecraft:polished_diorite_stairs //TODO: minecraft:polished_granite_stairs //TODO: minecraft:portal //TODO: minecraft:powered_comparator //TODO: minecraft:prismarine_bricks_stairs //TODO: minecraft:prismarine_stairs //TODO: minecraft:red_nether_brick_stairs //TODO: minecraft:repeating_command_block //TODO: minecraft:scaffolding //TODO: minecraft:sea_pickle //TODO: minecraft:seagrass //TODO: minecraft:shulker_box //TODO: minecraft:slime //TODO: minecraft:smithing_table //TODO: minecraft:smoker //TODO: minecraft:smooth_quartz_stairs //TODO: minecraft:smooth_red_sandstone_stairs //TODO: minecraft:smooth_sandstone_stairs //TODO: minecraft:smooth_stone //TODO: minecraft:spruce_button //TODO: minecraft:spruce_pressure_plate //TODO: minecraft:spruce_standing_sign //TODO: minecraft:spruce_trapdoor //TODO: minecraft:spruce_wall_sign //TODO: minecraft:sticky_piston //TODO: minecraft:stone_slab3 //TODO: minecraft:stone_slab4 //TODO: minecraft:stripped_acacia_log //TODO: minecraft:stripped_birch_log //TODO: minecraft:stripped_dark_oak_log //TODO: minecraft:stripped_jungle_log //TODO: minecraft:stripped_oak_log //TODO: minecraft:stripped_spruce_log //TODO: minecraft:structure_block //TODO: minecraft:turtle_egg //TODO: minecraft:undyed_shulker_box //TODO: minecraft:unpowered_comparator } public static function isInit() : bool{ return self::$fullList !== null; } /** * Registers a block type into the index. Plugins may use this method to register new block types or override * existing ones. * * NOTE: If you are registering a new block type, you will need to add it to the creative inventory yourself - it * will not automatically appear there. * * @param Block $block * @param bool $override Whether to override existing registrations * * @throws \RuntimeException if something attempted to override an already-registered block without specifying the * $override parameter. */ public static function register(Block $block, bool $override = false) : void{ $variant = $block->getIdInfo()->getVariant(); $stateMask = $block->getStateBitmask(); if(($variant & $stateMask) !== 0){ throw new \InvalidArgumentException("Block variant collides with state bitmask"); } foreach($block->getIdInfo()->getAllBlockIds() as $id){ if(!$override and self::isRegistered($id, $variant)){ throw new \InvalidArgumentException("Block registration $id:$variant conflicts with an existing block"); } for($m = $variant; $m <= ($variant | $stateMask); ++$m){ if(($m & ~$stateMask) !== $variant){ continue; } if(!$override and self::isRegistered($id, $m)){ throw new \InvalidArgumentException("Block registration " . get_class($block) . " has states which conflict with other blocks"); } $index = ($id << 4) | $m; $v = clone $block; try{ $v->readStateFromData($id, $m & $stateMask); if($v->getMeta() !== $m){ throw new InvalidBlockStateException("Corrupted meta"); //don't register anything that isn't the same when we read it back again } }catch(InvalidBlockStateException $e){ //invalid property combination continue; } self::fillStaticArrays($index, $v); } if(!self::isRegistered($id, $variant)){ self::fillStaticArrays(($id << 4) | $variant, $block); //register default state mapped to variant, for blocks which don't use 0 as valid state } } } private static function fillStaticArrays(int $index, Block $block) : void{ self::$fullList[$index] = $block; self::$lightFilter[$index] = min(15, $block->getLightFilter() + 1); //opacity plus 1 standard light filter self::$diffusesSkyLight[$index] = $block->diffusesSkyLight(); self::$blastResistance[$index] = $block->getBlastResistance(); } /** * Returns a new Block instance with the specified ID, meta and position. * * @param int $id * @param int $meta * @param Position $pos * * @return Block */ public static function get(int $id, int $meta = 0, ?Position $pos = null) : Block{ if($meta < 0 or $meta > 0xf){ throw new \InvalidArgumentException("Block meta value $meta is out of bounds"); } /** @var Block|null $block */ $block = null; try{ $index = ($id << 4) | $meta; if(self::$fullList[$index] !== null){ $block = clone self::$fullList[$index]; } }catch(\RuntimeException $e){ throw new \InvalidArgumentException("Block ID $id is out of bounds"); } if($block === null){ $block = new UnknownBlock(new BID($id, $meta)); } if($pos !== null){ $block->position($pos->getLevel(), $pos->getFloorX(), $pos->getFloorY(), $pos->getFloorZ()); } return $block; } public static function fromFullBlock(int $fullState, ?Position $pos = null) : Block{ return self::get($fullState >> 4, $fullState & 0xf, $pos); } /** * Returns whether a specified block state is already registered in the block factory. * * @param int $id * @param int $meta * * @return bool */ public static function isRegistered(int $id, int $meta = 0) : bool{ $b = self::$fullList[($id << 4) | $meta]; return $b !== null and !($b instanceof UnknownBlock); } public static function registerStaticRuntimeIdMappings() : void{ /** @var mixed[] $runtimeIdMap */ $runtimeIdMap = json_decode(file_get_contents(\pocketmine\RESOURCE_PATH . "runtimeid_table.json"), true); $legacyIdMap = json_decode(file_get_contents(\pocketmine\RESOURCE_PATH . "legacy_id_map.json"), true); foreach($runtimeIdMap as $k => $obj){ //this has to use the json offset to make sure the mapping is consistent with what we send over network, even though we aren't using all the entries if(!isset($legacyIdMap[$obj["name"]])){ continue; } self::registerMapping($k, $legacyIdMap[$obj["name"]], $obj["data"]); } } /** * @internal * * @param int $id * @param int $meta * * @return int */ public static function toStaticRuntimeId(int $id, int $meta = 0) : int{ /* * try id+meta first * if not found, try id+0 (strip meta) * if still not found, return update! block */ return self::$staticRuntimeIdMap[($id << 4) | $meta] ?? self::$staticRuntimeIdMap[$id << 4] ?? self::$staticRuntimeIdMap[BlockIds::INFO_UPDATE << 4]; } /** * @internal * * @param int $runtimeId * * @return int[] [id, meta] */ public static function fromStaticRuntimeId(int $runtimeId) : array{ $v = self::$legacyIdMap[$runtimeId]; return [$v >> 4, $v & 0xf]; } private static function registerMapping(int $staticRuntimeId, int $legacyId, int $legacyMeta) : void{ self::$staticRuntimeIdMap[($legacyId << 4) | $legacyMeta] = $staticRuntimeId; self::$legacyIdMap[$staticRuntimeId] = ($legacyId << 4) | $legacyMeta; self::$lastRuntimeId = max(self::$lastRuntimeId, $staticRuntimeId); } /** * @return Block[] */ public static function getAllKnownStates() : array{ return array_filter(self::$fullList->toArray(), function(?Block $v) : bool{ return $v !== null; }); } }