mirror of
https://gitea.wildfiregames.com/0ad/0ad
synced 2026-06-17 05:44:08 -07:00
Atlas: Added simulation play/pause/reset controls; automatically plays while recording cinematics. FFmpeg: Fixed broken output files. MapReader: Fixed entity loading. ObjectHandlers: Made unit preview more robust when the preview unit gets destroyed. Various: Replaced manual matrix construction with SetYRotation call. Turned some more CStr8 back into CStr. h_mgr: Optimisation - don't calculate slow debug-output strings if they're never going to be seen (since it takes a few hundred milliseconds). TerrainRenderer: Made more tolerant of accidental negative times. Entity: Fixed m_refd being out of date when deleteAll is called. Fixed problems when doing an initializeAll...deleteAll...initializeAll sequence. SCN: Removed non-useful AoE3Ed code that never did anything. SVNLog: Made output more valid and made titles more descriptive, so it works properly in FF's live bookmarks. This was SVN commit r4779.
388 lines
8.5 KiB
C++
388 lines
8.5 KiB
C++
#include "precompiled.h"
|
|
|
|
#include "View.h"
|
|
|
|
#include "ActorViewer.h"
|
|
#include "GameLoop.h"
|
|
#include "Messages.h"
|
|
|
|
#include "graphics/CinemaTrack.h"
|
|
#include "graphics/GameView.h"
|
|
#include "graphics/Model.h"
|
|
#include "graphics/ObjectBase.h"
|
|
#include "graphics/SColor.h"
|
|
#include "graphics/Unit.h"
|
|
#include "graphics/UnitManager.h"
|
|
#include "ps/Game.h"
|
|
#include "ps/GameSetup/GameSetup.h"
|
|
#include "ps/Player.h"
|
|
#include "renderer/Renderer.h"
|
|
#include "simulation/Entity.h"
|
|
#include "simulation/EntityManager.h"
|
|
#include "simulation/EntityTemplate.h"
|
|
#include "simulation/EntityTemplateCollection.h"
|
|
#include "simulation/Simulation.h"
|
|
|
|
extern void (*Atlas_GLSwapBuffers)(void* context);
|
|
|
|
extern int g_xres, g_yres;
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
void View::SetParam(const std::wstring& UNUSED(name), bool UNUSED(value))
|
|
{
|
|
}
|
|
|
|
void View::SetParam(const std::wstring& UNUSED(name), const AtlasMessage::Colour& UNUSED(value))
|
|
{
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
ViewActor::ViewActor()
|
|
: m_SpeedMultiplier(1.f), m_ActorViewer(new ActorViewer())
|
|
{
|
|
}
|
|
|
|
ViewActor::~ViewActor()
|
|
{
|
|
delete m_ActorViewer;
|
|
}
|
|
|
|
void ViewActor::Update(float frameLength)
|
|
{
|
|
m_ActorViewer->Update(frameLength * m_SpeedMultiplier);
|
|
}
|
|
|
|
void ViewActor::Render()
|
|
{
|
|
SViewPort vp = { 0, 0, g_xres, g_yres };
|
|
CCamera& camera = GetCamera();
|
|
camera.SetViewPort(&vp);
|
|
camera.SetProjection(CGameView::defaultNear, CGameView::defaultFar, CGameView::defaultFOV);
|
|
camera.UpdateFrustum();
|
|
|
|
m_ActorViewer->Render();
|
|
Atlas_GLSwapBuffers((void*)g_GameLoop->glCanvas);
|
|
}
|
|
|
|
CCamera& ViewActor::GetCamera()
|
|
{
|
|
return m_Camera;
|
|
}
|
|
|
|
CUnit* ViewActor::GetUnit(AtlasMessage::ObjectID UNUSED(id))
|
|
{
|
|
return m_ActorViewer->GetUnit();
|
|
}
|
|
|
|
bool ViewActor::WantsHighFramerate()
|
|
{
|
|
if (m_SpeedMultiplier != 0.f && m_ActorViewer->HasAnimation())
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void ViewActor::SetSpeedMultiplier(float speed)
|
|
{
|
|
m_SpeedMultiplier = speed;
|
|
}
|
|
|
|
ActorViewer& ViewActor::GetActorViewer()
|
|
{
|
|
return *m_ActorViewer;
|
|
}
|
|
|
|
void ViewActor::SetParam(const std::wstring& name, bool value)
|
|
{
|
|
if (name == L"wireframe")
|
|
g_Renderer.SetModelRenderMode(value ? WIREFRAME : SOLID);
|
|
else if (name == L"walk")
|
|
m_ActorViewer->SetWalkEnabled(value);
|
|
else if (name == L"ground")
|
|
m_ActorViewer->SetGroundEnabled(value);
|
|
else if (name == L"shadows")
|
|
m_ActorViewer->SetShadowsEnabled(value);
|
|
else if (name == L"stats")
|
|
m_ActorViewer->SetStatsEnabled(value);
|
|
}
|
|
|
|
void ViewActor::SetParam(const std::wstring& name, const AtlasMessage::Colour& value)
|
|
{
|
|
if (name == L"background")
|
|
{
|
|
m_ActorViewer->SetBackgroundColour(SColor4ub(value.r, value.g, value.b, 255));
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
namespace AtlasMessage
|
|
{
|
|
extern void AtlasRenderSelection();
|
|
}
|
|
|
|
struct SimState
|
|
{
|
|
struct Entity
|
|
{
|
|
CStrW templateName;
|
|
int unitID;
|
|
std::set<CStr> selections;
|
|
int playerID;
|
|
CVector3D position;
|
|
float angle;
|
|
};
|
|
|
|
struct Nonentity
|
|
{
|
|
CStrW actorName;
|
|
int unitID;
|
|
std::set<CStr> selections;
|
|
CVector3D position;
|
|
float angle;
|
|
};
|
|
|
|
bool onlyEntities;
|
|
std::vector<Entity> entities;
|
|
std::vector<Nonentity> nonentities;
|
|
};
|
|
|
|
template<typename T, typename S>
|
|
static void delete_pair_2nd(std::pair<T,S> v)
|
|
{
|
|
delete v.second;
|
|
}
|
|
|
|
ViewGame::ViewGame()
|
|
: m_SpeedMultiplier(0.f)
|
|
{
|
|
debug_assert(g_Game);
|
|
}
|
|
|
|
ViewGame::~ViewGame()
|
|
{
|
|
std::for_each(m_SavedStates.begin(), m_SavedStates.end(), delete_pair_2nd<std::wstring, SimState*>);
|
|
}
|
|
|
|
void ViewGame::Update(float frameLength)
|
|
{
|
|
if (m_SpeedMultiplier == 0.f)
|
|
{
|
|
// TODO: I don't think this line is necessary, but I'm not positive...
|
|
// g_EntityManager.updateAll(0);
|
|
// Update unit interpolation
|
|
g_Game->GetSimulation()->Update(0.0);
|
|
}
|
|
else
|
|
{
|
|
// Update the whole world
|
|
g_Game->Update(m_SpeedMultiplier * frameLength);
|
|
|
|
if (g_Game->GetView()->GetCinema()->IsPlaying())
|
|
g_Game->GetView()->GetCinema()->Update(m_SpeedMultiplier * frameLength);
|
|
}
|
|
}
|
|
|
|
void ViewGame::Render()
|
|
{
|
|
SViewPort vp = { 0, 0, g_xres, g_yres };
|
|
CCamera& camera = GetCamera();
|
|
camera.SetViewPort(&vp);
|
|
camera.SetProjection(CGameView::defaultNear, CGameView::defaultFar, CGameView::defaultFOV);
|
|
camera.UpdateFrustum();
|
|
|
|
::Render();
|
|
AtlasMessage::AtlasRenderSelection();
|
|
Atlas_GLSwapBuffers((void*)g_GameLoop->glCanvas);
|
|
}
|
|
|
|
CCamera& ViewGame::GetCamera()
|
|
{
|
|
return *g_Game->GetView()->GetCamera();
|
|
}
|
|
|
|
CUnit* ViewGame::GetUnit(AtlasMessage::ObjectID id)
|
|
{
|
|
return g_Game->GetWorld()->GetUnitManager().FindByID(id);
|
|
}
|
|
|
|
bool ViewGame::WantsHighFramerate()
|
|
{
|
|
if (g_Game->GetView()->GetCinema()->IsPlaying())
|
|
return true;
|
|
|
|
if (m_SpeedMultiplier != 0.f)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void ViewGame::SetSpeedMultiplier(float speed)
|
|
{
|
|
m_SpeedMultiplier = speed;
|
|
}
|
|
|
|
void ViewGame::SaveState(const std::wstring& label, bool onlyEntities)
|
|
{
|
|
SimState* simState = new SimState();
|
|
simState->onlyEntities = onlyEntities;
|
|
|
|
CUnitManager& unitMan = g_Game->GetWorld()->GetUnitManager();
|
|
const std::vector<CUnit*>& units = unitMan.GetUnits();
|
|
|
|
for (std::vector<CUnit*>::const_iterator unit = units.begin(); unit != units.end(); ++unit)
|
|
{
|
|
CEntity* entity = (*unit)->GetEntity();
|
|
|
|
// Ignore objects that aren't entities
|
|
if (! entity)
|
|
continue;
|
|
|
|
SimState::Entity e;
|
|
e.templateName = entity->m_base->m_Tag;
|
|
e.unitID = (*unit)->GetID();
|
|
e.selections = (*unit)->GetActorSelections();
|
|
e.playerID = entity->GetPlayer()->GetPlayerID();
|
|
e.position = entity->m_position;
|
|
e.angle = entity->m_orientation.Y;
|
|
|
|
// TODO: preserve random actor variations
|
|
// TODO: preserve IDs
|
|
|
|
simState->entities.push_back(e);
|
|
}
|
|
|
|
if (! onlyEntities)
|
|
{
|
|
for (std::vector<CUnit*>::const_iterator unit = units.begin(); unit != units.end(); ++unit) {
|
|
|
|
// Ignore objects that are entities
|
|
if ((*unit)->GetEntity())
|
|
continue;
|
|
|
|
SimState::Nonentity n;
|
|
n.actorName = (*unit)->GetObject()->m_Base->m_Name;
|
|
n.unitID = (*unit)->GetID();
|
|
n.selections = (*unit)->GetActorSelections();
|
|
n.position = (*unit)->GetModel()->GetTransform().GetTranslation();
|
|
CVector3D orient = (*unit)->GetModel()->GetTransform().GetIn();
|
|
n.angle = atan2(-orient.X, -orient.Z);
|
|
|
|
simState->nonentities.push_back(n);
|
|
}
|
|
}
|
|
|
|
delete m_SavedStates[label]; // in case it already exists
|
|
m_SavedStates[label] = simState;
|
|
}
|
|
|
|
void ViewGame::RestoreState(const std::wstring& label)
|
|
{
|
|
SimState* simState = m_SavedStates[label];
|
|
if (! simState)
|
|
return;
|
|
|
|
CUnitManager& unitMan = g_Game->GetWorld()->GetUnitManager();
|
|
|
|
// delete all existing entities
|
|
g_EntityManager.deleteAll();
|
|
|
|
if (! simState->onlyEntities)
|
|
{
|
|
// delete all remaining non-entity units
|
|
unitMan.DeleteAll();
|
|
// don't reset the unitID counter - there's no need, since it'll work alright anyway
|
|
}
|
|
|
|
for (size_t i = 0; i < simState->entities.size(); ++i)
|
|
{
|
|
SimState::Entity& e = simState->entities[i];
|
|
|
|
CEntityTemplate* base = g_EntityTemplateCollection.getTemplate(e.templateName, g_Game->GetPlayer(e.playerID));
|
|
if (base)
|
|
{
|
|
HEntity ent = g_EntityManager.create(base, e.position, e.angle, e.selections);
|
|
|
|
if (ent)
|
|
{
|
|
ent->m_actor->SetPlayerID(e.playerID);
|
|
ent->m_actor->SetID(e.unitID);
|
|
}
|
|
}
|
|
}
|
|
|
|
g_EntityManager.InitializeAll();
|
|
|
|
if (simState->onlyEntities)
|
|
return;
|
|
|
|
for (size_t i = 0; i < simState->nonentities.size(); ++i)
|
|
{
|
|
SimState::Nonentity& n = simState->nonentities[i];
|
|
|
|
CUnit* unit = unitMan.CreateUnit(n.actorName, NULL, n.selections);
|
|
|
|
if (unit)
|
|
{
|
|
CMatrix3D m;
|
|
m.SetYRotation(n.angle + PI);
|
|
m.Translate(n.position);
|
|
unit->GetModel()->SetTransform(m);
|
|
|
|
unit->SetID(n.unitID);
|
|
}
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
ViewNone* view_None = NULL;
|
|
ViewGame* view_Game = NULL;
|
|
ViewActor* view_Actor = NULL;
|
|
|
|
View::~View()
|
|
{
|
|
}
|
|
|
|
View* View::GetView(int /*eRenderView*/ view)
|
|
{
|
|
if (view == AtlasMessage::eRenderView::NONE) return View::GetView_None();
|
|
else if (view == AtlasMessage::eRenderView::GAME) return View::GetView_Game();
|
|
else if (view == AtlasMessage::eRenderView::ACTOR) return View::GetView_Actor();
|
|
else
|
|
{
|
|
debug_warn("Invalid view type");
|
|
return View::GetView_None();
|
|
}
|
|
}
|
|
|
|
View* View::GetView_None()
|
|
{
|
|
if (! view_None)
|
|
view_None = new ViewNone();
|
|
return view_None;
|
|
}
|
|
|
|
ViewGame* View::GetView_Game()
|
|
{
|
|
if (! view_Game)
|
|
view_Game = new ViewGame();
|
|
return view_Game;
|
|
}
|
|
|
|
ViewActor* View::GetView_Actor()
|
|
{
|
|
if (! view_Actor)
|
|
view_Actor = new ViewActor();
|
|
return view_Actor;
|
|
}
|
|
|
|
void View::DestroyViews()
|
|
{
|
|
delete view_None; view_None = NULL;
|
|
delete view_Game; view_Game = NULL;
|
|
delete view_Actor; view_Actor = NULL;
|
|
}
|