openspades/Sources/Client/FallingBlock.cpp

225 lines
6.5 KiB
C++

/*
Copyright (c) 2013 yvt
This file is part of OpenSpades.
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.
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.
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
*/
#include "FallingBlock.h"
#include <Core/Debug.h>
#include <Core/Exception.h>
#include "Client.h"
#include "GameMap.h"
#include "IModel.h"
#include "IRenderer.h"
#include "ParticleSpriteEntity.h"
#include "SmokeSpriteEntity.h"
#include "World.h"
#include <limits.h>
namespace spades {
namespace client {
FallingBlock::FallingBlock(Client *client, std::vector<IntVector3> blocks)
: client(client) {
if (blocks.empty())
SPRaise("No block given");
// find min/max
int maxX = -1, maxY = -1, maxZ = -1;
int minX = INT_MAX, minY = INT_MAX, minZ = INT_MAX;
uint64_t xSum = 0, ySum = 0, zSum = 0;
numBlocks = (int)blocks.size();
for (size_t i = 0; i < blocks.size(); i++) {
IntVector3 v = blocks[i];
if (v.x < minX)
minX = v.x;
if (v.y < minY)
minY = v.y;
if (v.z < minZ)
minZ = v.z;
if (v.x > maxX)
maxX = v.x;
if (v.y > maxY)
maxY = v.y;
if (v.z > maxZ)
maxZ = v.z;
xSum += v.x;
ySum += v.y;
zSum += v.z;
}
GameMap *map = client->GetWorld()->GetMap();
// build voxel model
vmodel = new VoxelModel(maxX - minX + 1, maxY - minY + 1, maxZ - minZ + 1);
for (size_t i = 0; i < blocks.size(); i++) {
IntVector3 v = blocks[i];
uint32_t col = map->GetColor(v.x, v.y, v.z);
// Use the default material
col &= 0xffffff;
vmodel->SetSolid(v.x - minX, v.y - minY, v.z - minZ, col);
}
// center of gravity
Vector3 origin;
origin.x = (float)minX - (float)xSum / (float)blocks.size();
origin.y = (float)minY - (float)ySum / (float)blocks.size();
origin.z = (float)minZ - (float)zSum / (float)blocks.size();
vmodel->SetOrigin(origin);
Vector3 matTrans = MakeVector3((float)minX, (float)minY, (float)minZ);
matTrans += .5f; // voxelmodel's (0,0,0) origins on block center
matTrans -= origin; // cancel origin
matrix = Matrix4::Translate(matTrans);
// build renderer model
model = client->GetRenderer()->CreateModel(vmodel);
time = 0.f;
}
FallingBlock::~FallingBlock() {
model->Release();
vmodel->Release();
}
bool FallingBlock::Update(float dt) {
time += dt;
GameMap *map = client->GetWorld()->GetMap();
Vector3 orig = matrix.GetOrigin();
if (time > 1.f || map->ClipBox(orig.x, orig.y, orig.z)) {
// destroy
int w = vmodel->GetWidth();
int h = vmodel->GetHeight();
int d = vmodel->GetDepth();
Matrix4 vmat = lastMatrix;
vmat = vmat * Matrix4::Translate(vmodel->GetOrigin());
// block center
Vector3 vmOrigin = vmat.GetOrigin();
Vector3 vmAxis1 = vmat.GetAxis(0);
Vector3 vmAxis2 = vmat.GetAxis(1);
Vector3 vmAxis3 = vmat.GetAxis(2);
Handle<IImage> img = client->GetRenderer()->RegisterImage("Gfx/White.tga");
bool usePrecisePhysics = false;
float dist =
(client->GetLastSceneDef().viewOrigin - matrix.GetOrigin()).GetLength();
if (dist < 16.f) {
if (numBlocks < 1000) {
usePrecisePhysics = true;
}
}
float impact;
if (dist > 170.f)
return false;
impact = (float)numBlocks / 100.f;
client->grenadeVibration += impact / (dist + 5.f);
if (client->grenadeVibration > 1.f)
client->grenadeVibration = 1.f;
auto *getRandom = SampleRandomFloat;
for (int x = 0; x < w; x++) {
Vector3 p1 = vmOrigin + vmAxis1 * (float)x;
for (int y = 0; y < h; y++) {
Vector3 p2 = p1 + vmAxis2 * (float)y;
for (int z = 0; z < d; z++) {
if (!vmodel->IsSolid(x, y, z))
continue;
// inner voxel?
if (x > 0 && y > 0 && z > 0 && x < w - 1 && y < h - 1 && z < d - 1 &&
vmodel->IsSolid(x - 1, y, z) && vmodel->IsSolid(x + 1, y, z) &&
vmodel->IsSolid(x, y - 1, z) && vmodel->IsSolid(x, y + 1, z) &&
vmodel->IsSolid(x, y, z - 1) && vmodel->IsSolid(x, y, z + 1))
continue;
uint32_t c = vmodel->GetColor(x, y, z);
Vector4 col;
col.x = (float)((uint8_t)(c)) / 255.f;
col.y = (float)((uint8_t)(c >> 8)) / 255.f;
col.z = (float)((uint8_t)(c >> 16)) / 255.f;
col.w = 1.;
Vector3 p3 = p2 + vmAxis3 * (float)z;
{
ParticleSpriteEntity *ent =
new SmokeSpriteEntity(client, col, 70.f);
ent->SetTrajectory(p3, (MakeVector3(getRandom() - getRandom(),
getRandom() - getRandom(),
getRandom() - getRandom())) *
0.2f,
1.f, 0.f);
ent->SetRotation(getRandom() * (float)M_PI * 2.f);
ent->SetRadius(1.0f, 0.5f);
ent->SetBlockHitAction(ParticleSpriteEntity::Ignore);
ent->SetLifeTime(1.0f + getRandom() * 0.5f, 0.f, 1.0f);
client->AddLocalEntity(ent);
}
col.w = 1.f;
for (int i = 0; i < 6; i++) {
ParticleSpriteEntity *ent =
new ParticleSpriteEntity(client, img, col);
ent->SetTrajectory(p3, MakeVector3(getRandom() - getRandom(),
getRandom() - getRandom(),
getRandom() - getRandom()) *
13.f,
1.f, .6f);
ent->SetRotation(getRandom() * (float)M_PI * 2.f);
ent->SetRadius(0.35f + getRandom() * getRandom() * 0.1f);
ent->SetLifeTime(2.f, 0.f, 1.f);
if (usePrecisePhysics)
ent->SetBlockHitAction(ParticleSpriteEntity::BounceWeak);
client->AddLocalEntity(ent);
}
}
}
}
return false;
}
lastMatrix = matrix;
Matrix4 rot;
rot = Matrix4::Rotate(MakeVector3(1.f, 1.f, 0.f), time * 1.4f * dt);
matrix = matrix * rot;
Matrix4 trans;
trans = Matrix4::Translate(0, 0, time * dt * 4.f);
matrix = trans * matrix;
return true;
}
void FallingBlock::Render3D() {
ModelRenderParam param;
param.matrix = matrix;
client->GetRenderer()->RenderModel(model, param);
}
}
}