I have recently been working a lot on a project for a system that represents an inventory and item usage in a game, with one of the important features is that it should be reusable in multiple projects. I think I managed to do this quite well, keeping the required interface as minimum as possible. I have come to a point where I think the project is finished, but I would like to get feedback on where I can still improve the code. It's really long, so I'm not sure if I should post everything here. I will just post the main Inventory header here, and the rest of the code can be found on github. If I do need to copy all code here, let me know.
Inventory.h
#pragma once
#include "IItem.h"
#include "ItemDispatcher.h"
#include <memory>
#include <functional>
#include <string>
#include <unordered_map>
#include <string_view>
#include <type_traits>
#include <utility>
template<unsigned int MAX_SIZE, typename GameObjTy = temp::GameObject, typename ItemTy = IItem>
class Inventory
{
private:
class Traits //class to simulate namespace inside class
{
public:
/*HasUseMethod type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasUseMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasUseMethodHelper<_Ty, std::void_t<decltype(std::declval<_Ty>().use(std::declval<ItemDispatcher<GameObjTy>&>()))>> : std::true_type
{
};
template<typename _Ty>
struct HasUseMethodT : HasUseMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasUseMethod = typename HasUseMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasUseMethodV = HasUseMethod<_Ty>::value;
/*HasEquippabmeMethod type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasEquippableMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasEquippableMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().equippable()), bool >)>> : std::true_type
{
};
template<typename _Ty>
struct HasEquippableMethodT : HasEquippableMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasEquippableMethod = typename HasEquippableMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasEquippableMethodV = HasEquippableMethod<_Ty>::value;
/*HasIsEquipped type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasIsEquippedMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasIsEquippedMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().is_equipped()), bool >)>> : std::true_type
{
};
template<typename _Ty>
struct HasIsEquippedMethodT : HasIsEquippedMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasIsEquippedMethod = typename HasIsEquippedMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasIsEquippedMethodV = HasIsEquippedMethod<_Ty>::value;
/*HasSetEquip type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasSetEquipMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasSetEquipMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().set_equip(std::declval<bool>())), void >)>>
: std::true_type
{
};
template<typename _Ty>
struct HasSetEquipMethodT : HasSetEquipMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasSetEquipMethod = typename HasSetEquipMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasSetEquipMethodV = HasSetEquipMethod<_Ty>::value;
/*HasUnequipMethod type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasUnequipMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasUnequipMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().unequip(std::declval<GameObjTy*>())), void >)>>
: std::true_type
{
};
template<typename _Ty>
struct HasUnequipMethodT : HasUnequipMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasUnequipMethod = typename HasUnequipMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasUnequipMethodV = HasUnequipMethod<_Ty>::value;
/*HasReusableMethod type trait*/
template<typename _Ty, typename = std::void_t<>>
struct HasReusableMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasReusableMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().reusable()), bool >)>>
: std::true_type
{
};
template<typename _Ty>
struct HasReusableMethodT : HasReusableMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasReusableMethod = typename HasReusableMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasReusableMethodV = HasReusableMethod<_Ty>::value;
template<typename _Ty, typename = std::void_t<>>
struct HasStackableMethodHelper : std::false_type
{
};
template<typename _Ty>
struct HasStackableMethodHelper<_Ty,
std::void_t<decltype(std::is_same_v<decltype(std::declval<_Ty>().stackable()), bool>)>>
: std::true_type
{
};
template<typename _Ty>
struct HasStackableMethodT : HasStackableMethodHelper<_Ty>::type
{
};
template<typename _Ty> using HasStackableMethod = typename HasStackableMethodT<_Ty>::type;
template<typename _Ty> static constexpr bool HasStackableMethodV = HasStackableMethod<_Ty>::value;
template<typename _Ty>
struct IsValidItemT
{
static constexpr bool value =
HasEquippableMethodV<_Ty>
&& HasUseMethodV<_Ty>
&& HasIsEquippedMethodV<_Ty>
&& HasSetEquipMethodV<_Ty>
&& HasEquippableMethodV<_Ty>
&& HasReusableMethodV<_Ty>
&& HasStackableMethodV<_Ty>;
};
template<typename _Ty> using IsValidItem = typename IsValidItemT<_Ty>::type;
template<typename _Ty> static constexpr bool IsValidItemV = IsValidItemT<_Ty>::value;
};
public:
static_assert(Traits::IsValidItemV<ItemTy>, "Item type is invalid. It should provide methods listed in documentation");
class Exception
{
private:
std::string msg;
public:
explicit inline Exception(std::string_view error) : msg {error} {}
inline std::string_view what() { return msg; }
};
using game_object_type = GameObjTy;
using item_type = ItemTy;
using item_pointer = std::unique_ptr<item_type>;
using game_object_pointer = game_object_type*;
using inventory_type = std::unordered_map<std::string, std::pair<item_pointer, unsigned int>>;
using iterator = typename inventory_type::iterator;
using const_iterator = typename inventory_type::const_iterator;
using size_type = typename inventory_type::size_type;
explicit Inventory(game_object_pointer owner);
Inventory(Inventory const& other) = delete;
Inventory(Inventory&& other);
Inventory& operator=(Inventory const& other) = delete;
Inventory& operator=(Inventory&& other);
inventory_type const& contents() const;
inventory_type& contents();
/*Adds a new item, stacked on top of an item with the same ID. The id parameter will be used to access the item*/
template<typename ItemT>
void addItem(std::string_view id);
/*constructs a new item and adds it to the inventory. The id parameter will be used to access the item*/
template<typename ItemT, typename... Args>
void emplaceItem(std::string_view id, Args... args);
void useItem(std::string_view name, game_object_pointer target = nullptr);
/*The iterators invalidate when a new Item is added to the inventory*/
iterator getItem(std::string_view name);
/*The iterators invalidate when a new Item is added to the inventory*/
const_iterator getItem(std::string_view name) const;
void removeItem(std::string_view name);
iterator begin();
iterator end();
const_iterator cbegin() const;
const_iterator cend() const;
size_type max_size() const;
size_type size() const;
/*Merges inventory A with this inventory. Leaves other empty, unless this inventory is full, in which case the leftover
elements will be deleted*/ //#TODO: leftover elements are left in old inventory?
template<unsigned int N>
void merge(Inventory<N, GameObjTy, ItemTy>& other);
template<unsigned int N>
bool merge_fits(Inventory<N, GameObjTy, ItemTy> const& other);
/*Transfers item with name parameter into the inventory specified in destination, unless destination does not have enough
*space left*/
template<unsigned int N>
void transfer(Inventory<N, GameObjTy, ItemTy>& destination, std::string_view name);
bool empty() const;
bool full() const;
void setOwner(game_object_pointer owner);
game_object_pointer getOwner() const;
void clear();
unsigned int getItemCount(std::string_view id) const;
private:
size_type m_size = 0;
inventory_type m_items { MAX_SIZE };
game_object_pointer m_owner { nullptr };
//these functions are private so you cannot accidentally pass an invalid iterator to one of these, causing undefined behavior
void useItem(iterator pos, game_object_pointer target = nullptr);
void removeItem(iterator pos);
inline Exception InventoryFullException() const { return Exception {"Inventory is full"}; }
inline Exception InvalidItemTypeException() const { return Exception {"Item type must be derived from Inventory::ItemTy, which defaults to IItem"}; }
inline Exception InvalidItemException() const { return Exception { "Invalid item name" }; }
inline Exception InvalidStackException() const { return Exception {"Tried to stack a non-stackable item"}; }
inline Exception InvalidIDException() const { return Exception {"ID not found in inventory"}; }
};
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
Inventory<MAX_SIZE, GameObjTy, ItemTy>::Inventory(game_object_pointer owner) : m_owner(owner)
{
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
Inventory<MAX_SIZE, GameObjTy, ItemTy>::Inventory(Inventory&& other) : m_owner(std::move(other.m_owner)), m_items(std::move(other.m_items)), m_size(other.m_size)
{
other.m_owner = nullptr;
other.m_items = inventory_type {};
other.m_size = 0;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
Inventory<MAX_SIZE, GameObjTy, ItemTy>& Inventory<MAX_SIZE, GameObjTy, ItemTy>::operator=(Inventory<MAX_SIZE, GameObjTy, ItemTy>&& other)
{
// #WARNING: Self assignment check is missing
m_owner = other.m_owner;
m_items = std::move(other.m_items);
m_size = other.m_size;
other.m_owner = nullptr;
other.m_items = inventory_type {};
other.m_size = 0;
return *this;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::inventory_type const& Inventory<MAX_SIZE, GameObjTy, ItemTy>::contents() const
{
return m_items;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::inventory_type& Inventory<MAX_SIZE, GameObjTy, ItemTy>::contents()
{
return m_items;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::begin()
{
return m_items.begin();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::end()
{
return m_items.end();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::const_iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::cbegin() const
{
return m_items.cbegin();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::const_iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::cend() const
{
return m_items.cend();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::getItem(std::string_view name)
{
return m_items.find(name.data());
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::const_iterator Inventory<MAX_SIZE, GameObjTy, ItemTy>::getItem(std::string_view name) const
{
return m_items.find(name.data());
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::useItem(std::string_view name, game_object_pointer target)
{
useItem(getItem(name), target);
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
template<typename ItemT>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::addItem(std::string_view id)
{
if constexpr (!std::is_base_of_v<item_type, ItemT>)
throw InvalidItemTypeException();
if (size() >= MAX_SIZE)
{
throw InventoryFullException();
}
if (m_items.find(id.data()) != m_items.end()) //if we already own this item, increment the count
{
if (!m_items[id.data()].first->stackable())
throw InvalidStackException();
m_items[id.data()].second += 1; //increment count
m_size += 1;
}
else
{
throw InvalidIDException();
}
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
template<typename ItemT, typename... Args>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::emplaceItem(std::string_view id, Args... args)
{
if constexpr (!std::is_base_of_v<item_type, ItemT>)
throw InvalidItemTypeException();
if (size() >= MAX_SIZE)
{
throw InventoryFullException();
}
m_items[id.data()] = std::make_pair(std::make_unique<ItemT>(std::forward<Args>(args)...), 1);
m_size += 1;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::useItem(iterator pos, game_object_pointer target)
{
if (pos == m_items.end()) throw InvalidItemException();
//use the item
ItemDispatcher<GameObjTy> dispatcher { target };
auto& it = *pos;
auto& itemPair = it.second;
auto& item = itemPair.first;
if (item->equippable())
{
dispatcher.setTarget(m_owner);
if (!item->is_equipped())
{
item->set_equip(true);
item->use(dispatcher);
}
else
{
item->set_equip(false);
item->use(dispatcher);
}
return;
}
else
{
dispatcher.setTarget(target);
item->use(dispatcher); //dispatcher.target == target, see construction above
}
if (!item->reusable())
{
if (!item->stackable())
removeItem(pos);
else
{
if (itemPair.second > 1) itemPair.second -= 1; //decrement count if we have more than 1
else removeItem(pos);
}
m_size -= 1;
}
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::removeItem(iterator pos)
{
m_items.erase(pos);
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::removeItem(std::string_view name)
{
removeItem(getItem(name));
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::setOwner(game_object_pointer owner)
{
m_owner = owner;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::game_object_pointer Inventory<MAX_SIZE, GameObjTy, ItemTy>::getOwner() const
{
return m_owner;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::size_type Inventory<MAX_SIZE, GameObjTy, ItemTy>::max_size() const
{
return MAX_SIZE;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
typename Inventory<MAX_SIZE, GameObjTy, ItemTy>::size_type Inventory<MAX_SIZE, GameObjTy, ItemTy>::size() const
{
return m_size;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
template<unsigned int N>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::merge(Inventory<N, GameObjTy, ItemTy>& other)
{
if (!merge_fits(other))
throw InventoryFullException();
for (auto& it = other.begin(); it != other.end(); std::advance(it, 1))
{
this->m_items[it->first] = std::move(it->second);
}
other.clear();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::clear()
{
m_size = 0;
m_items.clear();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
template<unsigned int N>
bool Inventory<MAX_SIZE, GameObjTy, ItemTy>::merge_fits(Inventory<N, GameObjTy, ItemTy> const& other)
{
return !(full() || other.size() + this->size() >= max_size());
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
template<unsigned int N>
void Inventory<MAX_SIZE, GameObjTy, ItemTy>::transfer(Inventory<N, GameObjTy, ItemTy>& destination, std::string_view name)
{
if (destination.full())
return;
auto& it = getItem(name);
auto& item = (*it).second;
destination.contents()[name.data()] = std::move(item);
m_items.erase(it);
m_size -= 1;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
bool Inventory<MAX_SIZE, GameObjTy, ItemTy>::empty() const
{
return size() <= 0;
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
bool Inventory<MAX_SIZE, GameObjTy, ItemTy>::full() const
{
return max_size() <= size();
}
template<unsigned int MAX_SIZE, typename GameObjTy, typename ItemTy>
unsigned int Inventory<MAX_SIZE, GameObjTy, ItemTy>::getItemCount(std::string_view id) const
{
if (m_items.find(id.data()) == m_items.end()) throw InvalidItemException();
return m_items.at(id.data()).second;
}
I will also add a main.cpp test file just to demonstrate how this can be used
main.cpp
#include "DamagePotion.h"
#include "Inventory.h"
#include "HealPotion.h"
#include "Sword.h"
#include "ItemUtil.h"
std::ostream& operator<<(std::ostream& out, ItemID const& id)
{
if (id == ItemID::DAMAGE_POTION)
{
out << "ID_DAMAGE_POTION";
}
else if (id == ItemID::DEFAULT_ITEM) out << "ID_DEFAULT_ITEM";
else if (id == ItemID::HEAL_POTION) out << "ID_HEAL_POTION";
else if (id == ItemID::SWORD) out << "ID_SWORD";
return out;
}
//Replace temp::GameObject class with the GameObject class used by your game
class Player : public temp::GameObject
{
public:
Player() : temp::GameObject(200)
{
try
{
m_inventory.emplaceItem<DamagePotion>("Damage Potion I", 50);
m_inventory.emplaceItem<HealPotion>("Heal Potion I", 70);
m_inventory.emplaceItem<Sword>("Sword I", 20);
std::cout << "Inventory contents after adding base items:\n";
for (auto const& it : m_inventory.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
m_inventory.useItem("Damage Potion I", this);
m_inventory.useItem("Heal Potion I", this);
m_inventory.useItem("Sword I");
std::cout << "Inventory contents after using base items:\n";
for (auto const& it : m_inventory.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
m_inventory.useItem("Sword I"); // will unequip Sword I
std::cout << "Inventory contents after unequipping Sword I:\n";
for (auto const& it : m_inventory.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
chest.emplaceItem<DamagePotion>("CDmgPot", 100);
chest.emplaceItem<HealPotion>("CHealPot", 200);
std::cout << "Chest contents after adding base items:\n";
for (auto const& it : chest.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
m_inventory.merge(chest);
std::cout << "Chest contents after merging with inventory:\n";
for (auto const& it : chest.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
std::cout << "Inventory contents after merging with chest:\n";
for (auto const& it : m_inventory.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
chest.emplaceItem<Sword>("CSword", 50);
std::cout << "Chest contents after adding CSword:\n";
for (auto const& it : chest.contents()) std::cout << it.second.first ->id() << "\n";
std::cout << "\n";
chest.transfer(m_inventory, "CSword");
std::cout << "Inventory contents after transferring CSword from chest:\n";
for (auto const& it : m_inventory.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
chest.emplaceItem<Sword>("CSword", 20);
std::cout << "Chest contents after adding a CSword:\n";
for (auto const& it : chest.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
chest.removeItem("CSword");
std::cout << "Chest contents after removing a CSword:\n";
for (auto const& it :chest.contents()) std::cout << it.second.first->id() << "\n";
std::cout << "\n";
}
catch (std::runtime_error e)
{
std::cerr << e.what();
}
}
private:
Inventory<200> m_inventory { this };
Inventory<5> chest { this };
};
struct InvalidItem
{
};
struct InvalidEquippable
{
void equippable()
{
}
};
class TestClass : public temp::GameObject
{
public:
TestClass() : temp::GameObject(50)
{
try
{
Inventory<100> inv { this };
inv.emplaceItem<ItemType<ItemID::DAMAGE_POTION>::type>("P", 20);
inv.addItem<DamagePotion>("P");
std::cout << ItemName(ItemID::HEAL_POTION) << ' ' << ItemId<Sword>() << '\n';
std::cout << inv.getItemCount("P") << '\n';
inv.useItem("P", this);
inv.useItem("P", this);
std::cout << getHealth() << '\n';
std::cout << inv.getItemCount("P") << '\n';
}
catch (Inventory<100>::Exception e)
{
std::cout << e.what() << "\n";
}
}
};
int main()
{
Player p;
// IItem* base_ptr;
// Inventory<200> {nullptr};
//Fails to compile due to traits
// Inventory<100, temp::GameObject, InvalidItem> {nullptr};
// Inventory<100, temp::GameObject, InvalidEquippable> {nullptr};
// base_ptr = new DamagePotion(20);
// temp::GameObject target { 100 };
/*
std::cout << "Using Item: \n";
std::cout << "Name:\t" << base_ptr->name() << "\n";
std::cout << "ID:\t" << base_ptr->id() << "\n";
base_ptr->use(&target);
*/
std::cin.get();
}
Note: I have just read a meta post that says that it is not allowed to post GitHub links. I am not sure what I should do in this case, as I mostly want the Inventory header reviewed. The files in the github repo are only for when there really is more information required.
EDIT: as per request in the comments, here are some further explanations:
Architecture
As I created this project with portability across other projects in mind, so as a kind of library, the architecture is designed to be as simple to use as possible. All Items will be inherited from a base IItem class, or any other class that supports the functions in the Inventory::Traits class. The inventory is responsible for managing the resources of every item, and for creating them. So when adding an item you call
inventory.emplaceItem<SomePotion>("Potion", 50); //the 50 is passed to SomePotion's constructor
MAX_SIZE
MAX_SIZE is a template parameter due to an unlucky design choice at the very beginning. Since I did not want to do lots of allocations for items, I wanted to have the size fixed. This is probably completely unneccesary, but annoying to remove now.
Item usage
Since the Inventory is responsible for using the items (by calling useItem("name", target_ptr);), I needed some way of doing that. I chose for the Visitor Pattern, which I implemented with the ItemDispatcher class.
Data Structure
The items are stored in a std::unordered_map<std::string, std::pair<std::unique_ptr<Item>, unsigned>>; I will break down why I chose for this. First of all, I wanted the items to be accessible in some way. Iterators were clumsy, because they would invalidate when an item is added to the inventory. I still kept the begin() and end() functions for if you would want to iterate through an inventory in a loop. So I chose for an undoredered map, with string indices so you can give every item its own string ID. The items are stored as a pair of [Item, Count], since I wanted to be able to stack items where possible (Item.stackable() == true).
Reusability
About the reusable in multiple projects. With that I mean that if I, at some point decide to make an RPG game, and the character needs an inventory, that I can just use this exact Inventory class without any modifications. If I then later want to make another, completely different game that also features an Inventory, or some other way of storing items, I can reuse this again.
Techniques
The Traits class is used as a wrapper for the SFINAE type traits that I use to detect if an eventual custom Item type supports all required functions, set for example like:
Inventory<200, MyCoolGameObject, MyCoolItem> inventory;
Traits is a class because I can't do
class X
{
namespace Y
{
};
};
Traitsclass), and summarize the architecture. \$\endgroup\$use(),equippable(),is_equipped(),set_equip(),unequip(), etc)? \$\endgroup\$_Tyis an _Ugly format name. Those are reserved for use by the library implementation. See timsong-cpp.github.io/cppwp/n4659/lex.name#3 . Writing your this in a way that mimics what you see in the compiler-supplied headers is exactly wrong! \$\endgroup\$