1
0
Fork 0
cuberite-2a/src/Item.h

267 lines
7.6 KiB
C
Raw Permalink Normal View History

// Item.h
// Declares the cItem class representing an item (in the inventory sense)
#pragma once
#include "Defines.h"
#include "Enchantments.h"
#include "WorldStorage/FireworksSerializer.h"
#include "Color.h"
// fwd:
class cItemHandler;
class cItemGrid;
class cColor;
namespace Json
{
class Value;
}
// tolua_begin
class cItem
{
public:
2014-02-11 05:30:11 -05:00
/** Creates an empty item */
cItem(void);
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Creates an item of the specified type, by default 1 piece with no damage and no enchantments */
cItem(
short a_ItemType,
char a_ItemCount = 1,
short a_ItemDamage = 0,
const AString & a_Enchantments = "",
const AString & a_CustomName = "",
const AStringVector & a_LoreTable = {}
);
2016-02-05 16:45:45 -05:00
// The constructor is disabled in code, because the compiler generates it anyway,
// but it needs to stay because ToLua needs to generate the binding for it
2017-08-28 09:36:23 -04:00
#ifdef TOLUA_EXPOSITION
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Creates an exact copy of the item */
2017-08-28 09:36:23 -04:00
cItem(const cItem & a_CopyFrom);
2016-02-05 16:45:45 -05:00
#endif
2016-02-05 16:45:45 -05:00
/** Empties the item and frees up any dynamic storage used by the internals. */
void Empty(void);
2016-02-05 16:45:45 -05:00
/** Empties the item and frees up any dynamic storage used by the internals.
TODO: What is the usage difference? Merge with Empty()? */
void Clear(void);
2016-02-05 16:45:45 -05:00
/** Returns true if the item represents an empty stack - either the type is invalid, or count is zero. */
bool IsEmpty(void) const
{
return ((m_ItemType <= 0) || (m_ItemCount <= 0));
}
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/* Returns true if this itemstack can stack with the specified stack (types match, enchantments etc.)
ItemCounts are ignored. */
bool IsEqual(const cItem & a_Item) const
{
return (
IsSameType(a_Item) &&
(m_ItemDamage == a_Item.m_ItemDamage) &&
(m_Enchantments == a_Item.m_Enchantments) &&
(m_CustomName == a_Item.m_CustomName) &&
(m_LoreTable == a_Item.m_LoreTable) &&
m_FireworkItem.IsEqualTo(a_Item.m_FireworkItem)
);
}
2016-02-05 16:45:45 -05:00
bool IsSameType(const cItem & a_Item) const
{
return (m_ItemType == a_Item.m_ItemType) || (IsEmpty() && a_Item.IsEmpty());
}
bool IsBothNameAndLoreEmpty(void) const
{
return (m_CustomName.empty() && m_LoreTable.empty());
}
bool IsCustomNameEmpty(void) const { return (m_CustomName.empty()); }
bool IsLoreEmpty(void) const { return (m_LoreTable.empty()); }
2014-02-11 05:30:11 -05:00
/** Returns a copy of this item with m_ItemCount set to 1. Useful to preserve enchantments etc. on stacked items */
cItem CopyOne(void) const;
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Adds the specified count to this object and returns the reference to self (useful for chaining) */
cItem & AddCount(char a_AmountToAdd);
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Returns the maximum damage value that this item can have; zero if damage is not applied */
short GetMaxDamage(void) const;
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Damages a weapon / tool. Returns true when damage reaches max value and the item should be destroyed */
bool DamageItem(short a_Amount = 1);
inline bool IsDamageable(void) const { return (GetMaxDamage() > 0); }
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Returns true if the item is stacked up to its maximum stacking. */
bool IsFullStack(void) const;
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Returns the maximum amount of stacked items of this type. */
2013-11-10 12:41:26 -05:00
char GetMaxStackSize(void) const;
// tolua_end
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Returns the cItemHandler responsible for this item type */
const cItemHandler & GetHandler(void) const;
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Saves the item data into JSON representation */
void GetJson(Json::Value & a_OutValue) const;
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Loads the item data from JSON representation */
void FromJson(const Json::Value & a_Value);
2016-02-05 16:45:45 -05:00
2014-08-30 09:06:43 -04:00
/** Returns true if the specified item type is enchantable.
If FromBook is true, the function is used in the anvil inventory with book enchantments.
So it checks the "only book enchantments" too. Example: You can only enchant a hoe with a book. */
static bool IsEnchantable(short a_ItemType, bool a_FromBook = false); // tolua_export
2014-04-17 07:15:35 -04:00
/** Returns the enchantability of the item. When the item hasn't a enchantability, it will returns 0 */
unsigned GetEnchantability(); // tolua_export
2014-04-17 07:15:35 -04:00
/** Randomly enchants the item using the specified number of XP levels.
Returns true if the item was enchanted, false if not (not enchantable / too many enchantments already).
Randomness is derived from the provided PRNG. */
bool EnchantByXPLevels(unsigned a_NumXPLevels, MTRand & a_Random); // Exported in ManualBindings.cpp
2014-04-17 07:15:35 -04:00
/** Adds this specific enchantment to this item, returning the cost.
FromBook specifies whether the enchantment should be treated as coming
from a book. If true, then the cost returned uses the book values, if
false it uses the normal item multipliers. */
int AddEnchantment(int a_EnchantmentID, unsigned int a_Level, bool a_FromBook); // tolua_export
/** Adds the enchantments on a_Other to this item, returning the
XP cost of the transfer. */
int AddEnchantmentsFromItem(const cItem & a_Other); // tolua_export
/** Returns whether or not this item is allowed to have the given enchantment. Note: Does not check whether the enchantment is exclusive with the current enchantments on the item. */
bool CanHaveEnchantment(int a_EnchantmentID);
// tolua_begin
2016-02-05 16:45:45 -05:00
short m_ItemType;
char m_ItemCount;
short m_ItemDamage;
cEnchantments m_Enchantments;
AString m_CustomName;
// tolua_end
AStringVector m_LoreTable; // Exported in ManualBindings.cpp
Introduce recipe book functionality (#4493) * Introduce recipe book functionality The recipe book helps especially new players. Missing it gives the impression that cuberite is not as advanced as it is. The handling of the recipe book uses the following functions: - Unlock Recipes (https://wiki.vg/index.php?title=Protocol&oldid=14204#Unlock_Recipes) to make recipes available and show the notification for new recipes. Initialization is done on player login for known ones, the update is done when new items are discovered. - Craft Recipe Request (https://wiki.vg/index.php?title=Protocol&oldid=14204#Craft_Recipe_Request) when the user selects a recipe from the recipe book to fill the slots. Known recipes are initialized on player login via `Unlock Recipes` with `Action` 0. As soon as a new recipe is discovered this is added via `Unlock Recipes` with `Action` 1. To be able to know and recognize new recipes the player class is extended with `KnownItems` and `KnownRecipes`. As soon as a player touches an item this is compared to the list of `KnownItems`, if the item is unknown the recipes are checked for this item and the other ingredients are checked with the list of `KnownItems`. If a full match is discovered the recipe is unlocked with the client and stored in the `KnownRecipes`. To unlock recipes the recipe ID is sent to the client. A mapping file (for protocol 1.12.2) translated the minecraft recipe names to ids. The crafting.txt is extended with and minecraft recipe names is possible. Limitations: Only a single recipe is added to the crafting area. Multiple clicks or shift click does not increase the number of builds. Co-authored-by: peterbell10 <peterbell10@live.co.uk> * Address first issues mentioned by @peterbell10 - Some linting - Extract loading of recipe specific protocol mapping into a function - Build `RecipeNameMap` only once - Use `std::optional` - Extract `LoadRecipe` from `Window` * Start to implement new suggestions * Update with suggestions from @peterbell10 * Some minor cleanup * Update protocol packet IDs * Remove unused include * Include header in cmake * Change a vector to integer counter * Change dromedaryCase method names to PascalCase * Address suggestions from @madmaxoft * Read Protocol subdirectories to load recipe books To load all recipebooks iterate over the `Protocol` subdirectories to find mapping files. Co-authored-by: peterbell10 <peterbell10@live.co.uk>
2020-07-14 12:56:42 -04:00
/**
Compares two items for the same type or category. Type of item is defined
via `m_ItemType` and `m_ItemDamage`. Some items (e.g. planks) have the same
`m_ItemType` and the wood kind is defined via `m_ItemDamage`. `-1` is used
as placeholder for all kinds (e.g. all kind of planks).
Items are different when the `ItemType` is different or the `ItemDamage`
is different and unequal -1.
*/
struct sItemCompare
{
bool operator() (const cItem & a_Lhs, const cItem & a_Rhs) const
{
if (a_Lhs.m_ItemType != a_Rhs.m_ItemType)
{
return (a_Lhs.m_ItemType < a_Rhs.m_ItemType);
}
if ((a_Lhs.m_ItemDamage == -1) || (a_Rhs.m_ItemDamage == -1))
{
return false; // -1 is a wildcard, damage of -1 alway compares equal
}
return (a_Lhs.m_ItemDamage < a_Rhs.m_ItemDamage);
}
};
// tolua_begin
2014-05-07 14:43:37 -04:00
int m_RepairCost;
2014-05-07 06:45:20 -04:00
cFireworkItem m_FireworkItem;
cColor m_ItemColor;
};
// tolua_end
/** This class bridges a vector of cItem for safe access via Lua. It checks boundaries for all accesses
Note that this class is zero-indexed!
*/
class cItems // tolua_export
: public std::vector<cItem>
{ // tolua_export
public:
cItems(const cItems &) = default;
cItems(cItems &&) = default;
cItems & operator = (const cItems &) = default;
cItems & operator = (cItems &&) = default;
/** Constructs a new instance containing the specified item. */
cItems(cItem && a_InitialItem);
// tolua_begin
2016-02-05 16:45:45 -05:00
2014-02-11 05:30:11 -05:00
/** Need a Lua-accessible constructor */
cItems(void) {}
2016-02-05 16:45:45 -05:00
cItem * Get (int a_Idx);
void Set (int a_Idx, const cItem & a_Item);
void Add (const cItem & a_Item) {push_back(a_Item); }
void Add (short a_ItemType) { emplace_back(a_ItemType); }
void Add (short a_ItemType, char a_ItemCount) { emplace_back(a_ItemType, a_ItemCount); }
void Delete(int a_Idx);
void Clear (void) {clear(); }
2014-05-12 14:38:52 -04:00
size_t Size (void) const { return size(); }
2014-01-06 16:22:33 -05:00
void Set (int a_Idx, short a_ItemType, char a_ItemCount, short a_ItemDamage);
bool Contains(const cItem & a_Item);
bool ContainsType(const cItem & a_Item);
2014-01-06 16:22:33 -05:00
void Add (short a_ItemType, char a_ItemCount, short a_ItemDamage)
{
emplace_back(a_ItemType, a_ItemCount, a_ItemDamage);
}
2016-02-05 16:45:45 -05:00
/** Adds a copy of all items in a_ItemGrid. */
void AddItemGrid(const cItemGrid & a_ItemGrid);
// tolua_end
} ; // tolua_export
2014-02-11 05:30:11 -05:00
/** Used to store loot probability tables */
class cLootProbab
{
public:
cItem m_Item;
int m_MinAmount;
int m_MaxAmount;
int m_Weight;
} ;