mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-10-17 04:06:54 +00:00
add unit testing on MaterialRepairRecipe
This commit is contained in:
@@ -25,7 +25,6 @@ namespace pocketmine\block\utils;
|
|||||||
|
|
||||||
use pocketmine\crafting\AnvilCraftResult;
|
use pocketmine\crafting\AnvilCraftResult;
|
||||||
use pocketmine\item\Item;
|
use pocketmine\item\Item;
|
||||||
use pocketmine\player\Player;
|
|
||||||
use pocketmine\Server;
|
use pocketmine\Server;
|
||||||
|
|
||||||
final class AnvilHelper{
|
final class AnvilHelper{
|
||||||
@@ -36,7 +35,7 @@ final class AnvilHelper{
|
|||||||
*
|
*
|
||||||
* Returns null if the operation can't do anything.
|
* Returns null if the operation can't do anything.
|
||||||
*/
|
*/
|
||||||
public static function calculateResult(Player $player, Item $base, Item $material, ?string $customName = null) : ?AnvilCraftResult{
|
public static function calculateResult(Item $base, Item $material, ?string $customName, bool $isCreative) : ?AnvilCraftResult{
|
||||||
|
|
||||||
$recipe = Server::getInstance()->getCraftingManager()->matchAnvilRecipe($base, $material);
|
$recipe = Server::getInstance()->getCraftingManager()->matchAnvilRecipe($base, $material);
|
||||||
if($recipe === null){
|
if($recipe === null){
|
||||||
@@ -58,7 +57,7 @@ final class AnvilHelper{
|
|||||||
$result = new AnvilCraftResult($xpCost, $resultItem, $result->getSacrificeResult());
|
$result = new AnvilCraftResult($xpCost, $resultItem, $result->getSacrificeResult());
|
||||||
}
|
}
|
||||||
|
|
||||||
if($result === null || $result->getXpCost() <= 0 || ($result->getXpCost() > self::COST_LIMIT && !$player->isCreative())){
|
if($result === null || $result->getXpCost() <= 0 || ($result->getXpCost() > self::COST_LIMIT && !$isCreative)){
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -23,9 +23,11 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace pocketmine\crafting;
|
namespace pocketmine\crafting;
|
||||||
|
|
||||||
|
use pocketmine\item\Durable;
|
||||||
use pocketmine\item\ToolTier;
|
use pocketmine\item\ToolTier;
|
||||||
use pocketmine\item\VanillaArmorMaterials;
|
use pocketmine\item\VanillaArmorMaterials;
|
||||||
use pocketmine\item\VanillaItems;
|
use pocketmine\item\VanillaItems;
|
||||||
|
use pocketmine\world\format\io\GlobalItemDataHandlers;
|
||||||
|
|
||||||
final class AnvilCraftingManagerDataFiller{
|
final class AnvilCraftingManagerDataFiller{
|
||||||
public static function fillData(CraftingManager $manager) : CraftingManager{
|
public static function fillData(CraftingManager $manager) : CraftingManager{
|
||||||
@@ -62,6 +64,16 @@ final class AnvilCraftingManagerDataFiller{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
foreach(VanillaItems::getAll() as $item){
|
||||||
|
if($item instanceof Durable){
|
||||||
|
$itemId = GlobalItemDataHandlers::getSerializer()->serializeType($item)->getName();
|
||||||
|
$manager->registerAnvilRecipe(new ItemCombineRecipe(
|
||||||
|
new MetaWildcardRecipeIngredient($itemId),
|
||||||
|
new MetaWildcardRecipeIngredient($itemId)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $manager;
|
return $manager;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -59,7 +59,7 @@ class MaterialRepairRecipe implements AnvilRecipe{
|
|||||||
return new AnvilCraftResult(
|
return new AnvilCraftResult(
|
||||||
$numberRepair,
|
$numberRepair,
|
||||||
(clone $input)->setDamage(max(0, $damage)),
|
(clone $input)->setDamage(max(0, $damage)),
|
||||||
(clone $material)->pop($numberRepair)
|
(clone $material)->setCount($material->getCount() - $numberRepair)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -46,7 +46,7 @@ class AnvilTransaction extends InventoryTransaction{
|
|||||||
Player $source,
|
Player $source,
|
||||||
private readonly AnvilCraftResult $expectedResult,
|
private readonly AnvilCraftResult $expectedResult,
|
||||||
private readonly ?string $customName
|
private readonly ?string $customName
|
||||||
) {
|
){
|
||||||
parent::__construct($source);
|
parent::__construct($source);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,8 +62,8 @@ class AnvilTransaction extends InventoryTransaction{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function validateInputs(Item $base, Item $material, Item $expectedOutput) : ?int {
|
private function validateInputs(Item $base, Item $material, Item $expectedOutput) : ?int{
|
||||||
$calculAttempt = AnvilHelper::calculateResult($this->source, $base, $material, $this->customName);
|
$calculAttempt = AnvilHelper::calculateResult($base, $material, $this->customName, $this->source->isCreative());
|
||||||
if($calculAttempt === null){
|
if($calculAttempt === null){
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@@ -355,7 +355,7 @@ class ItemStackRequestExecutor{
|
|||||||
}elseif($action instanceof CraftRecipeOptionalStackRequestAction){
|
}elseif($action instanceof CraftRecipeOptionalStackRequestAction){
|
||||||
$window = $this->player->getCurrentWindow();
|
$window = $this->player->getCurrentWindow();
|
||||||
if($window instanceof AnvilInventory){
|
if($window instanceof AnvilInventory){
|
||||||
$result = AnvilHelper::calculateResult($this->player, $window->getInput(), $window->getMaterial(), $this->request->getFilterStrings()[0] ?? null);
|
$result = AnvilHelper::calculateResult($window->getInput(), $window->getMaterial(), $this->request->getFilterStrings()[0] ?? null, $this->player->isCreative());
|
||||||
if($result !== null){
|
if($result !== null){
|
||||||
$this->specialTransaction = new AnvilTransaction($this->player, $result, $this->request->getFilterStrings()[0] ?? null);
|
$this->specialTransaction = new AnvilTransaction($this->player, $result, $this->request->getFilterStrings()[0] ?? null);
|
||||||
$this->setNextCreatedItem($result->getOutput());
|
$this->setNextCreatedItem($result->getOutput());
|
||||||
|
104
tests/phpunit/crafting/AnvilCraftTest.php
Normal file
104
tests/phpunit/crafting/AnvilCraftTest.php
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
<?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 Generator;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use pocketmine\item\Item;
|
||||||
|
use pocketmine\item\VanillaItems;
|
||||||
|
use function floor;
|
||||||
|
|
||||||
|
class AnvilCraftTest extends TestCase{
|
||||||
|
public static function materialRepairRecipe() : Generator{
|
||||||
|
yield "No repair available" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE(),
|
||||||
|
VanillaItems::DIAMOND(),
|
||||||
|
null
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Repair one damage" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage(1),
|
||||||
|
VanillaItems::DIAMOND(),
|
||||||
|
new AnvilCraftResult(1, VanillaItems::DIAMOND_PICKAXE(), null)
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Repair one damage with more materials than expected" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage(1),
|
||||||
|
VanillaItems::DIAMOND()->setCount(2),
|
||||||
|
new AnvilCraftResult(1, VanillaItems::DIAMOND_PICKAXE(), VanillaItems::DIAMOND())
|
||||||
|
];
|
||||||
|
|
||||||
|
$diamondPickaxeQuarter = (int) floor(VanillaItems::DIAMOND_PICKAXE()->getMaxDurability() / 4);
|
||||||
|
yield "Repair one quarter" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage($diamondPickaxeQuarter),
|
||||||
|
VanillaItems::DIAMOND()->setCount(1),
|
||||||
|
new AnvilCraftResult(1, VanillaItems::DIAMOND_PICKAXE(), null)
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Repair one quarter plus 1" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage($diamondPickaxeQuarter + 1),
|
||||||
|
VanillaItems::DIAMOND()->setCount(1),
|
||||||
|
new AnvilCraftResult(1, VanillaItems::DIAMOND_PICKAXE()->setDamage(1), null)
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Repair more than one quarter" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage($diamondPickaxeQuarter * 2),
|
||||||
|
VanillaItems::DIAMOND()->setCount(2),
|
||||||
|
new AnvilCraftResult(2, VanillaItems::DIAMOND_PICKAXE(), null)
|
||||||
|
];
|
||||||
|
|
||||||
|
yield "Repair more than one quarter with more materials than expected" => [
|
||||||
|
VanillaItems::DIAMOND_PICKAXE()->setDamage($diamondPickaxeQuarter * 2),
|
||||||
|
VanillaItems::DIAMOND()->setCount(3),
|
||||||
|
new AnvilCraftResult(2, VanillaItems::DIAMOND_PICKAXE(), VanillaItems::DIAMOND()->setCount(1))
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider materialRepairRecipe
|
||||||
|
*/
|
||||||
|
public function testMaterialRepairRecipe(Item $base, Item $material, ?AnvilCraftResult $expected) : void{
|
||||||
|
$recipe = new MaterialRepairRecipe(
|
||||||
|
new ExactRecipeIngredient((clone $base)->setCount(1)),
|
||||||
|
new ExactRecipeIngredient((clone $material)->setCount(1))
|
||||||
|
);
|
||||||
|
$result = $recipe->getResultFor($base, $material);
|
||||||
|
if($expected === null){
|
||||||
|
self::assertNull($result, "Recipe did not match expected result");
|
||||||
|
return;
|
||||||
|
}else{
|
||||||
|
self::assertNotNull($result, "Recipe did not match expected result");
|
||||||
|
}
|
||||||
|
self::assertEquals($expected->getXpCost(), $result->getXpCost(), "XP cost did not match expected result");
|
||||||
|
self::assertTrue($expected->getOutput()->equalsExact($result->getOutput()), "Recipe output did not match expected result");
|
||||||
|
$sacrificeResult = $expected->getSacrificeResult();
|
||||||
|
if($sacrificeResult !== null){
|
||||||
|
$resultExpected = $result->getSacrificeResult();
|
||||||
|
self::assertNotNull($resultExpected, "Recipe sacrifice result did not match expected result");
|
||||||
|
self::assertTrue($sacrificeResult->equalsExact($resultExpected), "Recipe sacrifice result did not match expected result");
|
||||||
|
}else{
|
||||||
|
self::assertNull($result->getSacrificeResult(), "Recipe sacrifice result did not match expected result");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user