AssetGenerator.cpp 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. /*
  2. * AssetGenerator.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 "AssetGenerator.h"
  12. #include "../gui/CGuiHandler.h"
  13. #include "../render/IImage.h"
  14. #include "../render/IImageLoader.h"
  15. #include "../render/Canvas.h"
  16. #include "../render/ColorFilter.h"
  17. #include "../render/IRenderHandler.h"
  18. #include "../lib/filesystem/Filesystem.h"
  19. void AssetGenerator::generateAll()
  20. {
  21. createBigSpellBook();
  22. createAdventureOptionsCleanBackground();
  23. for (int i = 0; i < PlayerColor::PLAYER_LIMIT_I; ++i)
  24. createPlayerColoredBackground(PlayerColor(i));
  25. createCombatUnitNumberWindow();
  26. }
  27. void AssetGenerator::createAdventureOptionsCleanBackground()
  28. {
  29. std::string filename = "data/AdventureOptionsBackgroundClear.png";
  30. if(CResourceHandler::get()->existsResource(ResourcePath(filename))) // overridden by mod, no generation
  31. return;
  32. if(!CResourceHandler::get("local")->createResource(filename))
  33. return;
  34. ResourcePath savePath(filename, EResType::IMAGE);
  35. auto locator = ImageLocator(ImagePath::builtin("ADVOPTBK"));
  36. locator.scalingFactor = 1;
  37. std::shared_ptr<IImage> img = GH.renderHandler().loadImage(locator, EImageBlitMode::OPAQUE);
  38. Canvas canvas = Canvas(Point(575, 585), CanvasScalingPolicy::IGNORE);
  39. canvas.draw(img, Point(0, 0), Rect(0, 0, 575, 585));
  40. canvas.draw(img, Point(54, 121), Rect(54, 123, 335, 1));
  41. canvas.draw(img, Point(158, 84), Rect(156, 84, 2, 37));
  42. canvas.draw(img, Point(234, 84), Rect(232, 84, 2, 37));
  43. canvas.draw(img, Point(310, 84), Rect(308, 84, 2, 37));
  44. canvas.draw(img, Point(53, 567), Rect(53, 520, 339, 3));
  45. canvas.draw(img, Point(53, 520), Rect(53, 264, 339, 47));
  46. std::shared_ptr<IImage> image = GH.renderHandler().createImage(canvas.getInternalSurface());
  47. image->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePath));
  48. }
  49. void AssetGenerator::createBigSpellBook()
  50. {
  51. std::string filename = "data/SpellBookLarge.png";
  52. if(CResourceHandler::get()->existsResource(ResourcePath(filename))) // overridden by mod, no generation
  53. return;
  54. if(!CResourceHandler::get("local")->createResource(filename))
  55. return;
  56. ResourcePath savePath(filename, EResType::IMAGE);
  57. auto locator = ImageLocator(ImagePath::builtin("SpelBack"));
  58. locator.scalingFactor = 1;
  59. std::shared_ptr<IImage> img = GH.renderHandler().loadImage(locator, EImageBlitMode::OPAQUE);
  60. Canvas canvas = Canvas(Point(800, 600), CanvasScalingPolicy::IGNORE);
  61. // edges
  62. canvas.draw(img, Point(0, 0), Rect(15, 38, 90, 45));
  63. canvas.draw(img, Point(0, 460), Rect(15, 400, 90, 141));
  64. canvas.draw(img, Point(705, 0), Rect(509, 38, 95, 45));
  65. canvas.draw(img, Point(705, 460), Rect(509, 400, 95, 141));
  66. // left / right
  67. Canvas tmp1 = Canvas(Point(90, 355 - 45), CanvasScalingPolicy::IGNORE);
  68. tmp1.draw(img, Point(0, 0), Rect(15, 38 + 45, 90, 355 - 45));
  69. canvas.drawScaled(tmp1, Point(0, 45), Point(90, 415));
  70. Canvas tmp2 = Canvas(Point(95, 355 - 45), CanvasScalingPolicy::IGNORE);
  71. tmp2.draw(img, Point(0, 0), Rect(509, 38 + 45, 95, 355 - 45));
  72. canvas.drawScaled(tmp2, Point(705, 45), Point(95, 415));
  73. // top / bottom
  74. Canvas tmp3 = Canvas(Point(409, 45), CanvasScalingPolicy::IGNORE);
  75. tmp3.draw(img, Point(0, 0), Rect(100, 38, 409, 45));
  76. canvas.drawScaled(tmp3, Point(90, 0), Point(615, 45));
  77. Canvas tmp4 = Canvas(Point(409, 141), CanvasScalingPolicy::IGNORE);
  78. tmp4.draw(img, Point(0, 0), Rect(100, 400, 409, 141));
  79. canvas.drawScaled(tmp4, Point(90, 460), Point(615, 141));
  80. // middle
  81. Canvas tmp5 = Canvas(Point(409, 141), CanvasScalingPolicy::IGNORE);
  82. tmp5.draw(img, Point(0, 0), Rect(100, 38 + 45, 509 - 15, 400 - 38));
  83. canvas.drawScaled(tmp5, Point(90, 45), Point(615, 415));
  84. // carpet
  85. Canvas tmp6 = Canvas(Point(590, 59), CanvasScalingPolicy::IGNORE);
  86. tmp6.draw(img, Point(0, 0), Rect(15, 484, 590, 59));
  87. canvas.drawScaled(tmp6, Point(0, 545), Point(800, 59));
  88. // remove bookmarks
  89. for (int i = 0; i < 56; i++)
  90. canvas.draw(Canvas(canvas, Rect(i < 30 ? 268 : 327, 464, 1, 46)), Point(269 + i, 464));
  91. for (int i = 0; i < 56; i++)
  92. canvas.draw(Canvas(canvas, Rect(469, 464, 1, 42)), Point(470 + i, 464));
  93. for (int i = 0; i < 57; i++)
  94. canvas.draw(Canvas(canvas, Rect(i < 30 ? 564 : 630, 464, 1, 44)), Point(565 + i, 464));
  95. for (int i = 0; i < 56; i++)
  96. canvas.draw(Canvas(canvas, Rect(656, 464, 1, 47)), Point(657 + i, 464));
  97. // draw bookmarks
  98. canvas.draw(img, Point(278, 464), Rect(220, 405, 37, 47));
  99. canvas.draw(img, Point(481, 465), Rect(354, 406, 37, 41));
  100. canvas.draw(img, Point(575, 465), Rect(417, 406, 37, 45));
  101. canvas.draw(img, Point(667, 465), Rect(478, 406, 37, 47));
  102. std::shared_ptr<IImage> image = GH.renderHandler().createImage(canvas.getInternalSurface());
  103. image->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePath));
  104. }
  105. void AssetGenerator::createPlayerColoredBackground(const PlayerColor & player)
  106. {
  107. std::string filename = "data/DialogBoxBackground_" + player.toString() + ".png";
  108. if(CResourceHandler::get()->existsResource(ResourcePath(filename))) // overridden by mod, no generation
  109. return;
  110. if(!CResourceHandler::get("local")->createResource(filename))
  111. return;
  112. ResourcePath savePath(filename, EResType::IMAGE);
  113. auto locator = ImageLocator(ImagePath::builtin("DiBoxBck"));
  114. locator.scalingFactor = 1;
  115. std::shared_ptr<IImage> texture = GH.renderHandler().loadImage(locator, EImageBlitMode::OPAQUE);
  116. // Color transform to make color of brown DIBOX.PCX texture match color of specified player
  117. static const std::array<ColorFilter, PlayerColor::PLAYER_LIMIT_I> filters = {
  118. ColorFilter::genRangeShifter( 0.25, 0, 0, 1.25, 0.00, 0.00 ), // red
  119. ColorFilter::genRangeShifter( 0, 0, 0, 0.45, 1.20, 4.50 ), // blue
  120. ColorFilter::genRangeShifter( 0.40, 0.27, 0.23, 1.10, 1.20, 1.15 ), // tan
  121. ColorFilter::genRangeShifter( -0.27, 0.10, -0.27, 0.70, 1.70, 0.70 ), // green
  122. ColorFilter::genRangeShifter( 0.47, 0.17, -0.27, 1.60, 1.20, 0.70 ), // orange
  123. ColorFilter::genRangeShifter( 0.12, -0.1, 0.25, 1.15, 1.20, 2.20 ), // purple
  124. ColorFilter::genRangeShifter( -0.13, 0.23, 0.23, 0.90, 1.20, 2.20 ), // teal
  125. ColorFilter::genRangeShifter( 0.44, 0.15, 0.25, 1.00, 1.00, 1.75 ) // pink
  126. };
  127. assert(player.isValidPlayer());
  128. if (!player.isValidPlayer())
  129. {
  130. logGlobal->error("Unable to colorize to invalid player color %d!", static_cast<int>(player.getNum()));
  131. return;
  132. }
  133. texture->adjustPalette(filters[player.getNum()], 0);
  134. texture->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePath));
  135. }
  136. void AssetGenerator::createCombatUnitNumberWindow()
  137. {
  138. std::string filenameToSave = "data/combatUnitNumberWindow";
  139. ResourcePath savePathDefault(filenameToSave + "Default.png", EResType::IMAGE);
  140. ResourcePath savePathNeutral(filenameToSave + "Neutral.png", EResType::IMAGE);
  141. ResourcePath savePathPositive(filenameToSave + "Positive.png", EResType::IMAGE);
  142. ResourcePath savePathNegative(filenameToSave + "Negative.png", EResType::IMAGE);
  143. if(CResourceHandler::get()->existsResource(savePathDefault)) // overridden by mod, no generation
  144. return;
  145. if(!CResourceHandler::get("local")->createResource(savePathDefault.getOriginalName() + ".png") ||
  146. !CResourceHandler::get("local")->createResource(savePathNeutral.getOriginalName() + ".png") ||
  147. !CResourceHandler::get("local")->createResource(savePathPositive.getOriginalName() + ".png") ||
  148. !CResourceHandler::get("local")->createResource(savePathNegative.getOriginalName() + ".png"))
  149. return;
  150. auto locator = ImageLocator(ImagePath::builtin("CMNUMWIN"));
  151. locator.scalingFactor = 1;
  152. std::shared_ptr<IImage> texture = GH.renderHandler().loadImage(locator, EImageBlitMode::OPAQUE);
  153. static const auto shifterNormal = ColorFilter::genRangeShifter( 0.f, 0.f, 0.f, 0.6f, 0.2f, 1.0f );
  154. static const auto shifterPositive = ColorFilter::genRangeShifter( 0.f, 0.f, 0.f, 0.2f, 1.0f, 0.2f );
  155. static const auto shifterNegative = ColorFilter::genRangeShifter( 0.f, 0.f, 0.f, 1.0f, 0.2f, 0.2f );
  156. static const auto shifterNeutral = ColorFilter::genRangeShifter( 0.f, 0.f, 0.f, 1.0f, 1.0f, 0.2f );
  157. // do not change border color
  158. static const int32_t ignoredMask = 1 << 26;
  159. texture->adjustPalette(shifterNormal, ignoredMask);
  160. texture->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePathDefault));
  161. texture->adjustPalette(shifterPositive, ignoredMask);
  162. texture->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePathPositive));
  163. texture->adjustPalette(shifterNegative, ignoredMask);
  164. texture->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePathNegative));
  165. texture->adjustPalette(shifterNeutral, ignoredMask);
  166. texture->exportBitmap(*CResourceHandler::get("local")->getResourceName(savePathNeutral));
  167. }