mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-09-06 09:56:06 +00:00
This commit completely revamps the way that blocks are represented in memory at runtime. Instead of being represented by legacy Mojang block IDs and metadata, which are dated, limited and unchangeable, we now use custom PM block IDs, which are generated from VanillaBlocks. This means we have full control of how they are assigned, which opens the doors to finally addressing inconsistencies like glazed terracotta, stripped logs handling, etc. To represent state, BlockDataReader and BlockDataWriter have been introduced, and are used by blocks with state information to pack said information into a binary form that can be stored on a chunk at runtime. Conceptually it's pretty similar to legacy metadata, but the actual format shares no resemblance whatsoever to legacy metadata, and is fully controlled by PM. This means that the 'state data' may change in serialization format at any time, so it should **NOT** be stored on disk or in a config. In the future, this will be improved using more auto-generated code and attributes, instead of hand-baked decodeState() and encodeState(). For now, this opens the gateway to a significant expansion of features. It's not ideal, but it's a big step forwards.
148 lines
4.3 KiB
PHP
148 lines
4.3 KiB
PHP
<?php
|
|
|
|
/*
|
|
*
|
|
* ____ _ _ __ __ _ __ __ ____
|
|
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
|
|
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
|
|
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
|
|
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* @author PocketMine Team
|
|
* @link http://www.pocketmine.net/
|
|
*
|
|
*
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace pocketmine\block;
|
|
|
|
use pocketmine\block\utils\BlockDataReader;
|
|
use pocketmine\block\utils\BlockDataWriter;
|
|
use pocketmine\block\utils\SupportType;
|
|
use pocketmine\entity\Entity;
|
|
use pocketmine\event\block\BlockGrowEvent;
|
|
use pocketmine\event\entity\EntityDamageByBlockEvent;
|
|
use pocketmine\event\entity\EntityDamageEvent;
|
|
use pocketmine\item\Item;
|
|
use pocketmine\math\AxisAlignedBB;
|
|
use pocketmine\math\Facing;
|
|
use pocketmine\math\Vector3;
|
|
use pocketmine\player\Player;
|
|
use pocketmine\world\BlockTransaction;
|
|
|
|
class Cactus extends Transparent{
|
|
public const MAX_AGE = 15;
|
|
|
|
protected int $age = 0;
|
|
|
|
protected function decodeState(BlockDataReader $r) : void{
|
|
$this->age = $r->readBoundedInt(4, 0, self::MAX_AGE);
|
|
}
|
|
|
|
protected function encodeState(BlockDataWriter $w) : void{
|
|
$w->writeInt(4, $this->age);
|
|
}
|
|
|
|
public function getAge() : int{ return $this->age; }
|
|
|
|
/** @return $this */
|
|
public function setAge(int $age) : self{
|
|
if($age < 0 || $age > self::MAX_AGE){
|
|
throw new \InvalidArgumentException("Age must be in range 0 ... " . self::MAX_AGE);
|
|
}
|
|
$this->age = $age;
|
|
return $this;
|
|
}
|
|
|
|
public function hasEntityCollision() : bool{
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @return AxisAlignedBB[]
|
|
*/
|
|
protected function recalculateCollisionBoxes() : array{
|
|
static $shrinkSize = 1 / 16;
|
|
return [AxisAlignedBB::one()->contract($shrinkSize, 0, $shrinkSize)->trim(Facing::UP, $shrinkSize)];
|
|
}
|
|
|
|
public function getSupportType(int $facing) : SupportType{
|
|
return SupportType::NONE();
|
|
}
|
|
|
|
public function onEntityInside(Entity $entity) : bool{
|
|
$ev = new EntityDamageByBlockEvent($this, $entity, EntityDamageEvent::CAUSE_CONTACT, 1);
|
|
$entity->attack($ev);
|
|
return true;
|
|
}
|
|
|
|
public function onNearbyBlockChange() : void{
|
|
$down = $this->getSide(Facing::DOWN);
|
|
if($down->getTypeId() !== BlockTypeIds::SAND && $down->getTypeId() !== BlockTypeIds::RED_SAND && !$down->isSameType($this)){
|
|
$this->position->getWorld()->useBreakOn($this->position);
|
|
}else{
|
|
foreach(Facing::HORIZONTAL as $side){
|
|
$b = $this->getSide($side);
|
|
if($b->isSolid()){
|
|
$this->position->getWorld()->useBreakOn($this->position);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public function ticksRandomly() : bool{
|
|
return true;
|
|
}
|
|
|
|
public function onRandomTick() : void{
|
|
if(!$this->getSide(Facing::DOWN)->isSameType($this)){
|
|
if($this->age === self::MAX_AGE){
|
|
for($y = 1; $y < 3; ++$y){
|
|
if(!$this->position->getWorld()->isInWorld($this->position->x, $this->position->y + $y, $this->position->z)){
|
|
break;
|
|
}
|
|
$b = $this->position->getWorld()->getBlockAt($this->position->x, $this->position->y + $y, $this->position->z);
|
|
if($b->getTypeId() === BlockTypeIds::AIR){
|
|
$ev = new BlockGrowEvent($b, VanillaBlocks::CACTUS());
|
|
$ev->call();
|
|
if($ev->isCancelled()){
|
|
break;
|
|
}
|
|
$this->position->getWorld()->setBlock($b->position, $ev->getNewState());
|
|
}else{
|
|
break;
|
|
}
|
|
}
|
|
$this->age = 0;
|
|
$this->position->getWorld()->setBlock($this->position, $this);
|
|
}else{
|
|
++$this->age;
|
|
$this->position->getWorld()->setBlock($this->position, $this);
|
|
}
|
|
}
|
|
}
|
|
|
|
public function place(BlockTransaction $tx, Item $item, Block $blockReplace, Block $blockClicked, int $face, Vector3 $clickVector, ?Player $player = null) : bool{
|
|
$down = $this->getSide(Facing::DOWN);
|
|
if($down->getTypeId() === BlockTypeIds::SAND || $down->getTypeId() === BlockTypeIds::RED_SAND || $down->isSameType($this)){
|
|
foreach(Facing::HORIZONTAL as $side){
|
|
if($this->getSide($side)->isSolid()){
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return parent::place($tx, $item, $blockReplace, $blockClicked, $face, $clickVector, $player);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|