facing = $r->readHorizontalFacing(); $this->delay = $r->readBoundedInt(2, self::MIN_DELAY - 1, self::MAX_DELAY - 1) + 1; $this->powered = $r->readBool(); } protected function encodeState(BlockDataWriter $w) : void{ $w->writeHorizontalFacing($this->facing); $w->writeInt(2, $this->delay - 1); $w->writeBool($this->powered); } public function getDelay() : int{ return $this->delay; } /** @return $this */ public function setDelay(int $delay) : self{ if($delay < self::MIN_DELAY || $delay > self::MAX_DELAY){ throw new \InvalidArgumentException("Delay must be in range " . self::MIN_DELAY . " ... " . self::MAX_DELAY); } $this->delay = $delay; return $this; } /** * @return AxisAlignedBB[] */ protected function recalculateCollisionBoxes() : array{ return [AxisAlignedBB::one()->trim(Facing::UP, 7 / 8)]; } public function place(BlockTransaction $tx, Item $item, Block $blockReplace, Block $blockClicked, int $face, Vector3 $clickVector, ?Player $player = null) : bool{ if($this->canBeSupportedBy($blockReplace->getSide(Facing::DOWN))){ if($player !== null){ $this->facing = Facing::opposite($player->getHorizontalFacing()); } return parent::place($tx, $item, $blockReplace, $blockClicked, $face, $clickVector, $player); } return false; } public function onInteract(Item $item, int $face, Vector3 $clickVector, ?Player $player = null) : bool{ if(++$this->delay > self::MAX_DELAY){ $this->delay = self::MIN_DELAY; } $this->position->getWorld()->setBlock($this->position, $this); return true; } public function onNearbyBlockChange() : void{ if(!$this->canBeSupportedBy($this->getSide(Facing::DOWN))){ $this->position->getWorld()->useBreakOn($this->position); } } private function canBeSupportedBy(Block $block) : bool{ return !$block->getSupportType(Facing::UP)->equals(SupportType::NONE()); } //TODO: redstone functionality }