yvt 585cbfce1c Improve random number generation
The global RNG state is now protected from simultaneous accesses by
employing a mutex.

Performance sensitive code requiring generation of a large number of
random numbers can utilize `LocalRNG`. This is not thread-safe but
provides a much greater throughput. `LocalRNG` is seeded by the global
RNG only at construction time.
2018-09-17 23:48:38 +09:00

90 lines
2.3 KiB
C++

//
// Tracer.cpp
// OpenSpades
//
// Created by Tomoaki Kawada on 8/30/13.
// Copyright (c) 2013 yvt.jp. All rights reserved.
//
#include "Tracer.h"
#include "Client.h"
#include "IRenderer.h"
#include <Draw/SWRenderer.h>
namespace spades {
namespace client {
Tracer::Tracer(Client *cli, Vector3 p1, Vector3 p2, float bulletVel)
: client(cli), startPos(p1), velocity(bulletVel) {
dir = (p2 - p1).Normalize();
length = (p2 - p1).GetLength();
velocity *= 0.5f; // make it slower for visual effect
const float maxTimeSpread = 1.0f / 30.f;
const float shutterTime = 1.0f / 30.f;
visibleLength = shutterTime * velocity;
curDistance = -visibleLength;
curDistance += maxTimeSpread * SampleRandomFloat();
firstUpdate = true;
image = cli->GetRenderer()->RegisterImage("Gfx/Ball.png");
}
bool Tracer::Update(float dt) {
if (!firstUpdate) {
curDistance += dt * velocity;
if (curDistance > length) {
return false;
}
}
firstUpdate = false;
return true;
}
void Tracer::Render3D() {
IRenderer *r = client->GetRenderer();
if (dynamic_cast<draw::SWRenderer *>(r)) {
// SWRenderer doesn't support long sprites (yet)
float startDist = curDistance;
float endDist = curDistance + visibleLength;
startDist = std::max(startDist, 0.f);
endDist = std::min(endDist, length);
if (startDist >= endDist) {
return;
}
Vector3 pos1 = startPos + dir * startDist;
Vector3 pos2 = startPos + dir * endDist;
Vector4 col = {1.f, .6f, .2f, 0.f};
r->AddDebugLine(pos1, pos2, Vector4{1.0f, 0.6f, 0.2f, 1.0f});
} else {
for (float step = 0.0f; step <= 1.0f; step += 0.1f) {
float startDist = curDistance;
float endDist = curDistance + visibleLength;
float midDist = (startDist + endDist) * 0.5f;
startDist = Mix(startDist, midDist, step);
endDist = Mix(endDist, midDist, step);
startDist = std::max(startDist, 0.f);
endDist = std::min(endDist, length);
if (startDist >= endDist) {
continue;
}
Vector3 pos1 = startPos + dir * startDist;
Vector3 pos2 = startPos + dir * endDist;
Vector4 col = {1.f, .6f, .2f, 0.f};
r->SetColorAlphaPremultiplied(col * 0.4f);
r->AddLongSprite(image, pos1, pos2, .05f);
}
}
}
Tracer::~Tracer() {}
}
}