mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-06-17 08:54:22 +00:00
232 lines
8.6 KiB
PHP
232 lines
8.6 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\world\light;
|
|
|
|
use pocketmine\world\format\Chunk;
|
|
use pocketmine\world\format\HeightArray;
|
|
use pocketmine\world\format\LightArray;
|
|
use pocketmine\world\format\SubChunk;
|
|
use pocketmine\world\utils\SubChunkExplorer;
|
|
use pocketmine\world\utils\SubChunkExplorerStatus;
|
|
use pocketmine\world\World;
|
|
use function max;
|
|
|
|
class SkyLightUpdate extends LightUpdate{
|
|
/**
|
|
* @param int[] $lightFilters
|
|
* @param true[] $directSkyLightBlockers
|
|
* @phpstan-param array<int, int> $lightFilters
|
|
* @phpstan-param array<int, true> $directSkyLightBlockers
|
|
*/
|
|
public function __construct(
|
|
SubChunkExplorer $subChunkExplorer,
|
|
array $lightFilters,
|
|
private array $directSkyLightBlockers
|
|
){
|
|
parent::__construct($subChunkExplorer, $lightFilters);
|
|
}
|
|
|
|
protected function getCurrentLightArray() : LightArray{
|
|
return $this->subChunkExplorer->currentSubChunk->getBlockSkyLightArray();
|
|
}
|
|
|
|
protected function getEffectiveLight(int $x, int $y, int $z) : int{
|
|
if($y >= World::Y_MAX){
|
|
$this->subChunkExplorer->invalidate();
|
|
return 15;
|
|
}
|
|
return parent::getEffectiveLight($x, $y, $z);
|
|
}
|
|
|
|
public function recalculateNode(int $x, int $y, int $z) : void{
|
|
if($this->subChunkExplorer->moveTo($x, $y, $z) === SubChunkExplorerStatus::INVALID){
|
|
return;
|
|
}
|
|
$chunk = $this->subChunkExplorer->currentChunk;
|
|
|
|
$oldHeightMap = $chunk->getHeightMap($x & Chunk::COORD_MASK, $z & Chunk::COORD_MASK);
|
|
$source = $this->subChunkExplorer->currentSubChunk->getFullBlock($x & SubChunk::COORD_MASK, $y & SubChunk::COORD_MASK, $z & SubChunk::COORD_MASK);
|
|
|
|
$yPlusOne = $y + 1;
|
|
|
|
if($yPlusOne === $oldHeightMap){ //Block changed directly beneath the heightmap. Check if a block was removed or changed to a different light-filter.
|
|
$newHeightMap = self::recalculateHeightMapColumn($chunk, $x & Chunk::COORD_MASK, $z & Chunk::COORD_MASK, $this->directSkyLightBlockers);
|
|
$chunk->setHeightMap($x & Chunk::COORD_MASK, $z & Chunk::COORD_MASK, $newHeightMap);
|
|
}elseif($yPlusOne > $oldHeightMap){ //Block changed above the heightmap.
|
|
if(isset($this->directSkyLightBlockers[$source])){
|
|
$chunk->setHeightMap($x & Chunk::COORD_MASK, $z & Chunk::COORD_MASK, $yPlusOne);
|
|
$newHeightMap = $yPlusOne;
|
|
}else{ //Block changed which has no effect on direct sky light, for example placing or removing glass.
|
|
return;
|
|
}
|
|
}else{ //Block changed below heightmap
|
|
$newHeightMap = $oldHeightMap;
|
|
}
|
|
|
|
if($newHeightMap > $oldHeightMap){ //Heightmap increase, block placed, remove sky light
|
|
for($i = $y; $i >= $oldHeightMap; --$i){
|
|
$this->setAndUpdateLight($x, $i, $z, 0); //Remove all light beneath, adjacent recalculation will handle the rest.
|
|
}
|
|
}elseif($newHeightMap < $oldHeightMap){ //Heightmap decrease, block changed or removed, add sky light
|
|
for($i = $y; $i >= $newHeightMap; --$i){
|
|
$this->setAndUpdateLight($x, $i, $z, 15);
|
|
}
|
|
}else{ //No heightmap change, block changed "underground"
|
|
$this->setAndUpdateLight($x, $y, $z, max(0, $this->getHighestAdjacentLight($x, $y, $z) - ($this->lightFilters[$source] ?? self::BASE_LIGHT_FILTER)));
|
|
}
|
|
}
|
|
|
|
public function recalculateChunk(int $chunkX, int $chunkZ) : int{
|
|
if($this->subChunkExplorer->moveToChunk($chunkX, 0, $chunkZ) === SubChunkExplorerStatus::INVALID){
|
|
throw new \InvalidArgumentException("Chunk $chunkX $chunkZ does not exist");
|
|
}
|
|
$chunk = $this->subChunkExplorer->currentChunk;
|
|
|
|
$newHeightMap = self::recalculateHeightMap($chunk, $this->directSkyLightBlockers);
|
|
$chunk->setHeightMapArray($newHeightMap->getValues());
|
|
|
|
//setAndUpdateLight() won't bother propagating from nodes that are already what we want to change them to, so we
|
|
//have to avoid filling full light for any subchunk that contains a heightmap Y coordinate
|
|
$highestHeightMapPlusOne = max($chunk->getHeightMapArray()) + 1;
|
|
$lowestClearSubChunk = ($highestHeightMapPlusOne >> SubChunk::COORD_BIT_SIZE) + (($highestHeightMapPlusOne & SubChunk::COORD_MASK) !== 0 ? 1 : 0);
|
|
for($y = Chunk::MIN_SUBCHUNK_INDEX; $y < $lowestClearSubChunk && $y <= Chunk::MAX_SUBCHUNK_INDEX; $y++){
|
|
$chunk->getSubChunk($y)->setBlockSkyLightArray(LightArray::fill(0));
|
|
}
|
|
for($y = $lowestClearSubChunk; $y <= Chunk::MAX_SUBCHUNK_INDEX; $y++){
|
|
$chunk->getSubChunk($y)->setBlockSkyLightArray(LightArray::fill(15));
|
|
}
|
|
|
|
$lightSources = 0;
|
|
|
|
$baseX = $chunkX << Chunk::COORD_BIT_SIZE;
|
|
$baseZ = $chunkZ << Chunk::COORD_BIT_SIZE;
|
|
for($x = 0; $x < Chunk::EDGE_LENGTH; ++$x){
|
|
for($z = 0; $z < Chunk::EDGE_LENGTH; ++$z){
|
|
$currentHeight = $chunk->getHeightMap($x, $z);
|
|
$maxAdjacentHeight = World::Y_MIN;
|
|
if($x !== 0){
|
|
$maxAdjacentHeight = max($maxAdjacentHeight, $chunk->getHeightMap($x - 1, $z));
|
|
}
|
|
if($x !== 15){
|
|
$maxAdjacentHeight = max($maxAdjacentHeight, $chunk->getHeightMap($x + 1, $z));
|
|
}
|
|
if($z !== 0){
|
|
$maxAdjacentHeight = max($maxAdjacentHeight, $chunk->getHeightMap($x, $z - 1));
|
|
}
|
|
if($z !== 15){
|
|
$maxAdjacentHeight = max($maxAdjacentHeight, $chunk->getHeightMap($x, $z + 1));
|
|
}
|
|
|
|
/*
|
|
* We skip the top two blocks between current height and max adjacent (if there's a difference) because:
|
|
* - the block next to the highest adjacent will do nothing during propagation (it's surrounded by 15s)
|
|
* - the block below that block will do the same as the node in the highest adjacent
|
|
* NOTE: If block opacity becomes direction-aware in the future, the second point will become invalid.
|
|
*/
|
|
$nodeColumnEnd = max($currentHeight, $maxAdjacentHeight - 2);
|
|
|
|
for($y = $currentHeight; $y <= $nodeColumnEnd; $y++){
|
|
$this->setAndUpdateLight($x + $baseX, $y, $z + $baseZ, 15);
|
|
$lightSources++;
|
|
}
|
|
for($y = $nodeColumnEnd + 1, $yMax = $lowestClearSubChunk * SubChunk::EDGE_LENGTH; $y < $yMax; $y++){
|
|
if($this->subChunkExplorer->moveTo($x + $baseX, $y, $z + $baseZ) !== SubChunkExplorerStatus::INVALID){
|
|
$this->getCurrentLightArray()->set($x, $y & SubChunk::COORD_MASK, $z, 15);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return $lightSources;
|
|
}
|
|
|
|
/**
|
|
* Recalculates the heightmap for the whole chunk.
|
|
*
|
|
* @param true[] $directSkyLightBlockers
|
|
* @phpstan-param array<int, true> $directSkyLightBlockers
|
|
*/
|
|
private static function recalculateHeightMap(Chunk $chunk, array $directSkyLightBlockers) : HeightArray{
|
|
$maxSubChunkY = Chunk::MAX_SUBCHUNK_INDEX;
|
|
for(; $maxSubChunkY >= Chunk::MIN_SUBCHUNK_INDEX; $maxSubChunkY--){
|
|
if(!$chunk->getSubChunk($maxSubChunkY)->isEmptyFast()){
|
|
break;
|
|
}
|
|
}
|
|
$result = HeightArray::fill(World::Y_MIN);
|
|
if($maxSubChunkY < Chunk::MIN_SUBCHUNK_INDEX){ //whole column is definitely empty
|
|
return $result;
|
|
}
|
|
|
|
for($z = 0; $z < Chunk::EDGE_LENGTH; ++$z){
|
|
for($x = 0; $x < Chunk::EDGE_LENGTH; ++$x){
|
|
$y = null;
|
|
for($subChunkY = $maxSubChunkY; $subChunkY >= Chunk::MIN_SUBCHUNK_INDEX; $subChunkY--){
|
|
$subHighestBlockY = $chunk->getSubChunk($subChunkY)->getHighestBlockAt($x, $z);
|
|
if($subHighestBlockY !== null){
|
|
$y = ($subChunkY * SubChunk::EDGE_LENGTH) + $subHighestBlockY;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if($y === null){ //no blocks in the column
|
|
$result->set($x, $z, World::Y_MIN);
|
|
}else{
|
|
for(; $y >= World::Y_MIN; --$y){
|
|
if(isset($directSkyLightBlockers[$chunk->getFullBlock($x, $y, $z)])){
|
|
$result->set($x, $z, $y + 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Recalculates the heightmap for the block column at the specified X/Z chunk coordinates
|
|
*
|
|
* @param int $x 0-15
|
|
* @param int $z 0-15
|
|
* @param true[] $directSkyLightBlockers
|
|
* @phpstan-param array<int, true> $directSkyLightBlockers
|
|
*
|
|
* @return int New calculated heightmap value (0-256 inclusive)
|
|
*/
|
|
private static function recalculateHeightMapColumn(Chunk $chunk, int $x, int $z, array $directSkyLightBlockers) : int{
|
|
$y = $chunk->getHighestBlockAt($x, $z);
|
|
if($y === null){
|
|
return World::Y_MIN;
|
|
}
|
|
for(; $y >= World::Y_MIN; --$y){
|
|
if(isset($directSkyLightBlockers[$chunk->getFullBlock($x, $y, $z)])){
|
|
break;
|
|
}
|
|
}
|
|
|
|
return $y + 1;
|
|
}
|
|
}
|