#include "stdafx.h" #include #include #include #include "Item.h" #include "ItemFactory.h" #include "ItemContainer.h" namespace Item { class CNullItem : public CItem, public CSingleton { public: virtual ~CNullItem() { } private: CNullItem(const ItemInfo& itemInfo) : CItem(0, itemInfo) { } static ItemInfo ms_thisiteminfo; static CNullItem ms_this; }; }; Item::ItemInfo Item::CNullItem::ms_thisiteminfo(0xFFFF); Item::CNullItem Item::CNullItem::ms_this(Item::CNullItem::ms_thisiteminfo); // ---------------------------------------------------------------------------------------- // CItemContainer Item::CItemContainer::CItemContainer() : m_dwCID(0), m_nMaxSize(0), m_lpNullItem(&CNullItem::GetInstance()), m_lppItems(NULL), m_usFlags(0) { } Item::CItemContainer::~CItemContainer() { Destroy(); } bool Item::CItemContainer::Initialize(unsigned long dwCID, unsigned short nMaxSize) { m_dwCID = dwCID; m_nMaxSize = nMaxSize; m_lppItems = new CItem*[m_nMaxSize]; if(NULL != m_lppItems) { std::fill_n(m_lppItems, m_nMaxSize, reinterpret_cast(NULL)); return true; } return false; } void Item::CItemContainer::Destroy() { if(NULL != m_lppItems) { CItem** lppItem = m_lppItems; CItem** lppItemPastEnd = m_lppItems + m_nMaxSize; for(; lppItem != lppItemPastEnd; ++lppItem) { CItem* lpItem = *lppItem; if(NULL != lpItem && m_lpNullItem != lpItem) { delete lpItem; } } delete [] m_lppItems; m_lppItems = NULL; } } void Item::CItemContainer::DumpItemInfo() { const int MIN_BUFFER = 32; const int MAX_BUFFER = 256; char szUID[MIN_BUFFER]; char szBuffer[MAX_BUFFER]; CItem** lppItem = m_lppItems; CItem** lppItemPastEnd = m_lppItems + m_nMaxSize; for(; lppItem != lppItemPastEnd; ++lppItem) { const CItem* lpItem = *lppItem; if(NULL != lpItem && m_lpNullItem != lpItem) { Math::Convert::Hex64ToStr(szUID, lpItem->GetUID()); const ItemPos itemPos = lpItem->GetPos(); const DetailData& detailData = lpItem->GetItemInfo().m_DetailData; unsigned char cX, cY, cTab; itemPos.GetPos(cX, cY, cTab); _snprintf(szBuffer, MAX_BUFFER, "CID:0x%08x ÀÇ ¾ÆÀÌÅÛÀÔ´Ï´Ù. UID: %s Á¾·ùID : %5d," " ÁÂÇ¥:(%d - %2d,%2d,%2d), Å©±â:(%2d, %2d), ÇöÀç °³¼ö(³»±¸µµ):%2d", m_dwCID, szUID, lpItem->GetPrototypeID(), itemPos.m_cIndex, cX, cY, cTab, detailData.m_cXSize, detailData.m_cXSize, lpItem->GetNumOrDurability()); ERRLOG1(g_Log, "%s", szBuffer); } } } bool Item::CItemContainer::SerializeOut(char* szItemBuffer_Out, unsigned long& dwBufferSize_InOut) const { CItem** lppItem = m_lppItems; CItem** lppItemPastEnd = m_lppItems + m_nMaxSize; size_t nRemainBufferSize = static_cast(dwBufferSize_InOut); size_t nWritten = 0; for(; lppItem != lppItemPastEnd; ++lppItem) { CItem* lpItem = *lppItem; if(NULL != lpItem && m_lpNullItem != lpItem) { size_t nItemSize = nRemainBufferSize; if(!lpItem->SerializeOut(szItemBuffer_Out + nWritten, nItemSize)) { return false; } nWritten += nItemSize; nRemainBufferSize -= nItemSize; } } dwBufferSize_InOut = static_cast(nWritten); return true; } // ---------------------------------------------------------------------------------------- // CArrayContainer Item::CArrayContainer::CArrayContainer() { } Item::CArrayContainer::~CArrayContainer() { } bool Item::CArrayContainer::Initialize(unsigned long dwCID, unsigned char nXSize, unsigned char nYSize, unsigned char nTabNum) { m_dwCID = dwCID; m_nXSize = nXSize; m_nYSize = nYSize; m_nTabNum = nTabNum; m_nSizePerTab = nXSize * nYSize; return CItemContainer::Initialize(dwCID, nXSize * nYSize * nTabNum); } Item::CItem* Item::CArrayContainer::GetItem(Item::ItemPos itemPos) { unsigned char cX, cY, cTab; itemPos.GetPos(cX, cY, cTab); if(m_nXSize <= cX || m_nYSize <= cY || m_nTabNum <= cTab) { ERRLOG7(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ À§Ä¡°¡ ¿Ã¹Ù¸£Áö ¾Ê½À´Ï´Ù. X:%d/%d, Y:%d/%d, Tab:%d/%d", m_dwCID, cX, m_nXSize, cY, m_nYSize, cTab, m_nTabNum); return NULL; } CItem* lpItem = m_lppItems[m_nSizePerTab * cTab + m_nXSize * cY + cX]; return (lpItem != m_lpNullItem) ? lpItem : NULL; } bool Item::CArrayContainer::SetItem(Item::ItemPos itemPos, Item::CItem* lpItem) { if(NULL == lpItem) { ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛÀÌ NULLÀÔ´Ï´Ù.", m_dwCID); return false; } unsigned char cX, cY, cTab; itemPos.GetPos(cX, cY, cTab); unsigned char cXSize = lpItem->GetItemInfo().m_DetailData.m_cXSize; unsigned char cYSize = lpItem->GetItemInfo().m_DetailData.m_cYSize; if(m_nXSize < static_cast(cX + cXSize) || m_nYSize < static_cast(cY + cYSize) || m_nTabNum <= cTab) { ERRLOG9(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ À§Ä¡ ¹× Å©±â°¡ ¿Ã¹Ù¸£Áö ¾Ê½À´Ï´Ù. " "X:%d/%d, Y:%d/%d, Tab:%d/%d, XÅ©±â:%d, YÅ©±â:%d", m_dwCID, cX, m_nXSize, cY, m_nYSize, cTab, m_nTabNum, cXSize, cYSize); return false; } CItem** lppItemYIndexPastEnd = m_lppItems + m_nSizePerTab * cTab + m_nXSize * (cY + cYSize); CItem **lppItemYIndex, **lppItemXIndex, **lppItemXIndexPastEnd; for(lppItemYIndex = m_lppItems + m_nSizePerTab * cTab + cY * m_nXSize; lppItemYIndex != lppItemYIndexPastEnd; lppItemYIndex += m_nXSize) { lppItemXIndexPastEnd = lppItemYIndex + cX + cXSize; for(lppItemXIndex = lppItemYIndex + cX; lppItemXIndex != lppItemXIndexPastEnd; ++lppItemXIndex) { if(0 != *lppItemXIndex) { ERRLOG4(g_Log, "CID:0x%08x (%d,%d,%d) ÀÌ¹Ì ±× Àå¼Ò¿¡ ¾ÆÀÌÅÛÀÌ ÀÖ½À´Ï´Ù.", m_dwCID, cX, cY, cTab); return false; } } } for(lppItemYIndex = m_lppItems + m_nSizePerTab * cTab + cY * m_nXSize; lppItemYIndex != lppItemYIndexPastEnd; lppItemYIndex += m_nXSize) { lppItemXIndexPastEnd = lppItemYIndex + cX + cXSize; for(lppItemXIndex = lppItemYIndex + cX; lppItemXIndex != lppItemXIndexPastEnd; ++lppItemXIndex) { *lppItemXIndex = m_lpNullItem; } } m_lppItems[m_nSizePerTab * cTab + m_nXSize * cY + cX] = lpItem; lpItem->MoveItem(itemPos); return true; } bool Item::CArrayContainer::RemoveItem(Item::ItemPos itemPos) { unsigned char cX, cY, cTab; itemPos.GetPos(cX, cY, cTab); if(m_nXSize <= cX || m_nYSize <= cY || m_nTabNum <= cTab) { ERRLOG7(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ À§Ä¡ ¹× Å©±â°¡ ¿Ã¹Ù¸£Áö ¾Ê½À´Ï´Ù. X:%d/%d, Y:%d/%d, Tab:%d/%d", m_dwCID, cX, m_nXSize, cY, m_nYSize, cTab, m_nTabNum); return false; } CItem* lpItem = m_lppItems[m_nSizePerTab * cTab + m_nXSize * cY + cX]; if(NULL == lpItem) { ERRLOG4(g_Log, "CID:0x%08x ±× À§Ä¡¿¡ ¾ÆÀÌÅÛÀÌ ¾ø½À´Ï´Ù. X:%d, Y:%d, Tab:%d", m_dwCID, cX, cY, cTab); return false; } unsigned char cXSize = lpItem->GetItemInfo().m_DetailData.m_cXSize; unsigned char cYSize = lpItem->GetItemInfo().m_DetailData.m_cYSize; if(m_nXSize < static_cast(cX + cXSize) || m_nYSize < static_cast(cY + cYSize)) { ERRLOG9(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ À§Ä¡ ¹× Å©±â°¡ ¿Ã¹Ù¸£Áö ¾Ê½À´Ï´Ù. " "X:%d/%d, Y:%d/%d, Tab:%d/%d, XÅ©±â:%d, YÅ©±â:%d", m_dwCID, cX, m_nXSize, cY, m_nYSize, cTab, m_nTabNum, cXSize, cYSize); return false; } CItem** lppItemYIndexPastEnd = m_lppItems + m_nSizePerTab * cTab + m_nXSize * (cY + cYSize); CItem **lppItemYIndex, **lppItemXIndex, **lppItemXIndexPastEnd; for(lppItemYIndex = m_lppItems + m_nSizePerTab * cTab + cY * m_nXSize; lppItemYIndex != lppItemYIndexPastEnd; lppItemYIndex += m_nXSize) { lppItemXIndexPastEnd = lppItemYIndex + cX + cXSize; for(lppItemXIndex = lppItemYIndex + cX; lppItemXIndex != lppItemXIndexPastEnd; ++lppItemXIndex) { *lppItemXIndex = NULL; } } return true; } void Item::CArrayContainer::DumpItemInfo() { const int MAX_LINE = 128; char szLine[MAX_LINE]; int nBufferPos = 0; ERRLOG1(g_Log, "CID:0x%08x ÀÇ ¾ÆÀÌÅÛ ·Î±×¸¦ ½ÃÀÛÇÕ´Ï´Ù.", m_dwCID); for(unsigned short nTab = 0; nTab < m_nTabNum; ++nTab) { const unsigned short usTabPos = m_nSizePerTab * nTab; ERRLOG2(g_Log, "CID:0x%08x ÀÇ ÅÇ %dÀÇ ¾ÆÀÌÅÛÀÔ´Ï´Ù.", m_dwCID, nTab); for(unsigned short nHeight = 0; nHeight < m_nYSize; ++nHeight) { const unsigned short nHeightPos = m_nXSize * nHeight + usTabPos; nBufferPos = _snprintf(szLine, MAX_LINE, "CID:0x%08x ÀÇ ¾ÆÀÌÅÛÀÔ´Ï´Ù. ", m_dwCID); for(unsigned short nWidth = 0; nWidth < m_nXSize; ++nWidth) { const CItem* lpItem = m_lppItems[nHeightPos + nWidth]; nBufferPos += _snprintf(szLine + nBufferPos, MAX_LINE, " %5d ", ((0 != lpItem) ? lpItem->GetPrototypeID() : 0)); } ERRLOG0(g_Log, szLine); } } CItemContainer::DumpItemInfo(); } bool Item::CArrayContainer::SerializeIn(const char* szItemBuffer_In, unsigned long dwBufferSize_In) { size_t nBufferSize = static_cast(dwBufferSize_In); size_t nUsed = 0; Item::CItemFactory& ItemFactory = CItemFactory::GetInstance(); while(nBufferSize > 0) { size_t nItemSize = nBufferSize; Item::CItem* lpItem = ItemFactory.CreateItem(szItemBuffer_In + nUsed, nItemSize); if(NULL == lpItem) { nItemSize = reinterpret_cast(szItemBuffer_In + nUsed)->m_cItemSize; ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ »ý¼º¿¡ ½ÇÆÐÇß½À´Ï´Ù.", m_dwCID); } else if(!CArrayContainer::SetItem(lpItem->GetPos(), lpItem)) { nItemSize = reinterpret_cast(szItemBuffer_In + nUsed)->m_cItemSize; ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ ³õ±â¸¦ ½ÇÆÐÇß½À´Ï´Ù.", m_dwCID); delete lpItem; } nUsed += nItemSize; nBufferSize -= nItemSize; } return true; } // ---------------------------------------------------------------------------------------- // CListContainer Item::CListContainer::CListContainer() { } Item::CListContainer::~CListContainer() { } Item::CItem* Item::CListContainer::GetItem(Item::ItemPos itemPos) { if(itemPos.m_cIndex < m_nMaxSize) { CItem* lpItem = m_lppItems[itemPos.m_cIndex]; return (lpItem != m_lpNullItem) ? lpItem : NULL; } return NULL; } bool Item::CListContainer::SetItem(Item::ItemPos itemPos, Item::CItem* lpItem) { if(itemPos.m_cIndex < m_nMaxSize) { if(NULL == m_lppItems[itemPos.m_cIndex]) { m_lppItems[itemPos.m_cIndex] = lpItem; lpItem->MoveItem(itemPos); return true; } else { ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛÀÌ ÀÌ¹Ì Á¸ÀçÇÕ´Ï´Ù.", m_dwCID); } } return false; } bool Item::CListContainer::RemoveItem(Item::ItemPos itemPos) { if(itemPos.m_cIndex < m_nMaxSize) { if(NULL != m_lppItems[itemPos.m_cIndex]) { m_lppItems[itemPos.m_cIndex] = NULL; return true; } else { ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛÀÌ Á¸ÀçÇÏÁö ¾Ê½À´Ï´Ù.", m_dwCID); } } return false; } bool Item::CListContainer::SerializeIn(const char* szItemBuffer_In, unsigned long dwBufferSize_In) { size_t nBufferSize = static_cast(dwBufferSize_In); size_t nUsed = 0; Item::CItemFactory& ItemFactory = CItemFactory::GetInstance(); while(nBufferSize > 0) { size_t nItemSize = nBufferSize; Item::CItem* lpItem = ItemFactory.CreateItem(szItemBuffer_In + nUsed, nItemSize); if(NULL == lpItem) { nItemSize = reinterpret_cast(szItemBuffer_In + nUsed)->m_cItemSize; ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ »ý¼º¿¡ ½ÇÆÐÇß½À´Ï´Ù.", m_dwCID); } else if(!CListContainer::SetItem(lpItem->GetPos(), lpItem)) { nItemSize = reinterpret_cast(szItemBuffer_In + nUsed)->m_cItemSize; ERRLOG1(g_Log, "CID:0x%08x ¾ÆÀÌÅÛ ³õ±â¸¦ ½ÇÆÐÇß½À´Ï´Ù.", m_dwCID); delete lpItem; } nUsed += nItemSize; nBufferSize -= nItemSize; } return true; }