starfield.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. #include "starfield.h"
  2. #include "utils.h"
  3. Starfield::Starfield(door::Door &Door, std::mt19937 &Rng)
  4. : door{Door}, rng{Rng}, uni_x{1, Door.width}, uni_y{1, Door.height},
  5. white{door::COLOR::WHITE}, dark{door::COLOR::BLACK, door::ATTR::BRIGHT} {
  6. // std::uniform_int_distribution<int> uni_x(1, mx);
  7. // std::uniform_int_distribution<int> uni_y(1, my);
  8. regenerate();
  9. }
  10. star_pos Starfield::make_pos(void) {
  11. star_pos pos;
  12. do {
  13. pos.x = uni_x(rng);
  14. pos.y = uni_y(rng);
  15. } while (sky.find(pos) != sky.end());
  16. return pos;
  17. }
  18. void Starfield::regenerate(void) {
  19. int mx = door.width;
  20. int my = door.height;
  21. // Make uniform random distribution between 1 and MAX screen size X/Y
  22. sky.clear();
  23. // 10 is too many, 100 is too few. 40 looks ok.
  24. int MAX_STARS = ((mx * my) / 40);
  25. // door.log() << "Generating starmap using " << mx << "," << my << " : "
  26. // << MAX_STARS << " stars." << std::endl;
  27. for (int i = 0; i < MAX_STARS; i++) {
  28. star_pos pos = make_pos();
  29. // store symbol and color in star_pos.
  30. // If we do any animation, we won't be able to rely on the star_pos keeping
  31. // the same position in the set.
  32. pos.symbol = i % 2;
  33. pos.color = i % 5 < 2;
  34. sky.insert(pos);
  35. }
  36. }
  37. #ifdef NOMORE
  38. void Starfield::animate(void) {
  39. int rx = uni_x(rng);
  40. int ry = uni_y(rng);
  41. int mx = door.width;
  42. int my = door.height;
  43. int cx = mx / 2;
  44. int cy = my / 2;
  45. std::set<star_pos> original = sky;
  46. sky.clear();
  47. for (auto &spos : original) {
  48. star_pos orig = spos;
  49. star_pos newpos = orig;
  50. int dx = mx - abs(orig.x - cx);
  51. int dy = my - abs(orig.y - cy);
  52. if (dx > rx) {
  53. // something x
  54. if (newpos.x > cx)
  55. ++newpos.x;
  56. else
  57. --newpos.x;
  58. }
  59. if (dy > ry) {
  60. // something y
  61. if (newpos.y > cy)
  62. ++newpos.y;
  63. else
  64. --newpos.y;
  65. }
  66. if ((newpos.x == 0) or (newpos.x >= mx) or (newpos.y == 0) or
  67. (newpos.y >= my)) {
  68. newpos = make_pos();
  69. newpos.color = orig.color;
  70. newpos.symbol = orig.symbol;
  71. }
  72. if (sky.find(newpos) == sky.end()) {
  73. door << door::Goto(orig.x, orig.y) << " ";
  74. sky.insert(newpos);
  75. door << door::Goto(newpos.x, newpos.y);
  76. if (newpos.color)
  77. door << dark;
  78. else
  79. door << white;
  80. if (newpos.symbol)
  81. door << stars[0];
  82. else
  83. door << stars[1];
  84. } else {
  85. // rats!
  86. if (sky.find(orig) == sky.end()) {
  87. sky.insert(orig);
  88. } else {
  89. door << door::Goto(orig.x, orig.y) << " ";
  90. newpos = make_pos();
  91. newpos.color = orig.color;
  92. newpos.symbol = orig.symbol;
  93. sky.insert(newpos);
  94. door << door::Goto(newpos.x, newpos.y);
  95. if (newpos.color)
  96. door << dark;
  97. else
  98. door << white;
  99. if (newpos.symbol)
  100. door << stars[0];
  101. else
  102. door << stars[1];
  103. }
  104. }
  105. }
  106. }
  107. #endif
  108. void Starfield::display(void) {
  109. door << door::reset << door::cls;
  110. stars[0] = ".";
  111. if (door::unicode) {
  112. stars[1] = "\u2219"; // "\u00b7";
  113. } else {
  114. stars[1] = "\xf9"; // "\xfa";
  115. };
  116. int i = 0;
  117. // lose int i, and set last_pos to -1, -1.
  118. star_pos last_pos;
  119. for (auto &pos : sky) {
  120. bool use_goto = true;
  121. if (i != 0) {
  122. // check last_pos to current position
  123. if (pos.y == last_pos.y) {
  124. // Ok, same row -- try some optimizations
  125. int dx = pos.x - last_pos.x;
  126. if (dx == 0) {
  127. use_goto = false;
  128. } else {
  129. if (dx < 5) {
  130. door << std::string(dx, ' ');
  131. use_goto = false;
  132. } else {
  133. // Use ANSI Cursor Forward
  134. door << "\x1b[" << dx << "C";
  135. use_goto = false;
  136. }
  137. }
  138. }
  139. }
  140. if (use_goto) {
  141. door::Goto star_at(pos.x, pos.y);
  142. door << star_at;
  143. }
  144. if (pos.color)
  145. door << dark;
  146. else
  147. door << white;
  148. if (pos.symbol)
  149. door << stars[0];
  150. else
  151. door << stars[1];
  152. ++i;
  153. last_pos = pos;
  154. last_pos.x++; // star output moves us by one.
  155. }
  156. }
  157. #ifdef NOMORE
  158. void display___starfield(door::Door &door, std::mt19937 &rng) {
  159. door << door::reset << door::cls;
  160. int mx = door.width;
  161. int my = door.height;
  162. // display Starfield
  163. const char *stars[2];
  164. stars[0] = ".";
  165. if (door::unicode) {
  166. stars[1] = "\u2219"; // "\u00b7";
  167. } else {
  168. stars[1] = "\xf9"; // "\xfa";
  169. };
  170. {
  171. // Make uniform random distribution between 1 and MAX screen size X/Y
  172. std::uniform_int_distribution<int> uni_x(1, mx);
  173. std::uniform_int_distribution<int> uni_y(1, my);
  174. door::ANSIColor white(door::COLOR::WHITE);
  175. door::ANSIColor dark(door::COLOR::BLACK, door::ATTR::BRIGHT);
  176. // 10 is too many, 100 is too few. 40 looks ok.
  177. int MAX_STARS = ((mx * my) / 40);
  178. // door.log() << "Generating starmap using " << mx << "," << my << " : "
  179. // << MAX_STARS << " stars." << std::endl;
  180. std::set<star_pos> sky;
  181. for (int i = 0; i < MAX_STARS; i++) {
  182. star_pos pos;
  183. bool valid;
  184. do {
  185. pos.x = uni_x(rng);
  186. pos.y = uni_y(rng);
  187. auto ret = sky.insert(pos);
  188. // was insert a success? (not a duplicate)
  189. valid = ret.second;
  190. } while (!valid);
  191. }
  192. int i = 0;
  193. star_pos last_pos;
  194. for (auto &pos : sky) {
  195. bool use_goto = true;
  196. if (i != 0) {
  197. // check last_pos to current position
  198. if (pos.y == last_pos.y) {
  199. // Ok, same row -- try some optimizations
  200. int dx = pos.x - last_pos.x;
  201. if (dx == 0) {
  202. use_goto = false;
  203. } else {
  204. if (dx < 5) {
  205. door << std::string(dx, ' ');
  206. use_goto = false;
  207. } else {
  208. // Use ANSI Cursor Forward
  209. door << "\x1b[" << dx << "C";
  210. use_goto = false;
  211. }
  212. }
  213. }
  214. }
  215. if (use_goto) {
  216. door::Goto star_at(pos.x, pos.y);
  217. door << star_at;
  218. }
  219. if (i % 5 < 2)
  220. door << dark;
  221. else
  222. door << white;
  223. if (i % 2 == 0)
  224. door << stars[0];
  225. else
  226. door << stars[1];
  227. ++i;
  228. last_pos = pos;
  229. last_pos.x++; // star output moves us by one.
  230. }
  231. }
  232. }
  233. #endif
  234. AnimatedStarfield::AnimatedStarfield(door::Door &Door, std::mt19937 &Rng)
  235. : door{Door}, rng{Rng}, uni_x{1, Door.width}, uni_y{1, Door.height},
  236. white{door::COLOR::WHITE}, dark{door::COLOR::BLACK, door::ATTR::BRIGHT} {
  237. mx = door.width;
  238. my = door.height;
  239. cx = mx / 2.0;
  240. cy = my / 2.0;
  241. max_d = distance(0, 0);
  242. // std::uniform_int_distribution<int> uni_x(1, mx);
  243. // std::uniform_int_distribution<int> uni_y(1, my);
  244. regenerate();
  245. }
  246. moving_star AnimatedStarfield::make_pos(void) {
  247. moving_star pos;
  248. pos.x = uni_x(rng);
  249. pos.y = uni_y(rng);
  250. pos.xpos = pos.x;
  251. pos.ypos = pos.y;
  252. pos.movex = pos.xpos - cx;
  253. pos.movey = pos.ypos - cy;
  254. double bigd = max(abs(pos.movex), abs(pos.movey));
  255. pos.movex /= bigd;
  256. pos.movey /= bigd;
  257. /*
  258. if (get_logger)
  259. get_logger() << pos.x << "," << pos.y << " " << pos.movex << "/"
  260. << pos.movey << std::endl;
  261. */
  262. return pos;
  263. }
  264. void AnimatedStarfield::regenerate(void) {
  265. // Make uniform random distribution between 1 and MAX screen size X/Y
  266. sky.clear();
  267. // 10 is too many, 100 is too few. 40 looks ok.
  268. int MAX_STARS = ((mx * my) / 40);
  269. // door.log() << "Generating starmap using " << mx << "," << my << " : "
  270. // << MAX_STARS << " stars." << std::endl;
  271. for (int i = 0; i < MAX_STARS; i++) {
  272. moving_star pos = make_pos();
  273. // store symbol and color in star_pos.
  274. // If we do any animation, we won't be able to rely on the star_pos keeping
  275. // the same position in the set.
  276. pos.symbol = i % 2;
  277. pos.color = i % 5 < 2;
  278. pos.xpos = pos.x;
  279. pos.ypos = pos.y;
  280. // calculate movex, movey here and now!
  281. sky.push_back(pos);
  282. }
  283. }
  284. void AnimatedStarfield::display(void) {
  285. door << door::reset << door::cls;
  286. stars[0] = ".";
  287. if (door::unicode) {
  288. stars[1] = "\u2219"; // "\u00b7";
  289. } else {
  290. stars[1] = "\xf9"; // "\xfa";
  291. };
  292. for (auto &pos : sky) {
  293. door::Goto star_at(pos.x, pos.y);
  294. door << star_at;
  295. if (pos.color)
  296. door << dark;
  297. else
  298. door << white;
  299. if (pos.symbol)
  300. door << stars[0];
  301. else
  302. door << stars[1];
  303. }
  304. }
  305. void AnimatedStarfield::animate(void) {
  306. for (auto &star : sky) {
  307. // just start with basic movement
  308. // double speed = abs((cx / 2) - distance(star.movex, star.movey));
  309. double speed = max_d / distance(star.movex, star.movey);
  310. star.xpos += star.movex / (speed * 4);
  311. star.ypos += star.movey / speed;
  312. int nx = int(star.xpos + 0.5);
  313. int ny = int(star.ypos + 0.5);
  314. // watch the bottom of the screen! It'll scroll!
  315. // watch new position we're given!
  316. while ((star.xpos < 1) or (star.xpos >= mx - 1) or (star.ypos < 1) or
  317. (star.ypos >= my - 1)) {
  318. // off the screen! Whoops!
  319. moving_star new_pos = make_pos();
  320. nx = new_pos.x;
  321. ny = new_pos.y;
  322. star.xpos = new_pos.x;
  323. star.ypos = new_pos.y;
  324. star.movex = new_pos.movex;
  325. star.movey = new_pos.movey;
  326. // new position updated -- next section will erase and display in new
  327. // position
  328. }
  329. // if ((star.x != int(star.xpos)) or (star.y == int(star.ypos))) {
  330. if ((star.x != nx) or (star.y != ny)) {
  331. // star has moved, update time
  332. door << door::Goto(star.x, star.y) << " ";
  333. star.x = nx; // int(star.xpos);
  334. star.y = ny; // int(star.ypos);
  335. door << door::Goto(star.x, star.y);
  336. if (star.color)
  337. door << dark;
  338. else
  339. door << white;
  340. if (star.symbol)
  341. door << stars[0];
  342. else
  343. door << stars[1];
  344. }
  345. }
  346. }
  347. double AnimatedStarfield::distance(double x, double y) {
  348. return sqrt(((cx - x) * (cx - x)) + ((cy - y) * (cy - y)));
  349. }