102 lines
2.7 KiB
C++

//
// Tracer.cpp
// OpenSpades
//
// Created by Tomoaki Kawada on 8/30/13.
// Copyright (c) 2013 yvt.jp. All rights reserved.
//
#include <algorithm>
#include "Tracer.h"
#include "Client.h"
#include "IRenderer.h"
#include <Draw/SWRenderer.h>
namespace spades {
namespace client {
Tracer::Tracer(Client &_client, Vector3 p1, Vector3 p2, float bulletVel)
: client(_client), startPos(p1), velocity(bulletVel) {
dir = (p2 - p1).Normalize();
length = (p2 - p1).GetLength();
const float maxTimeSpread = 1.0f / 60.f;
const float shutterTime = 1.0f / 100.f;
visibleLength = shutterTime * velocity;
curDistance = -visibleLength;
// Randomize the starting position within the range of the shutter
// time. However, make sure the tracer is displayed for at least
// one frame.
curDistance +=
std::min(length + visibleLength, maxTimeSpread * SampleRandomFloat() * velocity);
firstUpdate = true;
image = client.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 {
SceneDefinition sceneDef = client.GetLastSceneDef();
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};
float distanceToCamera = (pos2 - sceneDef.viewOrigin).GetLength();
float radius = 0.002f * distanceToCamera;
r.SetColorAlphaPremultiplied(col * 0.4f);
r.AddLongSprite(*image, pos1, pos2, radius);
}
}
}
Tracer::~Tracer() {}
} // namespace client
} // namespace spades