CArtifactInstance.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. /*
  2. * CArtifactInstance.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "CArtifactInstance.h"
  12. #include "ArtifactUtils.h"
  13. #include "CArtHandler.h"
  14. #include "networkPacks/ArtifactLocation.h"
  15. VCMI_LIB_NAMESPACE_BEGIN
  16. void CCombinedArtifactInstance::addPart(CArtifactInstance * art, const ArtifactPosition & slot)
  17. {
  18. auto artInst = static_cast<CArtifactInstance*>(this);
  19. assert(vstd::contains_if(artInst->artType->getConstituents(),
  20. [=](const CArtifact * partType)
  21. {
  22. return partType->getId() == art->getTypeId();
  23. }));
  24. assert(art->getParentNodes().size() == 1 && art->getParentNodes().front() == art->artType);
  25. partsInfo.emplace_back(art, slot);
  26. artInst->attachTo(*art);
  27. }
  28. bool CCombinedArtifactInstance::isPart(const CArtifactInstance * supposedPart) const
  29. {
  30. if(supposedPart == this)
  31. return true;
  32. for(const PartInfo & constituent : partsInfo)
  33. {
  34. if(constituent.art == supposedPart)
  35. return true;
  36. }
  37. return false;
  38. }
  39. const std::vector<CCombinedArtifactInstance::PartInfo> & CCombinedArtifactInstance::getPartsInfo() const
  40. {
  41. return partsInfo;
  42. }
  43. void CCombinedArtifactInstance::addPlacementMap(CArtifactSet::ArtPlacementMap & placementMap)
  44. {
  45. if(!placementMap.empty())
  46. for(auto & part : partsInfo)
  47. {
  48. assert(placementMap.find(part.art) != placementMap.end());
  49. part.slot = placementMap.at(part.art);
  50. }
  51. }
  52. SpellID CScrollArtifactInstance::getScrollSpellID() const
  53. {
  54. auto artInst = static_cast<const CArtifactInstance*>(this);
  55. const auto bonus = artInst->getBonusLocalFirst(Selector::type()(BonusType::SPELL));
  56. if(!bonus)
  57. {
  58. logMod->warn("Warning: %s doesn't bear any spell!", artInst->nodeName());
  59. return SpellID::NONE;
  60. }
  61. return bonus->subtype.as<SpellID>();
  62. }
  63. void CGrowingArtifactInstance::growingUp()
  64. {
  65. auto artInst = static_cast<CArtifactInstance*>(this);
  66. if(artInst->artType->isGrowing())
  67. {
  68. auto bonus = std::make_shared<Bonus>();
  69. bonus->type = BonusType::LEVEL_COUNTER;
  70. bonus->val = 1;
  71. bonus->duration = BonusDuration::COMMANDER_KILLED;
  72. artInst->accumulateBonus(bonus);
  73. for(const auto & bonus : artInst->artType->getBonusesPerLevel())
  74. {
  75. // Every n levels
  76. if(artInst->valOfBonuses(BonusType::LEVEL_COUNTER) % bonus.first == 0)
  77. {
  78. artInst->accumulateBonus(std::make_shared<Bonus>(bonus.second));
  79. }
  80. }
  81. for(const auto & bonus : artInst->artType->getThresholdBonuses())
  82. {
  83. // At n level
  84. if(artInst->valOfBonuses(BonusType::LEVEL_COUNTER) == bonus.first)
  85. {
  86. artInst->addNewBonus(std::make_shared<Bonus>(bonus.second));
  87. }
  88. }
  89. }
  90. }
  91. void CArtifactInstance::init()
  92. {
  93. // Artifact to be randomized
  94. id = static_cast<ArtifactInstanceID>(ArtifactID::NONE);
  95. setNodeType(ARTIFACT_INSTANCE);
  96. }
  97. CArtifactInstance::CArtifactInstance(CArtifact * art)
  98. {
  99. init();
  100. setType(art);
  101. }
  102. CArtifactInstance::CArtifactInstance()
  103. {
  104. init();
  105. }
  106. void CArtifactInstance::setType(CArtifact * art)
  107. {
  108. artType = art;
  109. attachTo(*art);
  110. }
  111. std::string CArtifactInstance::nodeName() const
  112. {
  113. return "Artifact instance of " + (artType ? artType->getJsonKey() : std::string("uninitialized")) + " type";
  114. }
  115. std::string CArtifactInstance::getDescription() const
  116. {
  117. std::string text = artType->getDescriptionTranslated();
  118. if(artType->isScroll())
  119. ArtifactUtils::insertScrrollSpellName(text, getScrollSpellID());
  120. return text;
  121. }
  122. ArtifactID CArtifactInstance::getTypeId() const
  123. {
  124. return artType->getId();
  125. }
  126. ArtifactInstanceID CArtifactInstance::getId() const
  127. {
  128. return id;
  129. }
  130. void CArtifactInstance::setId(ArtifactInstanceID id)
  131. {
  132. this->id = id;
  133. }
  134. bool CArtifactInstance::canBePutAt(const CArtifactSet * artSet, ArtifactPosition slot, bool assumeDestRemoved) const
  135. {
  136. return artType->canBePutAt(artSet, slot, assumeDestRemoved);
  137. }
  138. bool CArtifactInstance::isCombined() const
  139. {
  140. return artType->isCombined();
  141. }
  142. void CArtifactInstance::putAt(CArtifactSet & set, const ArtifactPosition slot)
  143. {
  144. auto placementMap = set.putArtifact(slot, this);
  145. addPlacementMap(placementMap);
  146. }
  147. void CArtifactInstance::removeFrom(CArtifactSet & set, const ArtifactPosition slot)
  148. {
  149. set.removeArtifact(slot);
  150. for(auto & part : partsInfo)
  151. {
  152. if(part.slot != ArtifactPosition::PRE_FIRST)
  153. part.slot = ArtifactPosition::PRE_FIRST;
  154. }
  155. }
  156. void CArtifactInstance::move(CArtifactSet & srcSet, const ArtifactPosition srcSlot, CArtifactSet & dstSet, const ArtifactPosition dstSlot)
  157. {
  158. removeFrom(srcSet, srcSlot);
  159. putAt(dstSet, dstSlot);
  160. }
  161. void CArtifactInstance::deserializationFix()
  162. {
  163. setType(artType);
  164. for(PartInfo & part : partsInfo)
  165. attachTo(*part.art);
  166. }
  167. VCMI_LIB_NAMESPACE_END