parent
ed896e9770
commit
07a8e9aa5e
@ -0,0 +1,372 @@ |
||||
/*
|
||||
Procedural Generation: Programming The Universe |
||||
"Here we go again! Year 4 begins now..." - javidx9 |
||||
|
||||
License (OLC-3) |
||||
~~~~~~~~~~~~~~~ |
||||
|
||||
Copyright 2018-2020 OneLoneCoder.com |
||||
|
||||
Redistribution and use in source and binary forms, with or without |
||||
modification, are permitted provided that the following conditions |
||||
are met: |
||||
|
||||
1. Redistributions or derivations of source code must retain the above |
||||
copyright notice, this list of conditions and the following disclaimer. |
||||
|
||||
2. Redistributions or derivative works in binary form must reproduce |
||||
the above copyright notice. This list of conditions and the following |
||||
disclaimer must be reproduced in the documentation and/or other |
||||
materials provided with the distribution. |
||||
|
||||
3. Neither the name of the copyright holder nor the names of its |
||||
contributors may be used to endorse or promote products derived |
||||
from this software without specific prior written permission. |
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
|
||||
Relevant Video: https://youtu.be/ZZY9YE7rZJw
|
||||
|
||||
Links |
||||
~~~~~ |
||||
YouTube: https://www.youtube.com/javidx9
|
||||
https://www.youtube.com/javidx9extra
|
||||
Discord: https://discord.gg/WhwHUMV
|
||||
Twitter: https://www.twitter.com/javidx9
|
||||
Twitch: https://www.twitch.tv/javidx9
|
||||
GitHub: https://www.github.com/onelonecoder
|
||||
Patreon: https://www.patreon.com/javidx9
|
||||
Homepage: https://www.onelonecoder.com
|
||||
|
||||
Author |
||||
~~~~~~ |
||||
David Barr, aka javidx9, ©OneLoneCoder 2018, 2019, 2020 |
||||
*/ |
||||
|
||||
|
||||
|
||||
#define OLC_PGE_APPLICATION |
||||
#include "olcPixelGameEngine.h" |
||||
|
||||
#include <random> |
||||
|
||||
constexpr uint32_t g_starColours[8] =
|
||||
{ |
||||
0xFFFFFFFF, 0xFFD9FFFF, 0xFFA3FFFF, 0xFFFFC8C8, |
||||
0xFFFFCB9D, 0xFF9F9FFF, 0xFF415EFF, 0xFF28199D |
||||
}; |
||||
|
||||
|
||||
struct sPlanet |
||||
{ |
||||
double distance = 0.0; |
||||
double diameter = 0.0; |
||||
double foliage = 0.0; |
||||
double minerals = 0.0; |
||||
double water = 0.0; |
||||
double gases = 0.0; |
||||
double temperature = 0.0; |
||||
double population = 0.0; |
||||
bool ring = false; |
||||
std::vector<double> vMoons; |
||||
}; |
||||
|
||||
class cStarSystem |
||||
{ |
||||
public: |
||||
cStarSystem(uint32_t x, uint32_t y, bool bGenerateFullSystem = false) |
||||
{ |
||||
// Set seed based on location of star system
|
||||
nProcGen = (x & 0xFFFF) << 16 | (y & 0xFFFF); |
||||
|
||||
// Not all locations contain a system
|
||||
starExists = (rndInt(0, 20) == 1); |
||||
if (!starExists) return; |
||||
|
||||
// Generate Star
|
||||
starDiameter = rndDouble(10.0, 40.0); |
||||
starColour.n = g_starColours[rndInt(0, 8)]; |
||||
|
||||
// When viewing the galaxy map, we only care about the star
|
||||
// so abort early
|
||||
if (!bGenerateFullSystem) return; |
||||
|
||||
// If we are viewing the system map, we need to generate the
|
||||
// full system
|
||||
|
||||
// Generate Planets
|
||||
double dDistanceFromStar = rndDouble(60.0, 200.0); |
||||
int nPlanets = rndInt(0, 10); |
||||
for (int i = 0; i < nPlanets; i++) |
||||
{ |
||||
sPlanet p; |
||||
p.distance = dDistanceFromStar; |
||||
dDistanceFromStar += rndDouble(20.0, 200.0); |
||||
p.diameter = rndDouble(4.0, 20.0); |
||||
|
||||
// Could make temeprature a function of distance from star
|
||||
p.temperature = rndDouble(-200.0, 300.0); |
||||
|
||||
// Composition of planet
|
||||
p.foliage = rndDouble(0.0, 1.0); |
||||
p.minerals = rndDouble(0.0, 1.0); |
||||
p.gases = rndDouble(0.0, 1.0); |
||||
p.water = rndDouble(0.0, 1.0); |
||||
|
||||
// Normalise to 100%
|
||||
double dSum = 1.0 / (p.foliage + p.minerals + p.gases + p.water); |
||||
p.foliage *= dSum; |
||||
p.minerals *= dSum; |
||||
p.gases *= dSum; |
||||
p.water *= dSum; |
||||
|
||||
// Population could be a function of other habitat encouraging
|
||||
// properties, such as temperature and water
|
||||
p.population = std::max(rndInt(-5000000, 20000000), 0); |
||||
|
||||
// 10% of planets have a ring
|
||||
p.ring = rndInt(0, 10) == 1; |
||||
|
||||
// Satellites (Moons)
|
||||
int nMoons = std::max(rndInt(-5, 5), 0); |
||||
for (int n = 0; n < nMoons; n++) |
||||
{ |
||||
// A moon is just a diameter for now, but it could be
|
||||
// whatever you want!
|
||||
p.vMoons.push_back(rndDouble(1.0, 5.0)); |
||||
} |
||||
|
||||
// Add planet to vector
|
||||
vPlanets.push_back(p); |
||||
}
|
||||
} |
||||
|
||||
~cStarSystem() |
||||
{ |
||||
|
||||
} |
||||
|
||||
public: |
||||
std::vector<sPlanet> vPlanets; |
||||
|
||||
public: |
||||
bool starExists = false; |
||||
double starDiameter = 0.0f; |
||||
olc::Pixel starColour = olc::WHITE; |
||||
|
||||
private: |
||||
uint32_t nProcGen = 0; |
||||
|
||||
double rndDouble(double min, double max) |
||||
{ |
||||
return ((double)rnd() / (double)(0x7FFFFFFF)) * (max - min) + min; |
||||
} |
||||
|
||||
int rndInt(int min, int max) |
||||
{ |
||||
return (rnd() % (max - min)) + min; |
||||
} |
||||
|
||||
uint32_t rnd() |
||||
{ |
||||
nProcGen += 0xe120fc15; |
||||
uint64_t tmp; |
||||
tmp = (uint64_t)nProcGen * 0x4a39b70d; |
||||
uint32_t m1 = (tmp >> 32) ^ tmp; |
||||
tmp = (uint64_t)m1 * 0x12fad5c9; |
||||
uint32_t m2 = (tmp >> 32) ^ tmp; |
||||
return m2; |
||||
} |
||||
}; |
||||
|
||||
class olcGalaxy : public olc::PixelGameEngine |
||||
{ |
||||
public: |
||||
olcGalaxy() |
||||
{ |
||||
sAppName = "olcGalaxy"; |
||||
} |
||||
|
||||
public: |
||||
bool OnUserCreate() override |
||||
{ |
||||
return true; |
||||
} |
||||
|
||||
olc::vf2d vGalaxyOffset = { 0,0 }; |
||||
bool bStarSelected = false; |
||||
uint32_t nSelectedStarSeed1 = 0; |
||||
uint32_t nSelectedStarSeed2 = 0; |
||||
|
||||
|
||||
/*uint32_t nLehmer = 0;
|
||||
uint32_t Lehmer32() |
||||
{ |
||||
nLehmer += 0xe120fc15; |
||||
uint64_t tmp; |
||||
tmp = (uint64_t)nLehmer * 0x4a39b70d; |
||||
uint32_t m1 = (tmp >> 32) ^ tmp; |
||||
tmp = (uint64_t)m1 * 0x12fad5c9; |
||||
uint32_t m2 = (tmp >> 32) ^ tmp; |
||||
return m2; |
||||
}*/ |
||||
|
||||
bool OnUserUpdate(float fElapsedTime) override |
||||
{ |
||||
if (fElapsedTime <= 0.0001f) return true; |
||||
Clear(olc::BLACK); |
||||
|
||||
//if (GetKey(olc::SPACE).bReleased)
|
||||
//{
|
||||
|
||||
// //srand(1000);
|
||||
|
||||
// std::random_device rd;
|
||||
// std::mt19937 mt(1000);
|
||||
// std::uniform_int_distribution<int> dist(0, 256);
|
||||
|
||||
// auto tp1 = std::chrono::system_clock::now();
|
||||
// // Ranomness Tests
|
||||
// for (int x = 0; x < ScreenWidth(); x++)
|
||||
// {
|
||||
// for (int y = 0; y < ScreenHeight(); y++)
|
||||
// {
|
||||
// bool bIsStar = false;
|
||||
// int nSeed = y << 16 | x;
|
||||
//
|
||||
// // Standard C++ rand()
|
||||
// //srand(nSeed);
|
||||
// //bIsStar = rand() % 256 < 32;
|
||||
|
||||
// // std::random
|
||||
// //mt.seed(nSeed);
|
||||
// //bIsStar = dist(mt) < 32;
|
||||
|
||||
// // Lehmer32
|
||||
// nLehmer = nSeed;
|
||||
// bIsStar = Lehmer32() % 256 < 32;
|
||||
|
||||
// Draw(x, y, bIsStar ? olc::WHITE : olc::BLACK);
|
||||
// }
|
||||
// }
|
||||
// auto tp2 = std::chrono::system_clock::now();
|
||||
// std::chrono::duration<float> elapsedTime = tp2 - tp1;
|
||||
// DrawString(3, 3, "Time: " + std::to_string(elapsedTime.count()), olc::RED, 2);
|
||||
//}
|
||||
|
||||
|
||||
//return true;
|
||||
|
||||
|
||||
if (GetKey(olc::W).bHeld) vGalaxyOffset.y -= 50.0f * fElapsedTime; |
||||
if (GetKey(olc::S).bHeld) vGalaxyOffset.y += 50.0f * fElapsedTime; |
||||
if (GetKey(olc::A).bHeld) vGalaxyOffset.x -= 50.0f * fElapsedTime; |
||||
if (GetKey(olc::D).bHeld) vGalaxyOffset.x += 50.0f * fElapsedTime; |
||||
|
||||
int nSectorsX = ScreenWidth() / 16; |
||||
int nSectorsY = ScreenHeight() / 16; |
||||
|
||||
olc::vi2d mouse = { GetMouseX() / 16, GetMouseY() / 16 }; |
||||
olc::vi2d galaxy_mouse = mouse + vGalaxyOffset; |
||||
olc::vi2d screen_sector = { 0,0 }; |
||||
|
||||
for (screen_sector.x = 0; screen_sector.x < nSectorsX; screen_sector.x++) |
||||
for (screen_sector.y = 0; screen_sector.y < nSectorsY; screen_sector.y++) |
||||
{ |
||||
uint32_t seed1 = (uint32_t)vGalaxyOffset.x + (uint32_t)screen_sector.x; |
||||
uint32_t seed2 = (uint32_t)vGalaxyOffset.y + (uint32_t)screen_sector.y; |
||||
|
||||
cStarSystem star(seed1, seed2); |
||||
if (star.starExists) |
||||
{ |
||||
FillCircle(screen_sector.x * 16 + 8, screen_sector.y * 16 + 8,
|
||||
(int)star.starDiameter / 8, star.starColour); |
||||
|
||||
// For convenience highlight hovered star
|
||||
if (mouse.x == screen_sector.x && mouse.y == screen_sector.y) |
||||
{ |
||||
DrawCircle(screen_sector.x * 16 + 8, screen_sector.y * 16 + 8, 12, olc::YELLOW); |
||||
} |
||||
} |
||||
} |
||||
|
||||
// Handle Mouse Click
|
||||
if (GetMouse(0).bPressed) |
||||
{ |
||||
uint32_t seed1 = (uint32_t)vGalaxyOffset.x + (uint32_t)mouse.x; |
||||
uint32_t seed2 = (uint32_t)vGalaxyOffset.y + (uint32_t)mouse.y; |
||||
|
||||
cStarSystem star(seed1, seed2); |
||||
if (star.starExists) |
||||
{ |
||||
bStarSelected = true; |
||||
nSelectedStarSeed1 = seed1; |
||||
nSelectedStarSeed2 = seed2; |
||||
} |
||||
else |
||||
bStarSelected = false; |
||||
} |
||||
|
||||
// Draw Details of selected star system
|
||||
if (bStarSelected) |
||||
{ |
||||
// Generate full star system
|
||||
cStarSystem star(nSelectedStarSeed1, nSelectedStarSeed2, true); |
||||
|
||||
// Draw Window
|
||||
FillRect(8, 240, 496, 232, olc::DARK_BLUE); |
||||
DrawRect(8, 240, 496, 232, olc::WHITE); |
||||
|
||||
// Draw Star
|
||||
olc::vi2d vBody = { 14, 356 }; |
||||
|
||||
vBody.x += star.starDiameter * 1.375; |
||||
FillCircle(vBody, (int)(star.starDiameter * 1.375), star.starColour);
|
||||
vBody.x += (star.starDiameter * 1.375) + 8; |
||||
|
||||
|
||||
|
||||
// Draw Planets
|
||||
for (auto& planet : star.vPlanets) |
||||
{ |
||||
if (vBody.x + planet.diameter >= 496) break; |
||||
|
||||
vBody.x += planet.diameter; |
||||
FillCircle(vBody, (int)(planet.diameter * 1.0), olc::RED); |
||||
|
||||
olc::vi2d vMoon = vBody; |
||||
vMoon.y += planet.diameter + 10; |
||||
|
||||
// Draw Moons
|
||||
for (auto& moon : planet.vMoons) |
||||
{ |
||||
vMoon.y += moon; |
||||
FillCircle(vMoon, (int)(moon * 1.0), olc::GREY); |
||||
vMoon.y += moon + 10; |
||||
} |
||||
|
||||
vBody.x += planet.diameter + 8; |
||||
} |
||||
} |
||||
|
||||
return true; |
||||
} |
||||
}; |
||||
|
||||
int main() |
||||
{ |
||||
olcGalaxy demo; |
||||
if (demo.Construct(512, 480, 2, 2, false, false)) |
||||
demo.Start(); |
||||
return 0; |
||||
} |
Loading…
Reference in new issue