483 lines
22 KiB
C++
483 lines
22 KiB
C++
#include "StarEnvironmentPainter.hpp"
|
|
#include "StarLexicalCast.hpp"
|
|
#include "StarTime.hpp"
|
|
#include "StarXXHash.hpp"
|
|
#include "StarJsonExtra.hpp"
|
|
#include "StarLogging.hpp"
|
|
#include "StarMathCommon.hpp"
|
|
|
|
namespace Star {
|
|
|
|
float const EnvironmentPainter::SunriseTime = 0.072f;
|
|
float const EnvironmentPainter::SunsetTime = 0.42f;
|
|
float const EnvironmentPainter::SunFadeRate = 0.07f;
|
|
float const EnvironmentPainter::MaxFade = 0.3f;
|
|
float const EnvironmentPainter::RayPerlinFrequency = 0.005f; // Arbitrary, part of using the Perlin as a PRNG
|
|
float const EnvironmentPainter::RayPerlinAmplitude = 2;
|
|
int const EnvironmentPainter::RayCount = 60;
|
|
float const EnvironmentPainter::RayMinWidth = 0.8f; // % of its sector
|
|
float const EnvironmentPainter::RayWidthVariance = 5.0265f; // % of its sector
|
|
float const EnvironmentPainter::RayAngleVariance = 6.2832f; // Radians
|
|
float const EnvironmentPainter::SunRadius = 50;
|
|
float const EnvironmentPainter::RayColorDependenceLevel = 3.0f;
|
|
float const EnvironmentPainter::RayColorDependenceScale = 0.00625f;
|
|
float const EnvironmentPainter::RayUnscaledAlphaVariance = 2.0943f;
|
|
float const EnvironmentPainter::RayMinUnscaledAlpha = 1;
|
|
Vec3B const EnvironmentPainter::RayColor = Vec3B(255, 255, 200);
|
|
|
|
EnvironmentPainter::EnvironmentPainter(RendererPtr renderer) {
|
|
m_renderer = std::move(renderer);
|
|
m_textureGroup = make_shared<AssetTextureGroup>(m_renderer->createTextureGroup(TextureGroupSize::Large));
|
|
m_timer = 0;
|
|
m_rayPerlin = PerlinF(1, RayPerlinFrequency, RayPerlinAmplitude, 0, 2.0f, 2.0f, Random::randu64());
|
|
}
|
|
|
|
void EnvironmentPainter::update(float dt) {
|
|
// Allows the rays to change alpha with time.
|
|
m_timer += dt;
|
|
m_timer = std::fmod(m_timer, Constants::pi * 100000.0);
|
|
}
|
|
|
|
void EnvironmentPainter::renderStars(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
float nightSkyAlpha = 1.0f - min(sky.dayLevel, sky.skyAlpha);
|
|
if (nightSkyAlpha <= 0.0f)
|
|
return;
|
|
|
|
Vec4B color(255, 255, 255, 255 * nightSkyAlpha);
|
|
|
|
Vec2F viewSize = screenSize / pixelRatio;
|
|
Vec2F viewCenter = viewSize / 2;
|
|
Vec2F viewMin = sky.starOffset - viewCenter;
|
|
|
|
auto newStarsHash = starsHash(sky, viewSize);
|
|
if (newStarsHash != m_starsHash) {
|
|
m_starsHash = newStarsHash;
|
|
setupStars(sky);
|
|
}
|
|
|
|
if (!sky.settings || sky.starFrames == 0 || sky.starTypes().empty())
|
|
return;
|
|
|
|
float screenBuffer = sky.settings.queryFloat("stars.screenBuffer");
|
|
|
|
PolyF field = PolyF(RectF::withSize(viewMin, Vec2F(viewSize)).padded(screenBuffer));
|
|
field.rotate(-sky.starRotation, Vec2F(sky.starOffset));
|
|
|
|
Mat3F transform = Mat3F::identity();
|
|
transform.translate(-viewMin);
|
|
transform.rotate(sky.starRotation, viewCenter);
|
|
|
|
int starTwinkleMin = sky.settings.queryInt("stars.twinkleMin");
|
|
int starTwinkleMax = sky.settings.queryInt("stars.twinkleMax");
|
|
size_t starTypesSize = sky.starTypes().size();
|
|
|
|
auto stars = m_starGenerator->generate(field, [&](RandomSource& rand) {
|
|
size_t starType = rand.randu32() % starTypesSize;
|
|
float frameOffset = rand.randu32() % sky.starFrames + rand.randf(starTwinkleMin, starTwinkleMax);
|
|
return pair<size_t, float>(starType, frameOffset);
|
|
});
|
|
|
|
RectF viewRect = RectF::withSize(Vec2F(), viewSize).padded(screenBuffer);
|
|
|
|
auto& primitives = m_renderer->immediatePrimitives();
|
|
|
|
for (auto& star : stars) {
|
|
Vec2F screenPos = transform.transformVec2(star.first);
|
|
if (viewRect.contains(screenPos)) {
|
|
size_t starFrame = (size_t)(sky.epochTime + star.second.second) % sky.starFrames;
|
|
if (auto const& texture = m_starTextures[star.second.first * sky.starFrames + starFrame])
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), texture, screenPos * pixelRatio - Vec2F(texture->size()) / 2, 1.0, color, 0.0f);
|
|
}
|
|
}
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::renderDebrisFields(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
if (!sky.settings)
|
|
return;
|
|
|
|
if (sky.type == SkyType::Orbital || sky.type == SkyType::Warp) {
|
|
Vec2F viewSize = screenSize / pixelRatio;
|
|
Vec2F viewCenter = viewSize / 2;
|
|
Vec2D viewMin = Vec2D(sky.starOffset - viewCenter);
|
|
|
|
Mat3F rotMatrix = Mat3F::rotation(sky.starRotation, viewCenter);
|
|
|
|
JsonArray debrisFields = sky.settings.queryArray("spaceDebrisFields");
|
|
for (size_t i = 0; i < debrisFields.size(); ++i) {
|
|
Json debrisField = debrisFields[i];
|
|
|
|
Vec2F spaceDebrisVelocityRange = jsonToVec2F(debrisField.query("velocityRange"));
|
|
float debrisXVel = staticRandomFloatRange(spaceDebrisVelocityRange[0], spaceDebrisVelocityRange[1], sky.skyParameters.seed, i, "DebrisFieldXVel");
|
|
float debrisYVel = staticRandomFloatRange(spaceDebrisVelocityRange[0], spaceDebrisVelocityRange[1], sky.skyParameters.seed, i, "DebrisFieldYVel");
|
|
|
|
// Translate the entire field to make the debris seem as though they are moving
|
|
Vec2D velocityOffset = -Vec2D(debrisXVel, debrisYVel) * sky.epochTime;
|
|
|
|
JsonArray imageOptions = debrisField.query("list").toArray();
|
|
Vec2U biggest = Vec2U();
|
|
for (Json const& json : imageOptions) {
|
|
TexturePtr texture = m_textureGroup->loadTexture(*json.stringPtr());
|
|
biggest = biggest.piecewiseMax(texture->size());
|
|
}
|
|
|
|
float screenBuffer = ceil((float)biggest.max() * (float)Constants::sqrt2);
|
|
PolyD field = PolyD(RectD::withSize(viewMin + velocityOffset, Vec2D(viewSize)).padded(screenBuffer));
|
|
Vec2F debrisAngularVelocityRange = jsonToVec2F(debrisField.query("angularVelocityRange"));
|
|
auto debrisItems = m_debrisGenerators[i]->generate(field,
|
|
[&](RandomSource& rand) {
|
|
StringView debrisImage = *rand.randFrom(imageOptions).stringPtr();
|
|
float debrisAngularVelocity = rand.randf(debrisAngularVelocityRange[0], debrisAngularVelocityRange[1]);
|
|
|
|
return pair<StringView, float>(debrisImage, debrisAngularVelocity);
|
|
});
|
|
|
|
Vec2D debrisPositionOffset = viewMin + velocityOffset;
|
|
|
|
for (auto& debrisItem : debrisItems) {
|
|
Vec2F debrisPosition = rotMatrix.transformVec2(Vec2F(debrisItem.first - debrisPositionOffset));
|
|
float debrisAngle = fmod(Constants::deg2rad * debrisItem.second.second * sky.epochTime, Constants::pi * 2) + sky.starRotation;
|
|
drawOrbiter(pixelRatio, screenSize, sky, {SkyOrbiterType::SpaceDebris, 1.0f, debrisAngle, debrisItem.second.first, debrisPosition});
|
|
}
|
|
}
|
|
|
|
m_renderer->flush();
|
|
}
|
|
}
|
|
|
|
void EnvironmentPainter::renderBackOrbiters(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
for (auto const& orbiter : sky.backOrbiters(screenSize / pixelRatio))
|
|
drawOrbiter(pixelRatio, screenSize, sky, orbiter);
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::renderPlanetHorizon(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
auto planetHorizon = sky.worldHorizon(screenSize / pixelRatio);
|
|
if (planetHorizon.empty())
|
|
return;
|
|
|
|
// Can't bail sooner, need to queue all textures
|
|
bool allLoaded = true;
|
|
for (auto const& layer : planetHorizon.layers) {
|
|
if (!m_textureGroup->tryTexture(layer.first) || !m_textureGroup->tryTexture(layer.second))
|
|
allLoaded = false;
|
|
}
|
|
|
|
if (!allLoaded)
|
|
return;
|
|
|
|
float planetPixelRatio = pixelRatio * planetHorizon.scale;
|
|
Vec2F center = planetHorizon.center * pixelRatio;
|
|
|
|
auto& primitives = m_renderer->immediatePrimitives();
|
|
|
|
for (auto const& layer : planetHorizon.layers) {
|
|
TexturePtr leftTexture = m_textureGroup->loadTexture(layer.first);
|
|
Vec2F leftTextureSize(leftTexture->size());
|
|
TexturePtr rightTexture = m_textureGroup->loadTexture(layer.second);
|
|
Vec2F rightTextureSize(rightTexture->size());
|
|
|
|
Vec2F leftLayer = center;
|
|
leftLayer[0] -= leftTextureSize[0] * planetPixelRatio;
|
|
auto leftRect = RectF::withSize(leftLayer, leftTextureSize * planetPixelRatio);
|
|
PolyF leftImage = PolyF(leftRect);
|
|
leftImage.rotate(planetHorizon.rotation, center);
|
|
|
|
auto rightRect = RectF::withSize(center, rightTextureSize * planetPixelRatio);
|
|
PolyF rightImage = PolyF(rightRect);
|
|
rightImage.rotate(planetHorizon.rotation, center);
|
|
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), std::move(leftTexture),
|
|
leftImage[0], Vec2F(0, 0),
|
|
leftImage[1], Vec2F(leftTextureSize[0], 0),
|
|
leftImage[2], Vec2F(leftTextureSize[0], leftTextureSize[1]),
|
|
leftImage[3], Vec2F(0, leftTextureSize[1]), Vec4B::filled(255), 0.0f);
|
|
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), std::move(rightTexture),
|
|
rightImage[0], Vec2F(0, 0),
|
|
rightImage[1], Vec2F(rightTextureSize[0], 0),
|
|
rightImage[2], Vec2F(rightTextureSize[0], rightTextureSize[1]),
|
|
rightImage[3], Vec2F(0, rightTextureSize[1]), Vec4B::filled(255), 0.0f);
|
|
}
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::renderFrontOrbiters(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
for (auto const& orbiter : sky.frontOrbiters(screenSize / pixelRatio))
|
|
drawOrbiter(pixelRatio, screenSize, sky, orbiter);
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::renderSky(Vec2F const& screenSize, SkyRenderData const& sky) {
|
|
auto& primitives = m_renderer->immediatePrimitives();
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), TexturePtr(),
|
|
RenderVertex{Vec2F(0, 0), Vec2F(), sky.bottomRectColor.toRgba(), 0.0f},
|
|
RenderVertex{Vec2F(screenSize[0], 0), Vec2F(), sky.bottomRectColor.toRgba(), 0.0f},
|
|
RenderVertex{screenSize, Vec2F(), sky.topRectColor.toRgba(), 0.0f},
|
|
RenderVertex{Vec2F(0, screenSize[1]), Vec2F(), sky.topRectColor.toRgba(), 0.0f});
|
|
|
|
// Flash overlay for Interstellar travel
|
|
Vec4B flashColor = sky.flashColor.toRgba();
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), RectF(Vec2F(), screenSize), flashColor, 0.0f);
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
// TODO: Fix this to work with decimal zoom levels. Currently, the clouds shake rapidly when interpolating between zoom levels.
|
|
void EnvironmentPainter::renderParallaxLayers(
|
|
Vec2F parallaxWorldPosition, WorldCamera const& camera, ParallaxLayers const& layers, SkyRenderData const& sky) {
|
|
|
|
// Note: the "parallax space" referenced below is a grid where the scale of each cell is the size of the parallax image
|
|
|
|
auto& primitives = m_renderer->immediatePrimitives();
|
|
|
|
for (auto& layer : layers) {
|
|
if (layer.alpha == 0)
|
|
continue;
|
|
|
|
Vec4B drawColor;
|
|
if (layer.unlit || layer.lightMapped)
|
|
drawColor = Vec4B(255, 255, 255, floor(255 * layer.alpha));
|
|
else
|
|
drawColor = Vec4B(sky.environmentLight.toRgb(), floor(255 * layer.alpha));
|
|
|
|
Vec2F parallaxValue = layer.parallaxValue;
|
|
Vec2B parallaxRepeat = layer.repeat;
|
|
Vec2F parallaxOrigin = {0.0f, layer.verticalOrigin};
|
|
|
|
AssetPath first = layer.textures.first();
|
|
first.directives += layer.directives;
|
|
Vec2F parallaxSize = Vec2F(m_textureGroup->loadTexture(first)->size());
|
|
Vec2F parallaxPixels = parallaxSize * camera.pixelRatio();
|
|
|
|
// texture offset in *screen pixel space*
|
|
Vec2F parallaxOffset = layer.parallaxOffset * camera.pixelRatio();
|
|
if (layer.speed[0] != 0) {
|
|
double drift = fmod((double)layer.speed[0] * (sky.epochTime / (double)sky.dayLength), (double)parallaxSize[0]);
|
|
parallaxOffset[0] = fmod(parallaxOffset[0] + drift * camera.pixelRatio(), parallaxPixels[0]);
|
|
}
|
|
if (layer.speed[1] != 0) {
|
|
double drift = fmod((double)layer.speed[1] * (sky.epochTime / (double)sky.dayLength), (double)parallaxSize[1]);
|
|
parallaxOffset[1] = fmod(parallaxOffset[1] + drift * camera.pixelRatio(), parallaxPixels[1]);
|
|
}
|
|
|
|
// parallax camera world position in *parallax space*
|
|
Vec2F parallaxCameraCenter = parallaxWorldPosition - parallaxOrigin;
|
|
parallaxCameraCenter =
|
|
Vec2F((((parallaxCameraCenter[0] / parallaxPixels[0]) * TilePixels) * camera.pixelRatio()) / parallaxValue[0],
|
|
(((parallaxCameraCenter[1] / parallaxPixels[1]) * TilePixels) * camera.pixelRatio()) / parallaxValue[1]);
|
|
|
|
// width / height of screen in *parallax space*
|
|
float parallaxScreenWidth = camera.screenSize()[0] / parallaxPixels[0];
|
|
float parallaxScreenHeight = camera.screenSize()[1] / parallaxPixels[1];
|
|
|
|
// screen world position in *parallax space*
|
|
float parallaxScreenLeft = parallaxCameraCenter[0] - parallaxScreenWidth / 2.0;
|
|
float parallaxScreenBottom = parallaxCameraCenter[1] - parallaxScreenHeight / 2.0;
|
|
|
|
// screen boundary world positions in *parallax space*
|
|
Vec2F parallaxScreenOffset = parallaxOffset.piecewiseDivide(parallaxPixels);
|
|
int left = floor(parallaxScreenLeft + parallaxScreenOffset[0]);
|
|
int bottom = floor(parallaxScreenBottom + parallaxScreenOffset[1]);
|
|
int right = ceil(parallaxScreenLeft + parallaxScreenWidth + parallaxScreenOffset[0]);
|
|
int top = ceil(parallaxScreenBottom + parallaxScreenHeight + parallaxScreenOffset[1]);
|
|
|
|
// positions to start tiling in *screen pixel space*
|
|
float pixelLeft = (left - parallaxScreenLeft) * parallaxPixels[0] - parallaxOffset[0];
|
|
float pixelBottom = (bottom - parallaxScreenBottom) * parallaxPixels[1] - parallaxOffset[1];
|
|
|
|
// vertical top and bottom cutoff points in *parallax space*
|
|
float tileLimitTop = top;
|
|
if (layer.tileLimitTop)
|
|
tileLimitTop = (layer.parallaxOffset[1] - layer.tileLimitTop.value()) / parallaxSize[1];
|
|
float tileLimitBottom = bottom;
|
|
if (layer.tileLimitBottom)
|
|
tileLimitBottom = (layer.parallaxOffset[1] - layer.tileLimitBottom.value()) / parallaxSize[1];
|
|
|
|
float lightMapMultiplier = (!layer.unlit && layer.lightMapped) ? 1.0f : 0.0f;
|
|
|
|
for (int y = bottom; y <= top; ++y) {
|
|
if (!(parallaxRepeat[1] || y == 0) || y > tileLimitTop || y + 1 < tileLimitBottom)
|
|
continue;
|
|
for (int x = left; x <= right; ++x) {
|
|
if (!(parallaxRepeat[0] || x == 0))
|
|
continue;
|
|
float pixelTileLeft = pixelLeft + (x - left) * parallaxPixels[0];
|
|
float pixelTileBottom = pixelBottom + (y - bottom) * parallaxPixels[1];
|
|
|
|
Vec2F anchorPoint(pixelTileLeft, pixelTileBottom);
|
|
|
|
RectF subImage = RectF::withSize(Vec2F(), parallaxSize);
|
|
if (tileLimitTop != top && y + 1 > tileLimitTop)
|
|
subImage.setYMin(parallaxSize[1] * (1.0f - fpart(tileLimitTop)));
|
|
if (tileLimitBottom != bottom && y < tileLimitBottom)
|
|
anchorPoint[1] += fpart(tileLimitBottom) * parallaxPixels[1];
|
|
|
|
for (auto const& textureImage : layer.textures) {
|
|
AssetPath withDirectives = textureImage;
|
|
withDirectives.directives += layer.directives;
|
|
if (auto texture = m_textureGroup->tryTexture(withDirectives)) {
|
|
RectF drawRect = RectF::withSize(anchorPoint, subImage.size() * camera.pixelRatio());
|
|
primitives.emplace_back(std::in_place_type_t<RenderQuad>(), std::move(texture),
|
|
RenderVertex{drawRect.min(), subImage.min(), drawColor, lightMapMultiplier},
|
|
RenderVertex{{drawRect.xMax(), drawRect.yMin()}, {subImage.xMax(), subImage.yMin()}, drawColor, lightMapMultiplier},
|
|
RenderVertex{drawRect.max(), subImage.max(), drawColor, lightMapMultiplier},
|
|
RenderVertex{{drawRect.xMin(), drawRect.yMax()}, {subImage.xMin(), subImage.yMax()}, drawColor, lightMapMultiplier});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::cleanup(int64_t textureTimeout) {
|
|
m_textureGroup->cleanup(textureTimeout);
|
|
}
|
|
|
|
void EnvironmentPainter::drawRays(
|
|
float pixelRatio, SkyRenderData const& sky, Vec2F start, float length, double time, float alpha) {
|
|
// All magic constants are either 2PI or arbritrary to allow the Perlin to act
|
|
// as a PRNG
|
|
float sectorWidth = 2 * Constants::pi / RayCount; // Radians
|
|
Vec3B color = sky.topRectColor.toRgb();
|
|
|
|
for (int i = 0; i < RayCount; i++)
|
|
drawRay(pixelRatio,
|
|
sky,
|
|
start,
|
|
sectorWidth * (std::abs(m_rayPerlin.get(i * 25)) * RayWidthVariance + RayMinWidth),
|
|
length,
|
|
i * sectorWidth + m_rayPerlin.get(i * 314) * RayAngleVariance,
|
|
time,
|
|
color,
|
|
alpha);
|
|
|
|
m_renderer->flush();
|
|
}
|
|
|
|
void EnvironmentPainter::drawRay(float pixelRatio,
|
|
SkyRenderData const& sky,
|
|
Vec2F start,
|
|
float width,
|
|
float length,
|
|
float angle,
|
|
double time,
|
|
Vec3B color,
|
|
float alpha) {
|
|
// All magic constants are arbritrary to allow the Perlin to act as a PRNG
|
|
|
|
float currentTime = sky.timeOfDay / sky.dayLength;
|
|
float timeSinceSunEvent = std::min(std::abs(currentTime - SunriseTime), std::abs(currentTime - SunsetTime));
|
|
float percentFaded = MaxFade * (1.0f - std::min(1.0f, std::pow(timeSinceSunEvent / SunFadeRate, 2.0f)));
|
|
// Gets the current average sky color
|
|
color = (Vec3B)((Vec3F)color * (1 - percentFaded) + (Vec3F)sky.mainSkyColor.toRgb() * percentFaded);
|
|
// Sum is used to vary the ray intensity based on sky color
|
|
// Rays show up more on darker backgrounds, so this scales to remove that
|
|
float sum = std::pow((color[0] + color[1]) * RayColorDependenceScale, RayColorDependenceLevel);
|
|
Vec3B rayColor;
|
|
if (sky.settings.queryBool("sun.dynamicImage.enabled", false) && !sky.skyParameters.sunType.empty())
|
|
rayColor = jsonToVec3B(sky.settings.query("sun.dynamicImage.rayColors." + sky.skyParameters.sunType, sky.settings.query("sun.rayColor", JsonArray{RayColor[0], RayColor[1], RayColor[2]})));
|
|
else
|
|
rayColor = jsonToVec3B(sky.settings.query("sun.rayColor", JsonArray{RayColor[0], RayColor[1], RayColor[2]}));
|
|
float sunScale = sky.settings.queryFloat("sun.scale", 1.0f);
|
|
m_renderer->immediatePrimitives().emplace_back(std::in_place_type_t<RenderQuad>(), TexturePtr(),
|
|
RenderVertex{start + Vec2F(std::cos(angle + width), std::sin(angle + width)) * length, {}, Vec4B(rayColor, 0), 0.0f},
|
|
RenderVertex{start + Vec2F(std::cos(angle + width), std::sin(angle + width)) * SunRadius * sunScale * pixelRatio,
|
|
{},
|
|
Vec4B(rayColor,
|
|
(int)(RayMinUnscaledAlpha + std::abs(m_rayPerlin.get(angle * 896 + time * 30) * RayUnscaledAlphaVariance))
|
|
* sum
|
|
* alpha), 0.0f},
|
|
RenderVertex{start + Vec2F(std::cos(angle), std::sin(angle)) * SunRadius * sunScale * pixelRatio,
|
|
{},
|
|
Vec4B(rayColor,
|
|
(int)(RayMinUnscaledAlpha + std::abs(m_rayPerlin.get(angle * 626 + time * 30) * RayUnscaledAlphaVariance))
|
|
* sum
|
|
* alpha), 0.0f},
|
|
RenderVertex{start + Vec2F(std::cos(angle), std::sin(angle)) * length, {}, Vec4B(rayColor, 0), 0.0f});
|
|
}
|
|
|
|
void EnvironmentPainter::drawOrbiter(float pixelRatio, Vec2F const& screenSize, SkyRenderData const& sky, SkyOrbiter const& orbiter) {
|
|
float alpha = 1.0f;
|
|
Vec2F position;
|
|
|
|
// The way Starbound positions these is weird.
|
|
// It's a random point on a 400 by 400 area from the bottom left of the screen.
|
|
// That origin point is then multiplied by the zoom level.
|
|
// This does not intuitively scale with higher-resolution monitors, so lets fix that.
|
|
if (orbiter.type == SkyOrbiterType::Moon) {
|
|
const Vec2F correctionOrigin = { 320, 180 };
|
|
// correctionOrigin is 1920x1080 / default zoom level / 2, the most likely dev setup at the time.
|
|
Vec2F offset = orbiter.position - correctionOrigin;
|
|
position = (screenSize / 2) + offset * pixelRatio;
|
|
}
|
|
else
|
|
position = orbiter.position * pixelRatio;
|
|
|
|
if (orbiter.type == SkyOrbiterType::Sun) {
|
|
alpha = sky.dayLevel;
|
|
drawRays(pixelRatio, sky, position, std::max(screenSize[0], screenSize[1]), m_timer, sky.skyAlpha);
|
|
}
|
|
|
|
TexturePtr texture = m_textureGroup->loadTexture(orbiter.image);
|
|
Vec2F texSize = Vec2F(texture->size());
|
|
|
|
Mat3F renderMatrix = Mat3F::rotation(orbiter.angle, position);
|
|
RectF renderRect = RectF::withCenter(position, texSize * orbiter.scale * pixelRatio);
|
|
Vec4B renderColor = Vec4B(255, 255, 255, 255 * alpha);
|
|
|
|
m_renderer->immediatePrimitives().emplace_back(std::in_place_type_t<RenderQuad>(), std::move(texture),
|
|
renderMatrix.transformVec2(renderRect.min()), Vec2F(0, 0),
|
|
renderMatrix.transformVec2(Vec2F{renderRect.xMax(), renderRect.yMin()}), Vec2F(texSize[0], 0),
|
|
renderMatrix.transformVec2(renderRect.max()), Vec2F(texSize[0], texSize[1]),
|
|
renderMatrix.transformVec2(Vec2F{renderRect.xMin(), renderRect.yMax()}), Vec2F(0, texSize[1]),
|
|
renderColor, 0.0f);
|
|
}
|
|
|
|
uint64_t EnvironmentPainter::starsHash(SkyRenderData const& sky, Vec2F const& viewSize) const {
|
|
XXHash64 hasher;
|
|
|
|
hasher.push(reinterpret_cast<char const*>(&viewSize[0]), sizeof(viewSize[0]));
|
|
hasher.push(reinterpret_cast<char const*>(&viewSize[1]), sizeof(viewSize[1]));
|
|
hasher.push(reinterpret_cast<char const*>(&sky.skyParameters.seed), sizeof(sky.skyParameters.seed));
|
|
hasher.push(reinterpret_cast<char const*>(&sky.type), sizeof(sky.type));
|
|
|
|
return hasher.digest();
|
|
}
|
|
|
|
void EnvironmentPainter::setupStars(SkyRenderData const& sky) {
|
|
if (!sky.settings)
|
|
return;
|
|
|
|
StringList const& starTypes = sky.starTypes();
|
|
size_t starTypesSize = starTypes.size();
|
|
|
|
m_starTextures.resize(starTypesSize * sky.starFrames);
|
|
for (size_t i = 0; i < starTypesSize; ++i) {
|
|
for (size_t j = 0; j < sky.starFrames; ++j)
|
|
m_starTextures[i * sky.starFrames + j] = m_textureGroup->loadTexture(starTypes[i] + ":" + toString(j));
|
|
}
|
|
|
|
int starCellSize = sky.settings.queryInt("stars.cellSize");
|
|
Vec2I starCount = jsonToVec2I(sky.settings.query("stars.cellCount"));
|
|
|
|
m_starGenerator = make_shared<Random2dPointGenerator<pair<size_t, float>>>(sky.skyParameters.seed, starCellSize, starCount);
|
|
|
|
JsonArray debrisFields = sky.settings.queryArray("spaceDebrisFields");
|
|
m_debrisGenerators.resize(debrisFields.size());
|
|
for (size_t i = 0; i < debrisFields.size(); ++i) {
|
|
int debrisCellSize = debrisFields[i].getInt("cellSize");
|
|
Vec2I debrisCountRange = jsonToVec2I(debrisFields[i].get("cellCountRange"));
|
|
uint64_t debrisSeed = staticRandomU64(sky.skyParameters.seed, i, "DebrisFieldSeed");
|
|
m_debrisGenerators[i] = make_shared<Random2dPointGenerator<pair<String, float>, double>>(debrisSeed, debrisCellSize, debrisCountRange);
|
|
}
|
|
}
|
|
|
|
}
|