102 lines
2.7 KiB
C++
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
|