123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412 |
- #include "starfield.h"
- #include "utils.h"
- Starfield::Starfield(door::Door &Door, std::mt19937 &Rng)
- : door{Door}, rng{Rng}, uni_x{1, Door.width}, uni_y{1, Door.height},
- white{door::COLOR::WHITE}, dark{door::COLOR::BLACK, door::ATTR::BRIGHT} {
- // std::uniform_int_distribution<int> uni_x(1, mx);
- // std::uniform_int_distribution<int> uni_y(1, my);
- regenerate();
- }
- star_pos Starfield::make_pos(void) {
- star_pos pos;
- do {
- pos.x = uni_x(rng);
- pos.y = uni_y(rng);
- } while (sky.find(pos) != sky.end());
- return pos;
- }
- void Starfield::regenerate(void) {
- int mx = door.width;
- int my = door.height;
- // Make uniform random distribution between 1 and MAX screen size X/Y
- sky.clear();
- // 10 is too many, 100 is too few. 40 looks ok.
- int MAX_STARS = ((mx * my) / 40);
- // door.log() << "Generating starmap using " << mx << "," << my << " : "
- // << MAX_STARS << " stars." << std::endl;
- for (int i = 0; i < MAX_STARS; i++) {
- star_pos pos = make_pos();
- // store symbol and color in star_pos.
- // If we do any animation, we won't be able to rely on the star_pos keeping
- // the same position in the set.
- pos.symbol = i % 2;
- pos.color = i % 5 < 2;
- sky.insert(pos);
- }
- }
- #ifdef NOMORE
- void Starfield::animate(void) {
- int rx = uni_x(rng);
- int ry = uni_y(rng);
- int mx = door.width;
- int my = door.height;
- int cx = mx / 2;
- int cy = my / 2;
- std::set<star_pos> original = sky;
- sky.clear();
- for (auto &spos : original) {
- star_pos orig = spos;
- star_pos newpos = orig;
- int dx = mx - abs(orig.x - cx);
- int dy = my - abs(orig.y - cy);
- if (dx > rx) {
- // something x
- if (newpos.x > cx)
- ++newpos.x;
- else
- --newpos.x;
- }
- if (dy > ry) {
- // something y
- if (newpos.y > cy)
- ++newpos.y;
- else
- --newpos.y;
- }
- if ((newpos.x == 0) or (newpos.x >= mx) or (newpos.y == 0) or
- (newpos.y >= my)) {
- newpos = make_pos();
- newpos.color = orig.color;
- newpos.symbol = orig.symbol;
- }
- if (sky.find(newpos) == sky.end()) {
- door << door::Goto(orig.x, orig.y) << " ";
- sky.insert(newpos);
- door << door::Goto(newpos.x, newpos.y);
- if (newpos.color)
- door << dark;
- else
- door << white;
- if (newpos.symbol)
- door << stars[0];
- else
- door << stars[1];
- } else {
- // rats!
- if (sky.find(orig) == sky.end()) {
- sky.insert(orig);
- } else {
- door << door::Goto(orig.x, orig.y) << " ";
- newpos = make_pos();
- newpos.color = orig.color;
- newpos.symbol = orig.symbol;
- sky.insert(newpos);
- door << door::Goto(newpos.x, newpos.y);
- if (newpos.color)
- door << dark;
- else
- door << white;
- if (newpos.symbol)
- door << stars[0];
- else
- door << stars[1];
- }
- }
- }
- }
- #endif
- void Starfield::display(void) {
- door << door::reset << door::cls;
- stars[0] = ".";
- if (door::unicode) {
- stars[1] = "\u2219"; // "\u00b7";
- } else {
- stars[1] = "\xf9"; // "\xfa";
- };
- int i = 0;
- // lose int i, and set last_pos to -1, -1.
- star_pos last_pos;
- for (auto &pos : sky) {
- bool use_goto = true;
- if (i != 0) {
- // check last_pos to current position
- if (pos.y == last_pos.y) {
- // Ok, same row -- try some optimizations
- int dx = pos.x - last_pos.x;
- if (dx == 0) {
- use_goto = false;
- } else {
- if (dx < 5) {
- door << std::string(dx, ' ');
- use_goto = false;
- } else {
- // Use ANSI Cursor Forward
- door << "\x1b[" << dx << "C";
- use_goto = false;
- }
- }
- }
- }
- if (use_goto) {
- door::Goto star_at(pos.x, pos.y);
- door << star_at;
- }
- if (pos.color)
- door << dark;
- else
- door << white;
- if (pos.symbol)
- door << stars[0];
- else
- door << stars[1];
- ++i;
- last_pos = pos;
- last_pos.x++; // star output moves us by one.
- }
- }
- #ifdef NOMORE
- void display___starfield(door::Door &door, std::mt19937 &rng) {
- door << door::reset << door::cls;
- int mx = door.width;
- int my = door.height;
- // display Starfield
- const char *stars[2];
- stars[0] = ".";
- if (door::unicode) {
- stars[1] = "\u2219"; // "\u00b7";
- } else {
- stars[1] = "\xf9"; // "\xfa";
- };
- {
- // Make uniform random distribution between 1 and MAX screen size X/Y
- std::uniform_int_distribution<int> uni_x(1, mx);
- std::uniform_int_distribution<int> uni_y(1, my);
- door::ANSIColor white(door::COLOR::WHITE);
- door::ANSIColor dark(door::COLOR::BLACK, door::ATTR::BRIGHT);
- // 10 is too many, 100 is too few. 40 looks ok.
- int MAX_STARS = ((mx * my) / 40);
- // door.log() << "Generating starmap using " << mx << "," << my << " : "
- // << MAX_STARS << " stars." << std::endl;
- std::set<star_pos> sky;
- for (int i = 0; i < MAX_STARS; i++) {
- star_pos pos;
- bool valid;
- do {
- pos.x = uni_x(rng);
- pos.y = uni_y(rng);
- auto ret = sky.insert(pos);
- // was insert a success? (not a duplicate)
- valid = ret.second;
- } while (!valid);
- }
- int i = 0;
- star_pos last_pos;
- for (auto &pos : sky) {
- bool use_goto = true;
- if (i != 0) {
- // check last_pos to current position
- if (pos.y == last_pos.y) {
- // Ok, same row -- try some optimizations
- int dx = pos.x - last_pos.x;
- if (dx == 0) {
- use_goto = false;
- } else {
- if (dx < 5) {
- door << std::string(dx, ' ');
- use_goto = false;
- } else {
- // Use ANSI Cursor Forward
- door << "\x1b[" << dx << "C";
- use_goto = false;
- }
- }
- }
- }
- if (use_goto) {
- door::Goto star_at(pos.x, pos.y);
- door << star_at;
- }
- if (i % 5 < 2)
- door << dark;
- else
- door << white;
- if (i % 2 == 0)
- door << stars[0];
- else
- door << stars[1];
- ++i;
- last_pos = pos;
- last_pos.x++; // star output moves us by one.
- }
- }
- }
- #endif
- AnimatedStarfield::AnimatedStarfield(door::Door &Door, std::mt19937 &Rng)
- : door{Door}, rng{Rng}, uni_x{1, Door.width}, uni_y{1, Door.height},
- white{door::COLOR::WHITE}, dark{door::COLOR::BLACK, door::ATTR::BRIGHT} {
- mx = door.width;
- my = door.height;
- cx = mx / 2.0;
- cy = my / 2.0;
- max_d = distance(0, 0);
- // std::uniform_int_distribution<int> uni_x(1, mx);
- // std::uniform_int_distribution<int> uni_y(1, my);
- regenerate();
- }
- moving_star AnimatedStarfield::make_pos(void) {
- moving_star pos;
- pos.x = uni_x(rng);
- pos.y = uni_y(rng);
- pos.xpos = pos.x;
- pos.ypos = pos.y;
- pos.movex = pos.xpos - cx;
- pos.movey = pos.ypos - cy;
- double bigd = max(abs(pos.movex), abs(pos.movey));
- pos.movex /= bigd;
- pos.movey /= bigd;
- /*
- if (get_logger)
- get_logger() << pos.x << "," << pos.y << " " << pos.movex << "/"
- << pos.movey << std::endl;
- */
- return pos;
- }
- void AnimatedStarfield::regenerate(void) {
- // Make uniform random distribution between 1 and MAX screen size X/Y
- sky.clear();
- // 10 is too many, 100 is too few. 40 looks ok.
- int MAX_STARS = ((mx * my) / 40);
- // door.log() << "Generating starmap using " << mx << "," << my << " : "
- // << MAX_STARS << " stars." << std::endl;
- for (int i = 0; i < MAX_STARS; i++) {
- moving_star pos = make_pos();
- // store symbol and color in star_pos.
- // If we do any animation, we won't be able to rely on the star_pos keeping
- // the same position in the set.
- pos.symbol = i % 2;
- pos.color = i % 5 < 2;
- pos.xpos = pos.x;
- pos.ypos = pos.y;
- // calculate movex, movey here and now!
- sky.push_back(pos);
- }
- }
- void AnimatedStarfield::display(void) {
- door << door::reset << door::cls;
- stars[0] = ".";
- if (door::unicode) {
- stars[1] = "\u2219"; // "\u00b7";
- } else {
- stars[1] = "\xf9"; // "\xfa";
- };
- for (auto &pos : sky) {
- door::Goto star_at(pos.x, pos.y);
- door << star_at;
- if (pos.color)
- door << dark;
- else
- door << white;
- if (pos.symbol)
- door << stars[0];
- else
- door << stars[1];
- }
- }
- void AnimatedStarfield::animate(void) {
- for (auto &star : sky) {
- // just start with basic movement
- // double speed = abs((cx / 2) - distance(star.movex, star.movey));
- double speed = max_d / distance(star.movex, star.movey);
- star.xpos += star.movex / (speed * 4);
- star.ypos += star.movey / speed;
- int nx = int(star.xpos + 0.5);
- int ny = int(star.ypos + 0.5);
- // watch the bottom of the screen! It'll scroll!
- // watch new position we're given!
- while ((star.xpos < 1) or (star.xpos >= mx - 1) or (star.ypos < 1) or
- (star.ypos >= my - 1)) {
- // off the screen! Whoops!
- moving_star new_pos = make_pos();
- nx = new_pos.x;
- ny = new_pos.y;
- star.xpos = new_pos.x;
- star.ypos = new_pos.y;
- star.movex = new_pos.movex;
- star.movey = new_pos.movey;
- // new position updated -- next section will erase and display in new
- // position
- }
- // if ((star.x != int(star.xpos)) or (star.y == int(star.ypos))) {
- if ((star.x != nx) or (star.y != ny)) {
- // star has moved, update time
- door << door::Goto(star.x, star.y) << " ";
- star.x = nx; // int(star.xpos);
- star.y = ny; // int(star.ypos);
- door << door::Goto(star.x, star.y);
- if (star.color)
- door << dark;
- else
- door << white;
- if (star.symbol)
- door << stars[0];
- else
- door << stars[1];
- }
- }
- }
- double AnimatedStarfield::distance(double x, double y) {
- return sqrt(((cx - x) * (cx - x)) + ((cy - y) * (cy - y)));
- }
|