////////////////////////////////////////////////////////////////////////////// // 일반 아이템을 처리한다. ////////////////////////////////////////////////////////////////////////////// void CGRequestRepairHandler::executeNormal (CGRequestRepair* pPacket , Player* pPlayer) throw(ProtocolException , Error) { __BEGIN_TRY __BEGIN_DEBUG_EX #ifdef __GAME_SERVER__ ObjectID_t ITEMOID = pPacket->getObjectID(); Creature* pPC = dynamic_cast<GamePlayer*>(pPlayer)->getCreature(); bool bSlayer = false; bool bVampire = false; bool bOusters = false; Gold_t playerMoney = 0; Price_t repairPrice = 0; Item* pItem = NULL; Slayer* pSlayer = NULL; Vampire* pVampire = NULL; Ousters* pOusters = NULL; int storage = 0; int X = 0; int Y = 0; GCNPCResponse response; // 플레이어가 슬레이어인지 뱀파이어인지 구분. if (pPC->isSlayer()) bSlayer = true; else if (pPC->isVampire()) bVampire = true; else if (pPC->isOusters()) bOusters = true; // 플레이어가 수리하려고 하는 아이템을 가지고 있는지 검사 if (bSlayer) { pSlayer = dynamic_cast<Slayer*>(pPC); playerMoney = pSlayer->getGold(); pItem = pSlayer->findItemOID(ITEMOID, storage, X, Y); } else if (bVampire) { pVampire = dynamic_cast<Vampire*>(pPC); playerMoney = pVampire->getGold(); pItem = pVampire->findItemOID(ITEMOID, storage, X, Y); } else if (bOusters) { pOusters = dynamic_cast<Ousters*>(pPC); playerMoney = pOusters->getGold(); pItem = pOusters->findItemOID(ITEMOID, storage, X, Y); } // 플레이어가 수리하려고 하는 아이템을 가지고 있는지 // 상위에서 검사를 하기 때문에, pItem이 널일리는 없다. // 단, 수리할 수 없는 아이템인지를 검사한다. if (isRepairableItem(pItem) == false) { response.setCode(NPC_RESPONSE_REPAIR_FAIL_ITEM_TYPE); pPlayer->sendPacket(&response); return; } // 이전 내구도를 저장한다. Durability_t oldDurability = pItem->getDurability(); repairPrice = g_pPriceManager->getRepairPrice(pItem); if (repairPrice > playerMoney) { response.setCode(NPC_RESPONSE_REPAIR_FAIL_MONEY); pPlayer->sendPacket(&response); return; } // 수리한다. repairItem(pItem); // 수리한 아이템이 기어창의 아이템이고 이전 내구도가 0 이었다면 정보를 새로 보내줘야한다. if (storage == STORAGE_GEAR && oldDurability == 0 ) { if (bSlayer && pSlayer != NULL ) { pSlayer->initAllStatAndSend(); pSlayer->sendRealWearingInfo(); } else if (bVampire && pVampire != NULL ) { pVampire->initAllStatAndSend(); pVampire->sendRealWearingInfo(); } else if (bOusters && pOusters != NULL ) { pOusters->initAllStatAndSend(); pOusters->sendRealWearingInfo(); } } // 돈을 줄인다. if (bSlayer) { //pSlayer->setGoldEx(playerMoney-repairPrice); // by sigi. 2002.9.4 pSlayer->decreaseGoldEx(repairPrice); //log(LOG_REPAIR_ITEM, pSlayer->getName(), "", pItem->toString()); } else if (bVampire) { // by sigi. 2002.9.4 pVampire->decreaseGoldEx(repairPrice); //log(LOG_REPAIR_ITEM, pVampire->getName(), "", pItem->toString()); } else if (bOusters) { // by sigi. 2002.9.4 pOusters->decreaseGoldEx(repairPrice); //log(LOG_REPAIR_ITEM, pOusters->getName(), "", pItem->toString()); } // 아이템을 수리했다는 정보를 DB에다가 저장해준다. // 단 분명히 STORAGE_STASH가 돌아올 수 있지만, // 보관함에 있는 것을 수리한다는 것은 말이 안 되므로, // 저장하지 않는다. // item저장 최적화. by sigi. 2002.5.17 if (repairPrice>0) { char pField[80]; if (pItem->getItemClass()==Item::ITEM_CLASS_SLAYER_PORTAL_ITEM) { SlayerPortalItem* pSPItem = dynamic_cast<SlayerPortalItem*>(pItem); sprintf(pField, "Charge=%d", pSPItem->getCharge()); } else if (pItem->getItemClass()==Item::ITEM_CLASS_OUSTERS_SUMMON_ITEM) { OustersSummonItem* pOSItem = dynamic_cast<OustersSummonItem*>(pItem); sprintf(pField, "Charge=%d", pOSItem->getCharge()); } else { sprintf(pField, "Durability=%d", pItem->getDurability()); } pItem->tinysave(pField); } /* // 뭐가 됐든.. durability만 바꾸면 된다. // 근데.. ItemObject에 Durability field가 없는 것도 있고 // Charge를 저장해야 하는 것도 있다. // 그래서.. 일단은 모두 다 저장하는 save를 이용하도록 한다. switch (storage) { case STORAGE_INVENTORY: { pItem->save(pPC->getName(), STORAGE_INVENTORY, 0, X, Y); } break; case STORAGE_GEAR: { if (bSlayer) { pItem->save(pSlayer->getName(), STORAGE_GEAR, 0, X, 0); } else { pItem->save(pVampire->getName(), STORAGE_GEAR, 0, X, 0); } } break; default: break; } */ // OK 패킷을 날려준다. response.setCode(NPC_RESPONSE_REPAIR_OK); response.setParameter(playerMoney-repairPrice); pPlayer->sendPacket(&response); #endif __END_DEBUG_EX __END_CATCH }
void CGUseBonusPointHandler::execute (CGUseBonusPoint* pPacket , Player* pPlayer) throw(Error) { __BEGIN_TRY __BEGIN_DEBUG_EX __BEGIN_DEBUG #ifdef __GAME_SERVER__ Assert(pPacket != NULL); Assert(pPlayer != NULL); // 정상적인 상태가 아니라면 리턴 GamePlayer* pGamePlayer = dynamic_cast<GamePlayer*>(pPlayer); if (pGamePlayer->getPlayerStatus() != GPS_NORMAL) return; Creature* pCreature = pGamePlayer->getCreature(); BYTE which = pPacket->getWhich(); Attr_t cur = 0; if (pCreature->isSlayer() ) { Slayer* pSlayer = dynamic_cast<Slayer*>(pCreature); if (pSlayer->getBonus() <= 0 ) { GCUseBonusPointFail failPkt; pPlayer->sendPacket(&failPkt); return; } bool success = false; if (which == INC_INT ) { success = pSlayer->putAdvancedBonusToINT(); } else if (which == INC_STR) { success = pSlayer->putAdvancedBonusToSTR(); } else if (which == INC_DEX) { success = pSlayer->putAdvancedBonusToDEX(); } if (success ) { GCUseBonusPointOK okpkt; pGamePlayer->sendPacket (&okpkt); pSlayer->saveExps(); pSlayer->initAllStatAndSend(); } else { GCUseBonusPointFail failPkt; pPlayer->sendPacket(&failPkt); return; } } else if (pCreature->isVampire() ) { // 보너스 포인트가 없다면 리턴 Vampire* pVampire = dynamic_cast<Vampire*>(pCreature); if (pVampire->getBonus() <= 0) { GCUseBonusPointFail failPkt; pPlayer->sendPacket(&failPkt); return; } VAMPIRE_RECORD oldRecord; // 능력치를 올리기 전에 기존의 능력치를 저장한다. pVampire->getVampireRecord(oldRecord); if (which == INC_INT) { cur = pVampire->getINT(ATTR_BASIC) + 1; pVampire->setINT(cur, ATTR_BASIC); StringStream sst; sst << "INTE = " << (int)cur; pVampire->tinysave(sst.toString()); /* // INT가 증가하면 새로운 기술을 배울 수 있는 가능성이 있다. SkillType_t lastSkill = pVampire->findLastSkill(); // lastSkill의 다음 level의 기술을 찾는다. // 못찾았다면 더이상 배울것이 없다는 것. for(int i = SKILL_BLOOD_DRAIN + 1 ; i < SKILL_MAX; i++) { SkillParentInfo* pParentInfo = g_pSkillParentInfoManager->getSkillParentInfo(i); if (pParentInfo->hasParent(lastSkill))// 찾았다! { SkillInfo* pNewSkillInfo = g_pSkillInfoManager->getSkillInfo(i); if (pNewSkillInfo->getEXP() <= cur && pVampire->hasSkill(i) == NULL) { //cout << "(" << pVampire->getName() << ") can learn new skill >> "; // 새로운 기술을 배울 수 있다. GCLearnSkillReady gcLSR; gcLSR.setSkillDomainType(SKILL_DOMAIN_VAMPIRE); pVampire->getPlayer()->sendPacket(&gcLSR); break; } } } */ //log(LOG_USE_BONUS_POINT, pVampire->getName(), "", "INT"); } else if (which == INC_STR) { cur = pVampire->getSTR(ATTR_BASIC) + 1; pVampire->setSTR(cur, ATTR_BASIC); StringStream sst; sst << "STR = " << (int)cur; pVampire->tinysave(sst.toString()); //log(LOG_USE_BONUS_POINT, pVampire->getName(), "", "STR"); } else if (which == INC_DEX) { cur = pVampire->getDEX(ATTR_BASIC) + 1; pVampire->setDEX(cur, ATTR_BASIC); StringStream sst; sst << "DEX = " << (int)cur; pVampire->tinysave(sst.toString()); //log(LOG_USE_BONUS_POINT, pVampire->getName(), "", "DEX"); } // 바뀐 보너스 포인트를 저장한다. Bonus_t OldBonus = pVampire->getBonus(); pVampire->setBonus(OldBonus - 1); StringStream sst; sst << "Bonus = " << (int)(OldBonus - 1); pVampire->tinysave(sst.toString()); // 능력치가 변화되었으니, stat을 새로 고친다. pVampire->initAllStat(); // 클라이언트의 계산 순서 때문에 생기는 버그로 인하여, // 먼저 인증 패킷을 날려준 후에, 바뀐 능력치에 대한 정보를 보낸다. // 나중에 CGUseBonusPointOK에다 바로 바뀐 능력치에 대한 정보를 // 실어보내도록 해야 한다. // OK 패킷을 보내준다. GCUseBonusPointOK okpkt; pGamePlayer->sendPacket (&okpkt); // 바뀐 능력치에 관한 정보를 보내준다. pVampire->sendModifyInfo(oldRecord); pVampire->sendRealWearingInfo(); } else if (pCreature->isOusters() ) { // 보너스 포인트가 없다면 리턴 Ousters* pOusters = dynamic_cast<Ousters*>(pCreature); if (pOusters->getBonus() <= 0) { GCUseBonusPointFail failPkt; pPlayer->sendPacket(&failPkt); return; } OUSTERS_RECORD oldRecord; // 능력치를 올리기 전에 기존의 능력치를 저장한다. pOusters->getOustersRecord(oldRecord); if (which == INC_INT) { cur = pOusters->getINT(ATTR_BASIC) + 1; pOusters->setINT(cur, ATTR_BASIC); StringStream sst; sst << "INTE = " << (int)cur; pOusters->tinysave(sst.toString()); } else if (which == INC_STR) { cur = pOusters->getSTR(ATTR_BASIC) + 1; pOusters->setSTR(cur, ATTR_BASIC); StringStream sst; sst << "STR = " << (int)cur; pOusters->tinysave(sst.toString()); } else if (which == INC_DEX) { cur = pOusters->getDEX(ATTR_BASIC) + 1; pOusters->setDEX(cur, ATTR_BASIC); StringStream sst; sst << "DEX = " << (int)cur; pOusters->tinysave(sst.toString()); } // 바뀐 보너스 포인트를 저장한다. Bonus_t OldBonus = pOusters->getBonus(); pOusters->setBonus(OldBonus - 1); StringStream sst; sst << "Bonus = " << (int)(OldBonus - 1); pOusters->tinysave(sst.toString()); // 능력치가 변화되었으니, stat을 새로 고친다. pOusters->initAllStat(); // OK 패킷을 보내준다. GCUseBonusPointOK okpkt; pGamePlayer->sendPacket(&okpkt); // 바뀐 능력치에 관한 정보를 보내준다. pOusters->sendModifyInfo(oldRecord); pOusters->sendRealWearingInfo(); } else { GCUseBonusPointFail failPkt; pPlayer->sendPacket(&failPkt); return; } #endif // __GAME_SERVER__ __END_DEBUG __END_DEBUG_EX __END_CATCH }
////////////////////////////////////////////////////////////////////////////// // 모든 아이템 수리하기 ////////////////////////////////////////////////////////////////////////////// void CGRequestRepairHandler::executeAll(CGRequestRepair* pPacket , Player* pPlayer) throw(ProtocolException , Error) { __BEGIN_TRY __BEGIN_DEBUG_EX #ifdef __GAME_SERVER__ Creature* pPC = dynamic_cast<GamePlayer*>(pPlayer)->getCreature(); Price_t repairPrice = 0; GCNPCResponse response; bool bSendRealWearingInfo = false; if (pPC->isSlayer()) { Slayer* pSlayer = dynamic_cast<Slayer*>(pPC); // 모든 아이템을 합한 수리비를 계산한다. for (int i=0; i<Slayer::WEAR_MAX; i++) { Item* pItem = pSlayer->getWearItem((Slayer::WearPart)i); if (pItem != NULL) { if (i == Slayer::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 오른손이고, 현재 들고 있는 무기가 양손 무기라면... // 수리 가격에 포함시킬 필요가 없다. } else { repairPrice += g_pPriceManager->getRepairPrice(pItem); } } } // 돈이 모자라다면 리턴한다. if (pSlayer->getGold() < repairPrice) { response.setCode(NPC_RESPONSE_REPAIR_FAIL_MONEY); pPlayer->sendPacket(&response); return; } // 각각의 아이템을 수리하고, DB에 저장한다. char pField[80]; for (int i=0; i<Slayer::WEAR_MAX; i++) { Item* pItem = pSlayer->getWearItem((Slayer::WearPart)i); if (pItem != NULL) { if (i == Slayer::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 오른손이고, 현재 들고 있는 무기가 양손 무기라면... // 수리할 필요가 없다. } else if (isRepairableItem(pItem ) ) { Durability_t oldDurability = pItem->getDurability(); repairItem(pItem); if (pItem->getDurability() != oldDurability) { // DB 쿼리를 줄이기 위해서 // 내구도의 변화가 생긴 경우에만 세이브한다. //pItem->save(pSlayer->getName(), STORAGE_GEAR, 0, i, 0); // item저장 최적화. by sigi. 2002.5.13 sprintf(pField, "Durability=%d", pItem->getDurability()); pItem->tinysave(pField); } if (oldDurability == 0 ) bSendRealWearingInfo = true; } } } // 돈을 줄이고... //pSlayer->setGoldEx(pSlayer->getGold() - repairPrice); // by sigi.2002.9.4 pSlayer->decreaseGoldEx(repairPrice); // 로그를 남긴다. //log(LOG_REPAIR_ITEM, pSlayer->getName(), "", "ALL"); // OK 패킷을 날려준다. response.setCode(NPC_RESPONSE_REPAIR_OK); response.setParameter(pSlayer->getGold()); pPlayer->sendPacket(&response); } else if (pPC->isVampire()) { Vampire* pVampire = dynamic_cast<Vampire*>(pPC); // 모든 아이템을 합한 수리비를 계산한다. for (int i=0; i<Vampire::VAMPIRE_WEAR_MAX; i++) { Item* pItem = pVampire->getWearItem((Vampire::WearPart)i); if (pItem != NULL) { if (i == Vampire::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 양손무기는 한쪽만 수리한다. } else { repairPrice += g_pPriceManager->getRepairPrice(pItem); } } } // 돈이 모자라다면 리턴한다. if (pVampire->getGold() < repairPrice) { response.setCode(NPC_RESPONSE_REPAIR_FAIL_MONEY); pPlayer->sendPacket(&response); return; } // 각각의 아이템을 수리하고, DB에 저장한다. char pField[80]; for (int i=0; i<Vampire::VAMPIRE_WEAR_MAX; i++) { Item* pItem = pVampire->getWearItem((Vampire::WearPart)i); if (pItem != NULL) { if (i == Vampire::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 양손무기는 한쪽만 수리한다. } else { Durability_t oldDurability = pItem->getDurability(); repairItem(pItem); if (pItem->getDurability() != oldDurability) { // DB 쿼리를 줄이기 위해서 // 내구도의 변화가 생긴 경우에만 세이브한다. //pItem->save(pVampire->getName(), STORAGE_GEAR, 0, i, 0); // item저장 최적화. by sigi. 2002.5.13 sprintf(pField, "Durability=%d", pItem->getDurability()); pItem->tinysave(pField); } if (oldDurability == 0 ) bSendRealWearingInfo = true; } } } // 돈을 줄이고... //pVampire->setGoldEx(pVampire->getGold() - repairPrice); // by sigi.2002.9.4 pVampire->decreaseGoldEx(repairPrice); // 로그를 남긴다. //log(LOG_REPAIR_ITEM, pVampire->getName(), "", "ALL"); // OK 패킷을 날려준다. response.setCode(NPC_RESPONSE_REPAIR_OK); response.setParameter(pVampire->getGold()); pPlayer->sendPacket(&response); } else if (pPC->isOusters()) { Ousters* pOusters = dynamic_cast<Ousters*>(pPC); // 모든 아이템을 합한 수리비를 계산한다. for (int i=0; i<Ousters::OUSTERS_WEAR_MAX; i++) { Item* pItem = pOusters->getWearItem((Ousters::WearPart)i); if (pItem != NULL) { if (i == Ousters::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 양손무기는 한쪽만 수리한다. } else { repairPrice += g_pPriceManager->getRepairPrice(pItem); } } } // 돈이 모자라다면 리턴한다. if (pOusters->getGold() < repairPrice) { response.setCode(NPC_RESPONSE_REPAIR_FAIL_MONEY); pPlayer->sendPacket(&response); return; } // 각각의 아이템을 수리하고, DB에 저장한다. char pField[80]; for (int i=0; i<Ousters::OUSTERS_WEAR_MAX; i++) { Item* pItem = pOusters->getWearItem((Ousters::WearPart)i); if (pItem != NULL) { if (i == Ousters::WEAR_RIGHTHAND && isTwohandWeapon(pItem)) { // 양손무기는 한쪽만 수리한다. } else { Durability_t oldDurability = pItem->getDurability(); repairItem(pItem); if (pItem->getDurability() != oldDurability) { // DB 쿼리를 줄이기 위해서 // 내구도의 변화가 생긴 경우에만 세이브한다. //pItem->save(pOusters->getName(), STORAGE_GEAR, 0, i, 0); // item저장 최적화. by sigi. 2002.5.13 sprintf(pField, "Durability=%d", pItem->getDurability()); pItem->tinysave(pField); } if (oldDurability == 0 ) bSendRealWearingInfo = true; } } } // 돈을 줄이고... //pOusters->setGoldEx(pOusters->getGold() - repairPrice); // by sigi.2002.9.4 pOusters->decreaseGoldEx(repairPrice); // 로그를 남긴다. //log(LOG_REPAIR_ITEM, pOusters->getName(), "", "ALL"); // OK 패킷을 날려준다. response.setCode(NPC_RESPONSE_REPAIR_OK); response.setParameter(pOusters->getGold()); pPlayer->sendPacket(&response); } if (bSendRealWearingInfo ) { if (pPC->isSlayer() ) { Slayer* pSlayer = dynamic_cast<Slayer*>(pPC); Assert(pSlayer != NULL); pSlayer->initAllStatAndSend(); pSlayer->sendRealWearingInfo(); } else if (pPC->isVampire() ) { Vampire* pVampire = dynamic_cast<Vampire*>(pPC); Assert(pVampire != NULL); pVampire->initAllStatAndSend(); pVampire->sendRealWearingInfo(); } else if (pPC->isOusters() ) { Ousters* pOusters = dynamic_cast<Ousters*>(pPC); Assert(pOusters != NULL); pOusters->initAllStatAndSend(); pOusters->sendRealWearingInfo(); } } #endif __END_DEBUG_EX __END_CATCH }