mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-05-15 10:19:39 +00:00
275 lines
7.5 KiB
PHP
275 lines
7.5 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\crafting;
|
|
|
|
use pocketmine\item\Item;
|
|
use pocketmine\network\mcpe\compression\CompressBatchPromise;
|
|
use pocketmine\network\mcpe\compression\Zlib;
|
|
use pocketmine\network\mcpe\PacketBatch;
|
|
use pocketmine\network\mcpe\protocol\CraftingDataPacket;
|
|
use pocketmine\timings\Timings;
|
|
use function array_map;
|
|
use function file_get_contents;
|
|
use function json_decode;
|
|
use function json_encode;
|
|
use function usort;
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
class CraftingManager{
|
|
/** @var ShapedRecipe[][] */
|
|
protected $shapedRecipes = [];
|
|
/** @var ShapelessRecipe[][] */
|
|
protected $shapelessRecipes = [];
|
|
/** @var FurnaceRecipe[] */
|
|
protected $furnaceRecipes = [];
|
|
|
|
/** @var CompressBatchPromise|null */
|
|
private $craftingDataCache;
|
|
|
|
public function __construct(){
|
|
$this->init();
|
|
}
|
|
|
|
public function init() : void{
|
|
$recipes = json_decode(file_get_contents(\pocketmine\RESOURCE_PATH . "vanilla" . DIRECTORY_SEPARATOR . "recipes.json"), true);
|
|
|
|
$itemDeserializerFunc = \Closure::fromCallable([Item::class, 'jsonDeserialize']);
|
|
foreach($recipes as $recipe){
|
|
switch($recipe["type"]){
|
|
case "shapeless":
|
|
if($recipe["block"] !== "crafting_table"){ //TODO: filter others out for now to avoid breaking economics
|
|
break;
|
|
}
|
|
$this->registerShapelessRecipe(new ShapelessRecipe(
|
|
array_map($itemDeserializerFunc, $recipe["input"]),
|
|
array_map($itemDeserializerFunc, $recipe["output"])
|
|
));
|
|
break;
|
|
case "shaped":
|
|
if($recipe["block"] !== "crafting_table"){ //TODO: filter others out for now to avoid breaking economics
|
|
break;
|
|
}
|
|
$this->registerShapedRecipe(new ShapedRecipe(
|
|
$recipe["shape"],
|
|
array_map($itemDeserializerFunc, $recipe["input"]),
|
|
array_map($itemDeserializerFunc, $recipe["output"])
|
|
));
|
|
break;
|
|
case "smelting":
|
|
if($recipe["block"] !== "furnace"){ //TODO: filter others out for now to avoid breaking economics
|
|
break;
|
|
}
|
|
$this->registerFurnaceRecipe(new FurnaceRecipe(
|
|
Item::jsonDeserialize($recipe["output"]),
|
|
Item::jsonDeserialize($recipe["input"]))
|
|
);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
$this->buildCraftingDataCache();
|
|
}
|
|
|
|
/**
|
|
* Rebuilds the cached CraftingDataPacket.
|
|
*/
|
|
public function buildCraftingDataCache() : void{
|
|
Timings::$craftingDataCacheRebuildTimer->startTiming();
|
|
$pk = new CraftingDataPacket();
|
|
$pk->cleanRecipes = true;
|
|
|
|
foreach($this->shapelessRecipes as $list){
|
|
foreach($list as $recipe){
|
|
$pk->addShapelessRecipe($recipe);
|
|
}
|
|
}
|
|
foreach($this->shapedRecipes as $list){
|
|
foreach($list as $recipe){
|
|
$pk->addShapedRecipe($recipe);
|
|
}
|
|
}
|
|
|
|
foreach($this->furnaceRecipes as $recipe){
|
|
$pk->addFurnaceRecipe($recipe);
|
|
}
|
|
|
|
$this->craftingDataCache = new CompressBatchPromise();
|
|
$this->craftingDataCache->resolve(Zlib::compress(PacketBatch::fromPackets($pk)->getBuffer()));
|
|
|
|
Timings::$craftingDataCacheRebuildTimer->stopTiming();
|
|
}
|
|
|
|
/**
|
|
* Returns a pre-compressed CraftingDataPacket for sending to players. Rebuilds the cache if it is not found.
|
|
*/
|
|
public function getCraftingDataPacket() : CompressBatchPromise{
|
|
if($this->craftingDataCache === null){
|
|
$this->buildCraftingDataCache();
|
|
}
|
|
|
|
return $this->craftingDataCache;
|
|
}
|
|
|
|
/**
|
|
* Function used to arrange Shapeless Recipe ingredient lists into a consistent order.
|
|
*/
|
|
public static function sort(Item $i1, Item $i2) : int{
|
|
//Use spaceship operator to compare each property, then try the next one if they are equivalent.
|
|
($retval = $i1->getId() <=> $i2->getId()) === 0 && ($retval = $i1->getMeta() <=> $i2->getMeta()) === 0 && ($retval = $i1->getCount() <=> $i2->getCount());
|
|
|
|
return $retval;
|
|
}
|
|
|
|
/**
|
|
* @param Item[] $items
|
|
*
|
|
* @return Item[]
|
|
*/
|
|
private static function pack(array $items) : array{
|
|
/** @var Item[] $result */
|
|
$result = [];
|
|
|
|
foreach($items as $i => $item){
|
|
foreach($result as $otherItem){
|
|
if($item->equals($otherItem)){
|
|
$otherItem->setCount($otherItem->getCount() + $item->getCount());
|
|
continue 2;
|
|
}
|
|
}
|
|
|
|
//No matching item found
|
|
$result[] = clone $item;
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
private static function hashOutputs(array $outputs) : string{
|
|
$outputs = self::pack($outputs);
|
|
usort($outputs, [self::class, "sort"]);
|
|
foreach($outputs as $o){
|
|
//this reduces accuracy of hash, but it's necessary to deal with recipe book shift-clicking stupidity
|
|
$o->setCount(1);
|
|
}
|
|
|
|
return json_encode($outputs);
|
|
}
|
|
|
|
/**
|
|
* @return ShapelessRecipe[][]
|
|
*/
|
|
public function getShapelessRecipes() : array{
|
|
return $this->shapelessRecipes;
|
|
}
|
|
|
|
/**
|
|
* @return ShapedRecipe[][]
|
|
*/
|
|
public function getShapedRecipes() : array{
|
|
return $this->shapedRecipes;
|
|
}
|
|
|
|
/**
|
|
* @return FurnaceRecipe[]
|
|
*/
|
|
public function getFurnaceRecipes() : array{
|
|
return $this->furnaceRecipes;
|
|
}
|
|
|
|
public function registerShapedRecipe(ShapedRecipe $recipe) : void{
|
|
$this->shapedRecipes[self::hashOutputs($recipe->getResults())][] = $recipe;
|
|
|
|
$this->craftingDataCache = null;
|
|
}
|
|
|
|
public function registerShapelessRecipe(ShapelessRecipe $recipe) : void{
|
|
$this->shapelessRecipes[self::hashOutputs($recipe->getResults())][] = $recipe;
|
|
|
|
$this->craftingDataCache = null;
|
|
}
|
|
|
|
public function registerFurnaceRecipe(FurnaceRecipe $recipe) : void{
|
|
$input = $recipe->getInput();
|
|
$this->furnaceRecipes[$input->getId() . ":" . ($input->hasAnyDamageValue() ? "?" : $input->getMeta())] = $recipe;
|
|
$this->craftingDataCache = null;
|
|
}
|
|
|
|
/**
|
|
* @param Item[] $outputs
|
|
*/
|
|
public function matchRecipe(CraftingGrid $grid, array $outputs) : ?CraftingRecipe{
|
|
//TODO: try to match special recipes before anything else (first they need to be implemented!)
|
|
|
|
$outputHash = self::hashOutputs($outputs);
|
|
|
|
if(isset($this->shapedRecipes[$outputHash])){
|
|
foreach($this->shapedRecipes[$outputHash] as $recipe){
|
|
if($recipe->matchesCraftingGrid($grid)){
|
|
return $recipe;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(isset($this->shapelessRecipes[$outputHash])){
|
|
foreach($this->shapelessRecipes[$outputHash] as $recipe){
|
|
if($recipe->matchesCraftingGrid($grid)){
|
|
return $recipe;
|
|
}
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param Item[] $outputs
|
|
*
|
|
* @return CraftingRecipe[]|\Generator
|
|
* @phpstan-return \Generator<int, CraftingRecipe, void, void>
|
|
*/
|
|
public function matchRecipeByOutputs(array $outputs) : \Generator{
|
|
//TODO: try to match special recipes before anything else (first they need to be implemented!)
|
|
|
|
$outputHash = self::hashOutputs($outputs);
|
|
|
|
if(isset($this->shapedRecipes[$outputHash])){
|
|
foreach($this->shapedRecipes[$outputHash] as $recipe){
|
|
yield $recipe;
|
|
}
|
|
}
|
|
|
|
if(isset($this->shapelessRecipes[$outputHash])){
|
|
foreach($this->shapelessRecipes[$outputHash] as $recipe){
|
|
yield $recipe;
|
|
}
|
|
}
|
|
}
|
|
|
|
public function matchFurnaceRecipe(Item $input) : ?FurnaceRecipe{
|
|
return $this->furnaceRecipes[$input->getId() . ":" . $input->getMeta()] ?? $this->furnaceRecipes[$input->getId() . ":?"] ?? null;
|
|
}
|
|
}
|