The open source repository for the action RPG game in development by Sig Productions titled 'Adventures in Lestoria'! https://forums.lestoria.net
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
AdventuresInLestoria/Crawler/Item.h

312 lines
11 KiB

#pragma region License
/*
License (OLC-3)
~~~~~~~~~~~~~~~
1 year ago
Copyright 2018 - 2023 OneLoneCoder.com
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
1. Redistributions or derivations of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions or derivative works in binary form must reproduce the above
copyright notice. This list of conditions and the following disclaimer must be
reproduced in the documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its contributors may
be used to endorse or promote products derived from this software without specific
prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE.
Portions of this software are copyright © 2023 The FreeType
Project (www.freetype.org). Please see LICENSE_FT.txt for more information.
All rights reserved.
*/
#pragma endregion
#pragma once
#include <string>
#include <functional>
#include <map>
#include "olcPixelGameEngine.h"
#include "olcUTIL_DataFile.h"
#include "AttributableStat.h"
#include "BitwiseEnum.h"
#include <optional>
#include "Merchant.h"
#include "CraftingRequirement.h"
class Crawler;
class ItemInfo;
class ItemProps;
using IT=std::string;
using ITCategory=std::string;
using ItemScript=std::function<bool(Crawler*,ItemProps)>;
enum class EquipSlot{
HELMET= 0b0000'0001,
WEAPON= 0b0000'0010,
ARMOR= 0b0000'0100,
GLOVES= 0b0000'1000,
PANTS= 0b0001'0000,
SHOES= 0b0010'0000,
RING1= 0b0100'0000,
RING2= 0b1000'0000,
NONE= 0b0000'0000,
};
enum class CompactText{
COMPACT,
NON_COMPACT
};
using enum CompactText;
struct Stats:ItemAttributable{
static safemap<int,float>maxDamageReductionTable;
static void InitializeDamageReductionTable();
public:
inline static const float&GetDamageReductionPct(int defense){
return maxDamageReductionTable.at(std::clamp(defense,0,1000));
}
inline Stats&operator+=(Stats&rhs){
for(auto&[key,value]:ItemAttribute::attributes){
A(value)+=rhs.get_readOnly(value);
}
return *this;
};
inline Stats&operator+=(ItemAttributable&rhs){
for(auto&[key,value]:ItemAttribute::attributes){
A(value)+=rhs.get_readOnly(value);
}
return *this;
};
auto begin()const{
return attributes.begin();
}
auto end()const{
return attributes.end();
}
const std::string GetStatsString(CompactText compact=NON_COMPACT)const;
};
class Stats;
class CraftingRequirement;
struct EnhancementLevelInfo{
const Stats&stats;
const CraftingRequirement&craftingRequirement;
EnhancementLevelInfo(const Stats&stats,const CraftingRequirement&craftingRequirement);
};
struct EnhancementInfo{
private:
std::vector<Stats>enhancementStats;
std::vector<CraftingRequirement>craftingRequirements;
public:
void SetAttribute(int enhanceLevel,ItemAttribute attribute,float value);
void SetCraftingRequirements(const int enhanceLevel,const std::vector<Item>&requiredItems,const uint32_t goldCost);
const EnhancementLevelInfo operator[](int level)const;
const size_t size()const;
};
//You cannot create instances of this class, access sets from ItemSet::sets and add the appropriate set bonuses.
class ItemSet{
friend class ItemInfo;
static std::map<std::string,ItemSet>sets;
std::array<Stats,8>setBonuses;
std::string name;
std::vector<std::pair<int,Stats>>setBonusBreakpoints;
public:
//NO CONSTRUCTOR REQUIRED!
//Specify the piece count(1-8) for a set bonus to be applied.
static void AddSetBonus(std::string setName,int pieceCount,Stats&bonuses);
//Gets a set bonus based on number of pieces (1-8)
const Stats&operator[](int setPieces)const;
const std::string&GetSetName()const;
bool operator<(const ItemSet&rhs)const{
return name<rhs.name;
}
const std::vector<std::pair<int,Stats>>&GetSetBonusBreakpoints()const;
};
class Item{
friend class Inventory;
friend class Crawler;
friend class Menu;
friend void Merchant::PurchaseItem(IT item,uint32_t amt);
friend void Merchant::SellItem(IT item,uint32_t amt);
private:
//The amount in the current item stack.
uint32_t amt;
uint8_t enhancementLevel;
ItemInfo*it;
void SetAmt(uint32_t newAmt);
static int IsBlankStaticCallCounter;
const bool _IsBlank()const;
public:
Item();
Item(uint32_t amt,IT item,uint8_t enhancementLevel=0);
uint32_t Amt()const;
//Use this for places where the item name is used for item tracking. NOT for drawing. Hooks directly into item info's base item name.
const std::string&ActualName()const;
//Use for places where the item name is actually displayed. Provides modified text that shows additional information like enhancement levels.
const std::string DisplayName()const;
const std::string Description(CompactText compact=COMPACT)const;
const ITCategory Category()const;
const EquipSlot GetEquipSlot()const;
const::Decal*const Decal()const;
const Stats GetStats()const;
const ItemScript&OnUseAction()const;
const float CastTime()const;
const bool UseDuringCast()const; //When true, the item is activated during the cast instead of after the cast.
const float CooldownTime()const;
//Use ISBLANK macro instead!! This should not be called directly!!
const bool IsBlank()const;
const uint8_t EnhancementLevel()const;
void EnhanceItem();
static std::shared_ptr<Item>BLANK;
const std::optional<const::ItemSet*const>ItemSet()const;
const bool IsEquippable()const;
const uint32_t BuyValue()const;
const uint32_t SellValue()const;
const bool CanBeSold()const;
const bool CanBePurchased()const;
//Use ISBLANK macro instead!! This should not be called directly!!
static bool IsBlank(std::shared_ptr<Item>item);
//Use ISBLANK macro instead!! This should not be called directly!!
static bool IsBlank(std::weak_ptr<Item>item);
friend const bool operator==(std::shared_ptr<Item>lhs,std::shared_ptr<Item>rhs){return lhs->it==rhs->it;};
friend const bool operator==(std::shared_ptr<Item>lhs,const IT&rhs){return lhs->ActualName()==rhs;};
friend const bool operator==(std::weak_ptr<Item>lhs,std::weak_ptr<Item>rhs){return !lhs.expired()&&!rhs.expired()&&lhs.lock()->it==rhs.lock()->it;};
friend const bool operator==(std::weak_ptr<Item>lhs,const IT&rhs){return !lhs.expired()&&lhs.lock()->ActualName()==rhs;};
friend const bool operator==(const IT&lhs,std::weak_ptr<Item>rhs){return operator==(rhs,lhs);};
friend const bool operator==(const IT&lhs,std::shared_ptr<Item>rhs){return operator==(rhs,lhs);};
};
class Inventory{
friend class ItemInfo;
friend class Item;
public:
static void AddItem(IT it,uint32_t amt=1,bool monsterDrop=false);
//Returns the actual amount available in your main inventory.
static uint32_t GetItemCount(IT it);
static std::shared_ptr<Item>CopyItem(IT it);
static std::weak_ptr<Item>GetItem(IT it);
//Auto-executes its use function and removes the amt specified from the inventory. Multiple amounts will cause the item to execute its useFunc multiple times.
static bool UseItem(IT it,uint32_t amt=1);
static bool RemoveItem(IT it,ITCategory inventory,uint32_t amt = 1);
static bool RemoveItem(IT it,uint32_t amt=1);
static std::vector<std::shared_ptr<Item>>&get(ITCategory itemCategory);
static const std::weak_ptr<Item>GetInventorySlot(ITCategory itemCategory,size_t slot);
static void Clear(ITCategory itemCategory);
static void EquipItem(const std::weak_ptr<Item>it,EquipSlot slot);
static void UnequipItem(EquipSlot slot);
static EquipSlot GetSlotEquippedIn(const std::weak_ptr<Item>it);
static std::weak_ptr<Item>GetEquip(EquipSlot slot);
static const std::map<ItemSet,int>GetEquippedItemSets();
static bool SwapItems(ITCategory itemCategory,uint32_t slot1,uint32_t slot2);
//Makes sure this is a valid category. Will error out if it doesn't exist! Use for ERROR HANDLING!
static bool ValidateItemCategory(ITCategory itemCategory){
get(itemCategory);
return true;
}
private:
static void InsertIntoSortedInv(IT item);
static void InsertIntoStageInventoryCategory(IT item,uint32_t amt,bool monsterDrop);
static bool ExecuteAction(IT item);
static std::map<IT,std::shared_ptr<Item>>_inventory;
static std::map<EquipSlot,std::weak_ptr<Item>>equipment;
//Only contains "1" of every item, as this is a map to index items and not the actual storage of items!
static std::map<ITCategory,std::vector<std::shared_ptr<Item>>>sortedInv;
};
class ItemProps{
friend class ItemInfo;
utils::datafile*scriptProps;
utils::datafile*customProps;
public:
ItemProps(utils::datafile*scriptProps,utils::datafile*customProps);
int GetIntProp(const std::string&prop,size_t index=0)const;
float GetFloatProp(const std::string&prop,size_t index=0)const;
std::string GetStringProp(const std::string&prop,size_t index=0)const;
const uint32_t PropCount(const std::string&prop)const;
};
class ItemInfo{
friend class Inventory;
std::string name;
std::string description;
std::string category;
float castTime=0;
float cooldownTime=0;
EquipSlot slot;
EnhancementInfo enhancement;
::Decal*img;
std::string set="";
//Returns true if the item can be used, false otherwise
std::string useFunc="";
//Custom properties for this specific item's script.
static utils::datafile NOPROPS;
ItemProps customProps;
uint32_t buyValue=0;
uint32_t sellValue=0;
//If true, this item's action is activated at the beginning of the cast instead of after the cast completes.
bool useDuringCast=false;
private:
static void InitializeScripts();
static void InitializeSets();
static std::map<std::string,EquipSlot>nameToEquipSlot;
public:
static void InitializeItems();
ItemInfo();
const std::string&Name()const;
const std::string&Description()const;
const ITCategory Category()const;
const::Decal*const Decal()const;
/*
For the useFunc, return true if the item can be used, false otherwise.
*/
const ItemScript&OnUseAction()const;
const Stats&GetStats(int enhancementLevel)const;
const float CastTime()const;
const float CooldownTime()const;
const EquipSlot Slot()const;
const std::optional<const::ItemSet*const>ItemSet()const;
ItemInfo&operator[](const IT&item);
const uint32_t GetBuyValue()const;
const uint32_t GetSellValue()const;
const bool CanBeSold()const;
const bool CanBePurchased()const;
const bool UseDuringCast()const;
};
class ItemOverlay{
ItemInfo it;
float timer=0;
float xOffset=0;
float width=0; //How wide the entire label is.
static std::vector<ItemOverlay>items;
ItemOverlay(ItemInfo item);
public:
static void AddToItemOverlay(const ItemInfo&it);
static void Update();
static void Draw();
void ResetTimer();
};
#define ISBLANK(itemRef) Item::IsBlank(itemRef)