#if defined(DEVEL) && defined(USE_DIRECTX) #include "penetration_display.hpp" #include "controller.hpp" #include "game.hpp" namespace ace { namespace vehicledamage { namespace debug { penetration_display::penetration_display() : dispatcher() { _active_vehicle = nullptr; add("register_vehicle", std::bind(&ace::vehicledamage::debug::penetration_display::register_vehicle, this, std::placeholders::_1, std::placeholders::_2)); add("show_hit", std::bind(&ace::vehicledamage::debug::penetration_display::show_hit, this, std::placeholders::_1, std::placeholders::_2)); } bool penetration_display::show_hit(const arguments &args, std::string &result) { std::lock_guard _lock(_render_lock); _active_hits.push_back(gamehit::create(args)); return true; } bool penetration_display::register_vehicle(const arguments &args, std::string &result) { std::lock_guard _lock(_render_lock); uint32_t id = args[0]; _active_vehicle = ace::vehicledamage::controller::get().vehicles[id]; return true; } bool penetration_display::init() { HRESULT hr = S_OK; d3d_display::init(); std::lock_guard _lock(_render_lock); _States.reset(new CommonStates(_pd3dDevice)); _FXFactory.reset(new EffectFactory(_pd3dDevice)); _Batch.reset(new PrimitiveBatch(_pImmediateContext)); //_Font.reset(new SpriteFont(_pd3dDevice, L"italic.spritefont")); _BatchEffect.reset(new BasicEffect(_pd3dDevice)); _BatchEffect->SetVertexColorEnabled(true); { void const* shaderByteCode; size_t byteCodeLength; _BatchEffect->GetVertexShaderBytecode(&shaderByteCode, &byteCodeLength); hr = _pd3dDevice->CreateInputLayout(VertexPositionColor::InputElements, VertexPositionColor::InputElementCount, shaderByteCode, byteCodeLength, &_pBatchInputLayout); if (FAILED(hr)) return hr; } _BatchEffect->SetView(XMLoadFloat4x4(&_View)); _BatchEffect->SetProjection(XMLoadFloat4x4(&_Projection)); } bool penetration_display::step(void) { _BatchEffect->Apply(_pImmediateContext); _pImmediateContext->IASetInputLayout(_pBatchInputLayout); const XMVECTORF32 xaxis = { 20.f, 0.f, 0.f }; const XMVECTORF32 yaxis = { 0.f, 0.f, 20.f }; DrawGrid(*_Batch, xaxis, yaxis, g_XMZero, 20, 20, Colors::Gray); if (_active_vehicle) { DrawObject(_active_vehicle->fire_lod, *_Batch, *_active_vehicle->object, Colors::Gray); } if (_active_hits.size() > 0) { DrawHits(0, *_Batch, Colors::Red); for (auto & hit : _active_hits) { std::vector> collisions; _active_vehicle->surface_raycast(hit->impactposition, hit->impactvelocity, collisions); DrawCollisions(collisions, *_Batch, Colors::Purple); } } _BatchEffect->SetView(XMLoadFloat4x4(&_View)); _BatchEffect->SetProjection(XMLoadFloat4x4(&_Projection)); return true; } void penetration_display::DrawCollisions(const std::vector> & collisions, PrimitiveBatch& batch, GXMVECTOR color) { batch.Begin(); float scale = 0.01f; for (auto & collision : collisions) { XMFLOAT3 A = { collision.x() + scale, collision.y()+scale, collision.z()+scale }; XMFLOAT3 B = { collision.x(), collision.y(), collision.z() }; XMFLOAT3 C = { collision.x()-scale, collision.y()-scale, collision.z() - scale }; VertexPositionColor v1(XMLoadFloat3(&A), color); VertexPositionColor v2(XMLoadFloat3(&B), color); VertexPositionColor v3(XMLoadFloat3(&C), color); batch.DrawLine(v1, v2); batch.DrawLine(v2, v3); batch.DrawLine(v3, v1); //batch.DrawTriangle(v1, v2, v3); } batch.End(); } void penetration_display::DrawHits(uint32_t lod, PrimitiveBatch& batch, GXMVECTOR color) { batch.Begin(); for (gamehit_p & hit : _active_hits) { ace::vector3 hit_from, hit_to; hit_from = hit->impactposition; hit_to = hit_from + (hit->impactvelocity * 0.005f); XMVECTORF32 from = { hit_from.x(), hit_from.y(), hit_from.z() }; XMVECTORF32 to = { hit_to.x(), hit_to.y(), hit_to.z() }; XMVECTORF32 to_a1 = { hit_to.x() + 0.5, hit_to.y() + 0.5, hit_to.z() }; XMVECTORF32 to_a2 = { hit_to.x() - 0.5, hit_to.y() - 0.5, hit_to.z() }; VertexPositionColor v1(from, color); VertexPositionColor v2(to, color); VertexPositionColor a1(to_a2, color); VertexPositionColor a2(to_a1, color); batch.DrawLine(v1, v2); batch.DrawLine(a1, v2); batch.DrawLine(a2, v2); } batch.End(); } void penetration_display::DrawObject(uint32_t lod, PrimitiveBatch& batch, ace::simulation::object & obj, GXMVECTOR color) { batch.Begin(); for (auto & face : obj.lods[lod]->faces) { ace::vector3 vertices[3]; vertices[0] = { face->vertices[0]->x(), face->vertices[0]->y(), face->vertices[0]->z() }; vertices[1] = { face->vertices[1]->x(), face->vertices[1]->y(), face->vertices[1]->z() }; vertices[2] = { face->vertices[2]->x(), face->vertices[2]->y(), face->vertices[2]->z() }; XMVECTORF32 v[3] = { { vertices[0].x(), vertices[0].y(), vertices[0].z() }, { vertices[1].x(), vertices[1].y(), vertices[1].z() }, { vertices[2].x(), vertices[2].y(), vertices[2].z() } }; //v[0] = v[0] + obj VertexPositionColor v1(v[0], color); VertexPositionColor v2(v[1], color); VertexPositionColor v3(v[2], color); batch.DrawLine(v1, v2); batch.DrawLine(v2, v3); batch.DrawLine(v3, v1); //batch.DrawTriangle(v1, v2, v3); } batch.End(); } void penetration_display::DrawGrid(PrimitiveBatch& batch, FXMVECTOR xAxis, FXMVECTOR yAxis, FXMVECTOR origin, size_t xdivs, size_t ydivs, GXMVECTOR color) { batch.Begin(); xdivs = std::max(1, xdivs); ydivs = std::max(1, ydivs); for (size_t i = 0; i <= xdivs; ++i) { float fPercent = float(i) / float(xdivs); fPercent = (fPercent * 2.0f) - 1.0f; XMVECTOR vScale = XMVectorScale(xAxis, fPercent); vScale = XMVectorAdd(vScale, origin); VertexPositionColor v1(XMVectorSubtract(vScale, yAxis), color); VertexPositionColor v2(XMVectorAdd(vScale, yAxis), color); batch.DrawLine(v1, v2); } for (size_t i = 0; i <= ydivs; i++) { FLOAT fPercent = float(i) / float(ydivs); fPercent = (fPercent * 2.0f) - 1.0f; XMVECTOR vScale = XMVectorScale(yAxis, fPercent); vScale = XMVectorAdd(vScale, origin); VertexPositionColor v1(XMVectorSubtract(vScale, xAxis), color); VertexPositionColor v2(XMVectorAdd(vScale, xAxis), color); batch.DrawLine(v1, v2); } batch.End(); } } } } #endif