TownBuildingInstance.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. /*
  2. * TownBuildingInstance.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 "TownBuildingInstance.h"
  12. #include "CGTownInstance.h"
  13. #include "../IGameCallback.h"
  14. #include "../mapObjects/CGHeroInstance.h"
  15. #include "../entities/building/CBuilding.h"
  16. #include <vstd/RNG.h>
  17. VCMI_LIB_NAMESPACE_BEGIN
  18. TownBuildingInstance::TownBuildingInstance(IGameCallback * cb)
  19. : IObjectInterface(cb)
  20. , town(nullptr)
  21. {}
  22. TownBuildingInstance::TownBuildingInstance(CGTownInstance * town, const BuildingID & index)
  23. : IObjectInterface(town->cb)
  24. , town(town)
  25. , bID(index)
  26. {}
  27. PlayerColor TownBuildingInstance::getOwner() const
  28. {
  29. return town->getOwner();
  30. }
  31. MapObjectID TownBuildingInstance::getObjGroupIndex() const
  32. {
  33. return -1;
  34. }
  35. MapObjectSubID TownBuildingInstance::getObjTypeIndex() const
  36. {
  37. return 0;
  38. }
  39. const IOwnableObject * TownBuildingInstance::asOwnable() const
  40. {
  41. return nullptr;
  42. }
  43. int3 TownBuildingInstance::visitablePos() const
  44. {
  45. return town->visitablePos();
  46. }
  47. int3 TownBuildingInstance::anchorPos() const
  48. {
  49. return town->anchorPos();
  50. }
  51. TownRewardableBuildingInstance::TownRewardableBuildingInstance(IGameCallback *cb)
  52. : TownBuildingInstance(cb)
  53. {}
  54. TownRewardableBuildingInstance::TownRewardableBuildingInstance(CGTownInstance * town, const BuildingID & index, vstd::RNG & rand)
  55. : TownBuildingInstance(town, index)
  56. {
  57. initObj(rand);
  58. }
  59. void TownRewardableBuildingInstance::initObj(vstd::RNG & rand)
  60. {
  61. assert(town && town->getTown());
  62. configuration = generateConfiguration(rand);
  63. }
  64. Rewardable::Configuration TownRewardableBuildingInstance::generateConfiguration(vstd::RNG & rand) const
  65. {
  66. Rewardable::Configuration result;
  67. auto building = town->getTown()->buildings.at(getBuildingType());
  68. building->rewardableObjectInfo.configureObject(result, rand, cb);
  69. for(auto & rewardInfo : result.info)
  70. {
  71. for (auto & bonus : rewardInfo.reward.bonuses)
  72. {
  73. bonus.source = BonusSource::TOWN_STRUCTURE;
  74. bonus.sid = BonusSourceID(building->getUniqueTypeID());
  75. }
  76. }
  77. return result;
  78. }
  79. void TownRewardableBuildingInstance::newTurn(vstd::RNG & rand) const
  80. {
  81. if (configuration.resetParameters.period != 0 && cb->getDate(Date::DAY) > 1 && ((cb->getDate(Date::DAY)-1) % configuration.resetParameters.period) == 0)
  82. {
  83. auto newConfiguration = generateConfiguration(rand);
  84. cb->setRewardableObjectConfiguration(town->id, getBuildingType(), newConfiguration);
  85. if(configuration.resetParameters.visitors)
  86. {
  87. cb->setObjPropertyValue(town->id, ObjProperty::STRUCTURE_CLEAR_VISITORS, getBuildingType());
  88. }
  89. }
  90. }
  91. void TownRewardableBuildingInstance::setProperty(ObjProperty what, ObjPropertyID identifier)
  92. {
  93. switch (what)
  94. {
  95. case ObjProperty::VISITORS:
  96. visitors.insert(identifier.as<ObjectInstanceID>());
  97. break;
  98. case ObjProperty::STRUCTURE_CLEAR_VISITORS:
  99. visitors.clear();
  100. break;
  101. case ObjProperty::REWARD_SELECT:
  102. selectedReward = identifier.getNum();
  103. break;
  104. }
  105. }
  106. void TownRewardableBuildingInstance::heroLevelUpDone(const CGHeroInstance *hero) const
  107. {
  108. grantRewardAfterLevelup(configuration.info.at(selectedReward), town, hero);
  109. }
  110. void TownRewardableBuildingInstance::blockingDialogAnswered(const CGHeroInstance *hero, int32_t answer) const
  111. {
  112. onBlockingDialogAnswered(hero, answer);
  113. }
  114. void TownRewardableBuildingInstance::grantReward(ui32 rewardID, const CGHeroInstance * hero) const
  115. {
  116. grantRewardBeforeLevelup(configuration.info.at(rewardID), hero);
  117. // hero is not blocked by levelup dialog - grant remainder immediately
  118. if(!cb->isVisitCoveredByAnotherQuery(town, hero))
  119. {
  120. grantRewardAfterLevelup(configuration.info.at(rewardID), town, hero);
  121. }
  122. }
  123. bool TownRewardableBuildingInstance::wasVisited(const CGHeroInstance * contextHero) const
  124. {
  125. return wasVisitedBefore(contextHero);
  126. }
  127. bool TownRewardableBuildingInstance::wasVisitedBefore(const CGHeroInstance * contextHero) const
  128. {
  129. switch (configuration.visitMode)
  130. {
  131. case Rewardable::VISIT_UNLIMITED:
  132. return false;
  133. case Rewardable::VISIT_ONCE:
  134. return !visitors.empty();
  135. case Rewardable::VISIT_PLAYER:
  136. return false; //not supported
  137. case Rewardable::VISIT_BONUS:
  138. {
  139. const auto building = town->getTown()->buildings.at(getBuildingType());
  140. return contextHero->hasBonusFrom(BonusSource::TOWN_STRUCTURE, BonusSourceID(building->getUniqueTypeID()));
  141. }
  142. case Rewardable::VISIT_HERO:
  143. return visitors.find(contextHero->id) != visitors.end();
  144. case Rewardable::VISIT_LIMITER:
  145. return configuration.visitLimiter.heroAllowed(contextHero);
  146. default:
  147. return false;
  148. }
  149. }
  150. void TownRewardableBuildingInstance::onHeroVisit(const CGHeroInstance *h) const
  151. {
  152. assert(town->hasBuilt(getBuildingType()));
  153. if(town->hasBuilt(getBuildingType()))
  154. doHeroVisit(h);
  155. }
  156. const IObjectInterface * TownRewardableBuildingInstance::getObject() const
  157. {
  158. return this;
  159. }
  160. bool TownRewardableBuildingInstance::wasVisited(PlayerColor player) const
  161. {
  162. switch (configuration.visitMode)
  163. {
  164. case Rewardable::VISIT_UNLIMITED:
  165. case Rewardable::VISIT_BONUS:
  166. case Rewardable::VISIT_HERO:
  167. case Rewardable::VISIT_LIMITER:
  168. return false;
  169. case Rewardable::VISIT_ONCE:
  170. case Rewardable::VISIT_PLAYER:
  171. return !visitors.empty();
  172. default:
  173. return false;
  174. }
  175. }
  176. void TownRewardableBuildingInstance::markAsVisited(const CGHeroInstance * hero) const
  177. {
  178. town->addHeroToStructureVisitors(hero, getBuildingType());
  179. }
  180. void TownRewardableBuildingInstance::markAsScouted(const CGHeroInstance * hero) const
  181. {
  182. // no-op - town building is always 'scouted' by owner
  183. }
  184. VCMI_LIB_NAMESPACE_END