openspades/Sources/Client/ParticleSpriteEntity.cpp

156 lines
4.3 KiB
C++
Raw Normal View History

2013-08-29 11:45:22 +09:00
/*
Copyright (c) 2013 yvt
2013-08-29 11:45:22 +09:00
This file is part of OpenSpades.
2013-08-29 11:45:22 +09:00
OpenSpades is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
2013-08-29 11:45:22 +09:00
OpenSpades is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2013-08-29 11:45:22 +09:00
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
2013-08-29 11:45:22 +09:00
*/
2013-08-18 16:18:06 +09:00
#include "ParticleSpriteEntity.h"
#include "GameMap.h"
#include "World.h"
#include <Core/Debug.h>
2013-08-18 16:18:06 +09:00
namespace spades {
namespace client {
ParticleSpriteEntity::ParticleSpriteEntity(Client *cli, Handle<IImage> image, Vector4 color)
: renderer(cli->GetRenderer()), image(image), color(color) {
position = MakeVector3(0, 0, 0);
2013-08-18 16:18:06 +09:00
velocity = MakeVector3(0, 0, 0);
radius = 1.f;
radiusVelocity = 0.f;
angle = 0.f;
rotationVelocity = 0.f;
velocityDamp = 1;
gravityScale = 1.f;
lifetime = 1.f;
radiusDamp = 1.f;
time = 0.f;
fadeInDuration = .1f;
fadeOutDuration = .5f;
additive = false;
blockHitAction = BlockHitAction::Delete;
if (cli->GetWorld())
map = cli->GetWorld()->GetMap();
2013-08-18 16:18:06 +09:00
else
map = NULL;
}
ParticleSpriteEntity::~ParticleSpriteEntity() {}
void ParticleSpriteEntity::SetLifeTime(float lifeTime, float fadeIn, float fadeOut) {
2013-08-18 16:18:06 +09:00
lifetime = lifeTime;
fadeInDuration = fadeIn;
fadeOutDuration = fadeOut;
}
void ParticleSpriteEntity::SetTrajectory(Vector3 pos, Vector3 vel, float damp, float grav) {
2013-08-18 16:18:06 +09:00
position = pos;
velocity = vel;
velocityDamp = damp;
gravityScale = grav;
}
void ParticleSpriteEntity::SetRotation(float initialAngle, float angleVelocity) {
2013-08-18 16:18:06 +09:00
angle = initialAngle;
rotationVelocity = angleVelocity;
}
void ParticleSpriteEntity::SetRadius(float initialRadius, float radiusVelocity,
float damp) {
2013-08-18 16:18:06 +09:00
radius = initialRadius;
this->radiusVelocity = radiusVelocity;
radiusDamp = damp;
}
2013-08-18 16:18:06 +09:00
bool ParticleSpriteEntity::Update(float dt) {
SPADES_MARK_FUNCTION_DEBUG();
Vector3 lastPos = position;
2013-08-18 16:18:06 +09:00
time += dt;
if (time > lifetime)
2013-08-18 16:18:06 +09:00
return false;
2013-08-18 16:18:06 +09:00
position += velocity * dt;
velocity.z += 32.f * dt * gravityScale;
2013-08-18 16:18:06 +09:00
// TODO: control clip action
if (blockHitAction != BlockHitAction::Ignore && map) {
if (map->ClipWorld(position.x, position.y, position.z)) {
if (blockHitAction == BlockHitAction::Delete) {
2013-08-18 16:18:06 +09:00
return false;
} else {
2013-08-18 16:18:06 +09:00
IntVector3 lp2 = lastPos.Floor();
IntVector3 lp = position.Floor();
if (lp.z != lp2.z && ((lp.x == lp2.x && lp.y == lp2.y) ||
!map->ClipWorld(lp.x, lp.y, lp2.z)))
2013-08-18 16:18:06 +09:00
velocity.z = -velocity.z;
else if (lp.x != lp2.x && ((lp.y == lp2.y && lp.z == lp2.z) ||
!map->ClipWorld(lp2.x, lp.y, lp.z)))
2013-08-18 16:18:06 +09:00
velocity.x = -velocity.x;
else if (lp.y != lp2.y && ((lp.x == lp2.x && lp.z == lp2.z) ||
!map->ClipWorld(lp.x, lp2.y, lp.z)))
2013-08-18 16:18:06 +09:00
velocity.y = -velocity.y;
velocity *= .36f;
position = lastPos;
}
}
}
2013-08-18 16:18:06 +09:00
// radius
if (radiusVelocity != 0.f)
2013-08-18 16:18:06 +09:00
radius += radiusVelocity * dt;
if (rotationVelocity != 0.f)
2013-08-18 16:18:06 +09:00
angle += rotationVelocity * dt;
if (velocityDamp != 1.f)
2013-08-18 16:18:06 +09:00
velocity *= powf(velocityDamp, dt);
if (radiusDamp != 1.f)
2013-08-18 16:18:06 +09:00
radiusVelocity *= powf(radiusDamp, dt);
2013-08-18 16:18:06 +09:00
return true;
}
2013-08-18 16:18:06 +09:00
void ParticleSpriteEntity::Render3D() {
SPADES_MARK_FUNCTION_DEBUG();
2013-08-18 16:18:06 +09:00
float fade = 1.f;
if (time < fadeInDuration) {
2013-08-18 16:18:06 +09:00
fade *= time / fadeInDuration;
}
if (time > lifetime - fadeOutDuration) {
2013-08-18 16:18:06 +09:00
fade *= (lifetime - time) / fadeOutDuration;
}
2013-08-18 16:18:06 +09:00
Vector4 col = color;
col.w *= fade;
// premultiplied alpha!
2013-08-18 16:18:06 +09:00
col.x *= col.w;
col.y *= col.w;
col.z *= col.w;
if (additive)
2013-08-18 16:18:06 +09:00
col.w = 0.f;
renderer.SetColorAlphaPremultiplied(col);
renderer.AddSprite(*image, position, radius, angle);
2013-08-18 16:18:06 +09:00
}
void ParticleSpriteEntity::SetImage(Handle<IImage> newImage) { image = newImage; }
} // namespace client
} // namespace spades