starfield.cpp 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #include "starfield.h"
  2. starfield::starfield(door::Door &Door, std::mt19937 &Rng)
  3. : door{Door}, rng{Rng} {
  4. regenerate();
  5. }
  6. void starfield::regenerate(void) {
  7. int mx = door.width;
  8. int my = door.height;
  9. // Make uniform random distribution between 1 and MAX screen size X/Y
  10. std::uniform_int_distribution<int> uni_x(1, mx);
  11. std::uniform_int_distribution<int> uni_y(1, my);
  12. // 10 is too many, 100 is too few. 40 looks ok.
  13. int MAX_STARS = ((mx * my) / 40);
  14. // door.log() << "Generating starmap using " << mx << "," << my << " : "
  15. // << MAX_STARS << " stars." << std::endl;
  16. for (int i = 0; i < MAX_STARS; i++) {
  17. star_pos pos;
  18. bool valid;
  19. do {
  20. pos.x = uni_x(rng);
  21. pos.y = uni_y(rng);
  22. auto ret = sky.insert(pos);
  23. // was insert a success? (not a duplicate)
  24. valid = ret.second;
  25. } while (!valid);
  26. }
  27. }
  28. void starfield::display(void) {
  29. door << door::reset << door::cls;
  30. door::ANSIColor white(door::COLOR::WHITE);
  31. door::ANSIColor dark(door::COLOR::BLACK, door::ATTR::BRIGHT);
  32. // display starfield
  33. const char *stars[2];
  34. stars[0] = ".";
  35. if (door::unicode) {
  36. stars[1] = "\u2219"; // "\u00b7";
  37. } else {
  38. stars[1] = "\xf9"; // "\xfa";
  39. };
  40. int i = 0;
  41. star_pos last_pos;
  42. for (auto &pos : sky) {
  43. bool use_goto = true;
  44. if (i != 0) {
  45. // check last_pos to current position
  46. if (pos.y == last_pos.y) {
  47. // Ok, same row -- try some optimizations
  48. int dx = pos.x - last_pos.x;
  49. if (dx == 0) {
  50. use_goto = false;
  51. } else {
  52. if (dx < 5) {
  53. door << std::string(dx, ' ');
  54. use_goto = false;
  55. } else {
  56. // Use ANSI Cursor Forward
  57. door << "\x1b[" << dx << "C";
  58. use_goto = false;
  59. }
  60. }
  61. }
  62. }
  63. if (use_goto) {
  64. door::Goto star_at(pos.x, pos.y);
  65. door << star_at;
  66. }
  67. if (i % 5 < 2)
  68. door << dark;
  69. else
  70. door << white;
  71. if (i % 2 == 0)
  72. door << stars[0];
  73. else
  74. door << stars[1];
  75. ++i;
  76. last_pos = pos;
  77. last_pos.x++; // star output moves us by one.
  78. }
  79. }
  80. void display___starfield(door::Door &door, std::mt19937 &rng) {
  81. door << door::reset << door::cls;
  82. int mx = door.width;
  83. int my = door.height;
  84. // display starfield
  85. const char *stars[2];
  86. stars[0] = ".";
  87. if (door::unicode) {
  88. stars[1] = "\u2219"; // "\u00b7";
  89. } else {
  90. stars[1] = "\xf9"; // "\xfa";
  91. };
  92. {
  93. // Make uniform random distribution between 1 and MAX screen size X/Y
  94. std::uniform_int_distribution<int> uni_x(1, mx);
  95. std::uniform_int_distribution<int> uni_y(1, my);
  96. door::ANSIColor white(door::COLOR::WHITE);
  97. door::ANSIColor dark(door::COLOR::BLACK, door::ATTR::BRIGHT);
  98. // 10 is too many, 100 is too few. 40 looks ok.
  99. int MAX_STARS = ((mx * my) / 40);
  100. // door.log() << "Generating starmap using " << mx << "," << my << " : "
  101. // << MAX_STARS << " stars." << std::endl;
  102. std::set<star_pos> sky;
  103. for (int i = 0; i < MAX_STARS; i++) {
  104. star_pos pos;
  105. bool valid;
  106. do {
  107. pos.x = uni_x(rng);
  108. pos.y = uni_y(rng);
  109. auto ret = sky.insert(pos);
  110. // was insert a success? (not a duplicate)
  111. valid = ret.second;
  112. } while (!valid);
  113. }
  114. int i = 0;
  115. star_pos last_pos;
  116. for (auto &pos : sky) {
  117. bool use_goto = true;
  118. if (i != 0) {
  119. // check last_pos to current position
  120. if (pos.y == last_pos.y) {
  121. // Ok, same row -- try some optimizations
  122. int dx = pos.x - last_pos.x;
  123. if (dx == 0) {
  124. use_goto = false;
  125. } else {
  126. if (dx < 5) {
  127. door << std::string(dx, ' ');
  128. use_goto = false;
  129. } else {
  130. // Use ANSI Cursor Forward
  131. door << "\x1b[" << dx << "C";
  132. use_goto = false;
  133. }
  134. }
  135. }
  136. }
  137. if (use_goto) {
  138. door::Goto star_at(pos.x, pos.y);
  139. door << star_at;
  140. }
  141. if (i % 5 < 2)
  142. door << dark;
  143. else
  144. door << white;
  145. if (i % 2 == 0)
  146. door << stars[0];
  147. else
  148. door << stars[1];
  149. ++i;
  150. last_pos = pos;
  151. last_pos.x++; // star output moves us by one.
  152. }
  153. }
  154. }