/*
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 .
*/
#include "Corpse.h"
#include "GameMap.h"
#include "IModel.h"
#include "IRenderer.h"
#include "Player.h"
#include "World.h"
#include
#include
using namespace std;
DEFINE_SPADES_SETTING(r_corpseLineCollision, "1");
namespace spades {
namespace client {
Corpse::Corpse(IRenderer &renderer, GameMap &map, Player &p)
: renderer{renderer}, map{map} {
SPADES_MARK_FUNCTION();
playerId = p.GetId();
IntVector3 col = p.GetWorld().GetTeam(p.GetTeamId()).color;
color = MakeVector3(col.x / 255.f, col.y / 255.f, col.z / 255.f);
bool crouch = p.GetInput().crouch;
Vector3 front = p.GetFront();
float yaw = atan2(front.y, front.x) + static_cast(M_PI) * .5f;
// float pitch = -atan2(front.z, sqrt(front.x * front.x + front.y * front.y));
// lower axis
Matrix4 lower = Matrix4::Translate(p.GetOrigin());
lower = lower * Matrix4::Rotate(MakeVector3(0, 0, 1), yaw);
Matrix4 torso;
if (crouch) {
lower = lower * Matrix4::Translate(0, 0, -0.4f);
torso = lower * Matrix4::Translate(0, 0, -0.3f);
SetNode(Torso1, torso * MakeVector3(0.4f, -.15f, 0.1f));
SetNode(Torso2, torso * MakeVector3(-0.4f, -.15f, 0.1f));
SetNode(Torso3, torso * MakeVector3(-0.4f, .8f, 0.7f));
SetNode(Torso4, torso * MakeVector3(0.4f, .8f, 0.7f));
SetNode(Leg1, lower * MakeVector3(-0.4f, .1f, 1.f));
SetNode(Leg2, lower * MakeVector3(0.4f, .1f, 1.f));
SetNode(Arm1, torso * MakeVector3(0.2f, -.4f, .2f));
SetNode(Arm2, torso * MakeVector3(-0.2f, -.4f, .2f));
} else {
torso = lower * Matrix4::Translate(0, 0, -1.1f);
SetNode(Torso1, torso * MakeVector3(0.4f, 0.f, 0.1f));
SetNode(Torso2, torso * MakeVector3(-0.4f, 0.f, 0.1f));
SetNode(Torso3, torso * MakeVector3(-0.4f, .0f, 1.f));
SetNode(Torso4, torso * MakeVector3(0.4f, .0f, 1.f));
SetNode(Leg1, lower * MakeVector3(-0.4f, .0f, 1.f));
SetNode(Leg2, lower * MakeVector3(0.4f, .0f, 1.f));
SetNode(Arm1, torso * MakeVector3(0.2f, -.4f, .2f));
SetNode(Arm2, torso * MakeVector3(-0.2f, -.4f, .2f));
}
SetNode(Head, (nodes[Torso1].pos + nodes[Torso2].pos) * .5f + MakeVector3(0, 0, -0.6f));
}
void Corpse::SetNode(NodeType n, spades::Vector3 v) {
auto velNoise = [&] { return (SampleRandomFloat() - SampleRandomFloat()) * 2.0f; };
SPAssert(n >= 0);
SPAssert(n < NodeCount);
nodes[n].pos = v;
nodes[n].vel = MakeVector3(velNoise(), velNoise(), 0.f);
nodes[n].lastPos = v;
nodes[n].lastForce = MakeVector3(0, 0, 0);
}
void Corpse::SetNode(NodeType n, spades::Vector4 v) { SetNode(n, v.GetXYZ()); }
Corpse::~Corpse() {}
void Corpse::Spring(NodeType n1, NodeType n2, float distance, float dt) {
SPADES_MARK_FUNCTION_DEBUG();
SPAssert(n1 >= 0);
SPAssert(n1 < NodeCount);
SPAssert(n2 >= 0);
SPAssert(n2 < NodeCount);
Node &a = nodes[n1];
Node &b = nodes[n2];
Vector3 diff = b.pos - a.pos;
float dist = diff.GetLength();
Vector3 force = diff.Normalize() * (distance - dist);
force *= dt * 50.f;
b.vel += force;
a.vel -= force;
b.pos += force / (dt * 50.f) * 0.5f;
a.pos -= force / (dt * 50.f) * 0.5f;
Vector3 velMid = (a.vel + b.vel) * .5f;
float dump = 1.f - powf(.1f, dt);
a.vel += (velMid - a.vel) * dump;
b.vel += (velMid - b.vel) * dump;
}
void Corpse::Spring(NodeType n1a, NodeType n1b, NodeType n2, float distance, float dt) {
SPADES_MARK_FUNCTION_DEBUG();
SPAssert(n1a >= 0);
SPAssert(n1a < NodeCount);
SPAssert(n1b >= 0);
SPAssert(n1b < NodeCount);
SPAssert(n2 >= 0);
SPAssert(n2 < NodeCount);
Node &x = nodes[n1a];
Node &y = nodes[n1b];
Node &b = nodes[n2];
Vector3 diff = b.pos - (x.pos + y.pos) * .5f;
float dist = diff.GetLength();
Vector3 force = diff.Normalize() * (distance - dist);
force *= dt * 50.f;
b.vel += force;
force *= .5f;
x.vel -= force;
y.vel -= force;
Vector3 velMid = (x.vel + y.vel) * .25f + b.vel * .5f;
float dump = 1.f - powf(.05f, dt);
x.vel += (velMid - x.vel) * dump;
y.vel += (velMid - y.vel) * dump;
b.vel += (velMid - b.vel) * dump;
}
static float MyACos(float v) {
SPAssert(!isnan(v));
if (v >= 1.f)
return 0.f;
if (v <= -1.f)
return static_cast(M_PI);
float vv = acosf(v);
if (isnan(vv)) {
vv = acosf(v * .9999f);
}
SPAssert(!isnan(vv));
return vv;
}
void Corpse::AngleSpring(NodeType base, NodeType n1id, NodeType n2id, float minDot,
float maxDot, float dt) {
Node &nBase = nodes[base];
Node &n1 = nodes[n1id];
Node &n2 = nodes[n2id];
Vector3 d1 = n1.pos - nBase.pos;
Vector3 d2 = n2.pos - nBase.pos;
float ln1 = d1.GetLength();
float ln2 = d2.GetLength();
float dot = Vector3::Dot(d1, d2) / (ln1 * ln2 + 0.0000001f);
if (dot >= minDot && dot <= maxDot)
return;
Vector3 diff = n2.pos - n1.pos;
float strength = 0.f;
Vector3 a1 = Vector3::Cross(d1, diff);
a1 = Vector3::Cross(d1, a1).Normalize();
Vector3 a2 = Vector3::Cross(d2, diff);
a2 = Vector3::Cross(d2, a2).Normalize();
// a1=-a1; a2=-a2;
// a1 = -a1;
a2 = -a2;
if (dot > maxDot) {
strength = MyACos(dot) - MyACos(maxDot);
} else if (dot < minDot) {
strength = MyACos(dot) - MyACos(minDot);
}
SPAssert(!isnan(strength));
strength *= 20.f;
strength *= dt;
a1 *= strength;
a2 *= strength;
a2 *= 0.f;
n2.vel += a1;
n1.vel += a2;
nBase.vel -= a1 + a2;
/*
d1 += a1 * 0.01;
d2 += a2 * 0.01;
float nd = Vector3::Dot(d1, d2) / (d1.GetLength() * d2.GetLength());
if(dot > maxDot){
if(nd < dot)
printf("GOOD %f -> %f\n", dot, nd);
else
printf("BAD %f -> %f\n", dot, nd);
}else{
if(nd > dot)
printf("GOOD %f -> %f\n", dot, nd);
else
printf("BAD %f -> %f\n", dot, nd);
}*/
}
void Corpse::AngleSpring(NodeType n1id, NodeType n2id, Vector3 dir, float minDot,
float maxDot, float dt) {
Node &n1 = nodes[n1id];
Node &n2 = nodes[n2id];
Vector3 diff = n2.pos - n1.pos;
float ln1 = diff.GetLength();
float dot = Vector3::Dot(diff, dir) / (ln1 + 0.000000001f);
if (dot >= minDot && dot <= maxDot)
return;
float strength = 0.f;
Vector3 a1 = Vector3::Cross(dir, diff);
a1 = Vector3::Cross(diff, a1).Normalize();
Vector3 a2 = -a1;
// a1=-a1; a2=-a2;
// a1 = -a1;
if (dot > maxDot) {
strength = MyACos(dot) - MyACos(maxDot);
} else if (dot < minDot) {
strength = MyACos(dot) - MyACos(minDot);
}
SPAssert(!isnan(strength));
strength *= 100.f;
strength *= dt;
a1 *= strength;
a2 *= strength;
n2.vel += a1;
n1.vel += a2;
// nBase.vel -= a1 + a2;
/*
d1 += a1 * 0.01;
d2 += a2 * 0.01;
float nd = Vector3::Dot(d1, d2) / (d1.GetLength() * d2.GetLength());
if(dot > maxDot){
if(nd < dot)
printf("GOOD %f -> %f\n", dot, nd);
else
printf("BAD %f -> %f\n", dot, nd);
}else{
if(nd > dot)
printf("GOOD %f -> %f\n", dot, nd);
else
printf("BAD %f -> %f\n", dot, nd);
}*/
}
static float fractf(float v) { return v - floorf(v); }
static void CheckEscape(GameMap &map, IntVector3 hitBlock, IntVector3 a, IntVector3 b,
IntVector3 dir, float &bestDist, IntVector3 &bestDir) {
hitBlock += dir;
IntVector3 aa = a + dir;
IntVector3 bb = b + dir;
if (map.IsSolidWrapped(hitBlock.x, hitBlock.y, hitBlock.z))
return;
if (map.IsSolidWrapped(aa.x, aa.y, aa.z))
return;
if (map.IsSolidWrapped(bb.x, bb.y, bb.z))
return;
float dist;
if (dir.x == 1) {
dist = 1.f - fractf(a.x);
dist += 1.f - fractf(b.x);
} else if (dir.x == -1) {
dist = fractf(a.x);
dist += fractf(b.x);
} else if (dir.y == 1) {
dist = 1.f - fractf(a.y);
dist += 1.f - fractf(b.y);
} else if (dir.y == -1) {
dist = fractf(a.y);
dist += fractf(b.y);
} else if (dir.z == 1) {
dist = 1.f - fractf(a.z);
dist += 1.f - fractf(b.z);
} else if (dir.z == -1) {
dist = fractf(a.z);
dist += fractf(b.z);
} else {
SPAssert(false);
return;
}
if (dist < bestDist) {
bestDist = dist;
bestDir = dir;
}
}
void Corpse::LineCollision(NodeType a, NodeType b, float dt) {
if (!r_corpseLineCollision)
return;
Node &n1 = nodes[a];
Node &n2 = nodes[b];
IntVector3 hitBlock;
if (map.CastRay(n1.lastPos, n2.lastPos, 16.f, hitBlock)) {
GameMap::RayCastResult res1 = map.CastRay2(n1.lastPos, n2.lastPos - n1.lastPos, 8);
GameMap::RayCastResult res2 = map.CastRay2(n2.lastPos, n1.lastPos - n2.lastPos, 8);
if (!res1.hit)
return;
if (!res2.hit)
return;
if (res1.startSolid || res2.startSolid) {
return;
}
// really hit?
if (Vector3::Dot(res1.hitPos - n1.lastPos, n2.lastPos - n1.lastPos) >
(n2.pos - n1.pos).GetPoweredLength()) {
return;
}
if (Vector3::Dot(res1.hitPos - n1.lastPos, n2.lastPos - n1.lastPos) < 0.f) {
return;
}
if (Vector3::Dot(res2.hitPos - n2.lastPos, n1.lastPos - n2.lastPos) >
(n2.pos - n1.pos).GetPoweredLength()) {
return;
}
if (Vector3::Dot(res2.hitPos - n2.lastPos, n1.lastPos - n2.lastPos) < 0.f) {
return;
}
Vector3 blockPos;
blockPos.x = hitBlock.x + .5f;
blockPos.y = hitBlock.y + .5f;
blockPos.z = hitBlock.z + .5f;
float inlen = (res1.hitPos - res2.hitPos).GetLength();
IntVector3 ivec = {0, 0, 0};
ivec.x += res1.normal.x;
ivec.y += res1.normal.y;
ivec.z += res1.normal.z;
ivec.x += res2.normal.x;
ivec.y += res2.normal.y;
ivec.z += res2.normal.z;
Vector3 dir = {0.f, 0.f, 0.f};
if (ivec.x == 0 && ivec.y == 0 && ivec.z == 0) {
// hanging. which direction to escape?
float bestDist = 1000.f;
IntVector3 bestDir;
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(1, 0, 0), bestDist, bestDir);
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(-1, 0, 0), bestDist, bestDir);
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(0, 1, 0), bestDist, bestDir);
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(0, -1, 0), bestDist, bestDir);
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(0, 0, 1), bestDist, bestDir);
CheckEscape(map, hitBlock, n1.pos.Floor(), n2.pos.Floor(),
IntVector3::Make(0, 0, -1), bestDist, bestDir);
if (bestDist > 10.f) {
// failed to find appropriate direction.
return;
}
ivec = bestDir;
inlen = bestDist + .1f;
}
dir.x = ivec.x;
dir.y = ivec.y;
dir.z = ivec.z;
Vector3 normDir = dir; // |D|
n1.vel -= normDir * std::min(Vector3::Dot(normDir, n1.vel), 0.f);
n2.vel -= normDir * std::min(Vector3::Dot(normDir, n2.vel), 0.f);
dir *= dt * inlen * 5.f;
n1.vel += dir;
n2.vel += dir;
// friction
n1.vel -= (n1.vel - normDir * Vector3::Dot(normDir, n1.vel)) * .2f;
n2.vel -= (n2.vel - normDir * Vector3::Dot(normDir, n2.vel)) * .2f;
}
}
void Corpse::AngularMomentum(int eId, NodeType a, NodeType b) {
Edge &e = edges[eId];
e.velDiff = nodes[b].vel - nodes[a].vel;
if (e.node1 != a || e.node2 != b) {
e.lastVelDiff = e.velDiff;
e.node1 = a;
e.node2 = b;
return;
}
Vector3 force = e.lastVelDiff - e.velDiff;
force *= .5f;
nodes[b].vel += force;
nodes[a].vel -= force;
e.lastVelDiff = e.velDiff;
}
void Corpse::ApplyConstraint(float dt) {
SPADES_MARK_FUNCTION();
AngularMomentum(0, Torso1, Torso2);
AngularMomentum(1, Torso2, Torso3);
AngularMomentum(2, Torso3, Torso4);
AngularMomentum(3, Torso4, Torso1);
AngularMomentum(4, Torso1, Arm1);
AngularMomentum(5, Torso2, Arm2);
AngularMomentum(6, Torso3, Leg1);
AngularMomentum(7, Torso4, Leg2);
Spring(Torso1, Torso2, 0.8f, dt);
Spring(Torso3, Torso4, 0.8f, dt);
Spring(Torso1, Torso4, 0.9f, dt);
Spring(Torso2, Torso3, 0.9f, dt);
Spring(Torso1, Torso3, 1.204f, dt);
Spring(Torso2, Torso4, 1.204f, dt);
Spring(Arm1, Torso1, 1.f, dt);
Spring(Arm2, Torso2, 1.f, dt);
Spring(Leg1, Torso3, 1.f, dt);
Spring(Leg2, Torso4, 1.f, dt);
AngleSpring(Torso1, Arm1, Torso3, -1.f, 0.6f, dt);
AngleSpring(Torso2, Arm2, Torso4, -1.f, 0.6f, dt);
AngleSpring(Torso3, Leg1, Torso2, -1.f, -0.2f, dt);
AngleSpring(Torso4, Leg2, Torso1, -1.f, -0.2f, dt);
Spring(Torso1, Torso2, Head, .6f, dt);
/*
AngleSpring(Torso1,
Torso2, Head,
0.5f, 1.f, dt);
AngleSpring(Torso2,
Torso1, Head,
0.5f, 1.f, dt);
*/
LineCollision(Torso1, Torso2, dt);
LineCollision(Torso2, Torso3, dt);
LineCollision(Torso3, Torso4, dt);
LineCollision(Torso4, Torso1, dt);
LineCollision(Torso1, Torso3, dt);
LineCollision(Torso2, Torso4, dt);
LineCollision(Torso1, Arm1, dt);
LineCollision(Torso2, Arm2, dt);
LineCollision(Torso3, Leg1, dt);
LineCollision(Torso4, Leg2, dt);
}
void Corpse::Update(float dt) {
SPADES_MARK_FUNCTION();
float damp = 1.f;
float damp2 = 1.f;
if (dt > 0.f) {
damp = powf(.9f, dt);
damp2 = powf(.371f, dt);
}
// dt *= 0.1f;
for (int i = 0; i < NodeCount; i++) {
Node &node = nodes[i];
Vector3 oldPos = node.lastPos;
node.pos += node.vel * dt;
SPAssert(!isnan(node.pos.x));
SPAssert(!isnan(node.pos.y));
SPAssert(!isnan(node.pos.z));
if (node.pos.z > 63.f) {
node.vel.z -= dt * 6.f; // buoyancy
node.vel *= damp;
} else {
node.vel.z += dt * 32.f; // gravity
node.vel.z *= damp2;
}
// node.vel *= damp;
if (!map.ClipBox(oldPos.x, oldPos.y, oldPos.z)) {
if (map.ClipBox(node.pos.x, oldPos.y, oldPos.z)) {
node.vel.x = -node.vel.x * .2f;
if (fabsf(node.vel.x) < .3f)
node.vel.x = 0.f;
node.pos.x = oldPos.x;
node.vel.y *= .5f;
node.vel.z *= .5f;
}
if (map.ClipBox(node.pos.x, node.pos.y, oldPos.z)) {
node.vel.y = -node.vel.y * .2f;
if (fabsf(node.vel.y) < .3f)
node.vel.y = 0.f;
node.pos.y = oldPos.y;
node.vel.x *= .5f;
node.vel.z *= .5f;
}
if (map.ClipBox(node.pos.x, node.pos.y, node.pos.z)) {
node.vel.z = -node.vel.z * .2f;
if (fabsf(node.vel.z) < .3f)
node.vel.z = 0.f;
node.pos.z = oldPos.z;
node.vel.x *= .5f;
node.vel.y *= .5f;
}
if (map.ClipBox(node.pos.x, node.pos.y, node.pos.z)) {
// TODO: getting out block
// node.pos = oldPos;
// node.vel *= .5f;
}
}
/*
if(map.ClipBox(node.pos.x,
node.pos.y,
node.pos.z)){
if(!map.ClipBox(node.pos.x,
node.pos.y,
oldPos.z)){
node.vel.z = -node.vel.z * .2f;
if(fabsf(node.vel.z) < .3f)
node.vel.z = 0.f;
node.pos.z = oldPos.z;
}
if(!map.ClipBox(node.pos.x,
oldPos.y,
node.pos.z)){
node.vel.y = -node.vel.y * .2f;
if(fabsf(node.vel.y) < .3f)
node.vel.y = 0.f;
node.pos.y = oldPos.y;
}
if(!map.ClipBox(oldPos.x,
node.pos.y,
node.pos.z)){
node.vel.x = -node.vel.x * .2f;
if(fabsf(node.vel.x) < .3f)
node.vel.x = 0.f;
node.pos.x = oldPos.x;
}
node.vel *= .8f;
//node.pos = oldPos;
if(node.vel.GetLength() < .02f){
node.vel *= 0.f;
}
}*/
node.lastPos = node.pos;
node.lastForce = node.vel;
}
ApplyConstraint(dt);
for (int i = 0; i < NodeCount; i++) {
nodes[i].lastForce = nodes[i].vel - nodes[i].lastForce;
}
}
void Corpse::AddToScene() {
ModelRenderParam param;
param.customColor = color;
Handle model;
Matrix4 scaler = Matrix4::Scale(.1f);
// draw torso
Matrix4 torso;
Vector3 tX, tY;
{
Vector3 tX1 = nodes[Torso1].pos - nodes[Torso2].pos;
Vector3 tX2 = nodes[Torso4].pos - nodes[Torso3].pos;
Vector3 tY1 = nodes[Torso1].pos + nodes[Torso2].pos;
Vector3 tY2 = nodes[Torso4].pos + nodes[Torso3].pos;
tX = ((tX1 + tX2) * .5f).Normalize();
tY = ((tY2 - tY1) * .5f).Normalize();
Vector3 tZ = Vector3::Cross(tX, tY).Normalize();
tY = Vector3::Cross(tX, tZ).Normalize();
Vector3 tOrigin = tY1 * .5f;
torso = Matrix4::FromAxis(tX, -tZ, -tY, tOrigin);
param.matrix = torso * scaler;
model = renderer.RegisterModel("Models/Player/Torso.kv6");
renderer.RenderModel(*model, param);
}
// draw Head
{
Vector3 headBase = (torso * MakeVector3(0.0f, 0.f, 0.f)).GetXYZ();
model = renderer.RegisterModel("Models/Player/Head.kv6");
Vector3 aX, aY, aZ;
Vector3 center = (nodes[Torso1].pos + nodes[Torso2].pos) * .5f;
aZ = nodes[Head].pos - center;
aZ = -torso.GetAxis(2);
aZ = aZ.Normalize();
aY = nodes[Torso2].pos - nodes[Torso1].pos;
aY = Vector3::Cross(aY, aZ).Normalize();
aX = Vector3::Cross(aY, aZ).Normalize();
param.matrix = Matrix4::FromAxis(-aX, aY, -aZ, headBase) * scaler;
renderer.RenderModel(*model, param);
}
// draw Arms
{
Vector3 arm1Base = (torso * MakeVector3(0.4f, 0.f, 0.2f)).GetXYZ();
Vector3 arm2Base = (torso * MakeVector3(-0.4f, 0.f, 0.2f)).GetXYZ();
model = renderer.RegisterModel("Models/Player/Arm.kv6");
Vector3 aX, aY, aZ;
aZ = nodes[Arm1].pos - nodes[Torso1].pos;
aZ = aZ.Normalize();
aY = nodes[Torso2].pos - nodes[Torso1].pos;
aY = Vector3::Cross(aY, aZ).Normalize();
aX = Vector3::Cross(aY, aZ).Normalize();
param.matrix = Matrix4::FromAxis(aX, aY, aZ, arm1Base) * scaler;
renderer.RenderModel(*model, param);
aZ = nodes[Arm2].pos - nodes[Torso2].pos;
aZ = aZ.Normalize();
aY = nodes[Torso1].pos - nodes[Torso2].pos;
aY = Vector3::Cross(aY, aZ).Normalize();
aX = Vector3::Cross(aY, aZ).Normalize();
param.matrix = Matrix4::FromAxis(aX, aY, aZ, arm2Base) * scaler;
renderer.RenderModel(*model, param);
}
// draw Leg
{
Vector3 leg1Base = (torso * MakeVector3(0.25f, 0.f, 0.9f)).GetXYZ();
Vector3 leg2Base = (torso * MakeVector3(-0.25f, 0.f, 0.9f)).GetXYZ();
model = renderer.RegisterModel("Models/Player/Leg.kv6");
Vector3 aX, aY, aZ;
aZ = nodes[Leg1].pos - nodes[Torso3].pos;
aZ = aZ.Normalize();
aY = nodes[Torso1].pos - nodes[Torso2].pos;
aY = Vector3::Cross(aY, aZ).Normalize();
aX = Vector3::Cross(aY, aZ).Normalize();
param.matrix = Matrix4::FromAxis(aX, aY, aZ, leg1Base) * scaler;
renderer.RenderModel(*model, param);
aZ = nodes[Leg2].pos - nodes[Torso4].pos;
aZ = aZ.Normalize();
aY = nodes[Torso1].pos - nodes[Torso2].pos;
aY = Vector3::Cross(aY, aZ).Normalize();
aX = Vector3::Cross(aY, aZ).Normalize();
param.matrix = Matrix4::FromAxis(aX, aY, aZ, leg2Base) * scaler;
renderer.RenderModel(*model, param);
}
}
Vector3 Corpse::GetCenter() {
Vector3 v = {0, 0, 0};
for (int i = 0; i < NodeCount; i++)
v += nodes[i].pos;
v *= 1.f / (float)NodeCount;
return v;
}
bool Corpse::IsVisibleFrom(spades::Vector3 eye) {
// distance culled?
if ((eye - GetCenter()).GetLength() > 150.f)
return false;
for (int i = 0; i < NodeCount; i++) {
IntVector3 outBlk;
if (map.CastRay(eye, nodes[i].pos, 256.f, outBlk))
return true;
}
return false;
}
void Corpse::AddImpulse(spades::Vector3 v) {
for (int i = 0; i < NodeCount; i++)
nodes[i].vel += v;
}
} // namespace client
} // namespace spades