warzone2100/src/mapdisplay.cpp

181 lines
4.8 KiB
C++

/*
This file is part of Warzone 2100.
Copyright (C) 1999-2004 Eidos Interactive
Copyright (C) 2005-2013 Warzone 2100 Project
Warzone 2100 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 2 of the License, or
(at your option) any later version.
Warzone 2100 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 Warzone 2100; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
/*
MapDisplay - Renders the world view necessary for the intelligence map
Alex McLean, Pumpkin Studios, EIDOS Interactive, 1997
Makes heavy use of the functions available in display3d.c. Could have
messed about with display3d.c to make to world render dual purpose, but
it's neater as a separate file, as the intelligence map has special requirements
and overlays and needs to render to a specified buffer for later use.
*/
#include "lib/framework/frame.h"
#include "lib/ivis_opengl/piematrix.h"
#include "component.h"
#include "intdisplay.h"
#include "mapdisplay.h"
#define ROTATE_TIME (2*GAME_TICKS_PER_SEC)
/* renders the Research IMDs into the surface - used by message display in
Intelligence Map */
void renderResearchToBuffer(RESEARCH *psResearch, UDWORD OriginX, UDWORD OriginY)
{
UDWORD angle = 0;
BASE_STATS *psResGraphic;
UDWORD compID, IMDType;
Vector3i Rotation,Position;
UDWORD basePlateSize, Radius;
SDWORD scale = 0;
// Set identity (present) context
pie_MatBegin();
pie_SetGeometricOffset(OriginX+10, OriginY+10);
// Rotate round
// full rotation once every 2 seconds..
angle = (realTime % ROTATE_TIME) * 360 / ROTATE_TIME;
Position.x = 0;
Position.y = 0;
Position.z = BUTTON_DEPTH;
// Rotate round
Rotation.x = -30;
Rotation.y = angle;
Rotation.z = 0;
//draw the IMD for the research
if (psResearch->psStat)
{
//we have a Stat associated with this research topic
if (StatIsStructure(psResearch->psStat))
{
//this defines how the button is drawn
IMDType = IMDTYPE_STRUCTURESTAT;
psResGraphic = psResearch->psStat;
//set up the scale
basePlateSize= getStructureStatSizeMax((STRUCTURE_STATS*)psResearch->psStat);
if(basePlateSize == 1)
{
scale = RESEARCH_COMPONENT_SCALE / 2;
/*HACK HACK HACK!
if its a 'tall thin (ie tower)' structure stat with something on
the top - offset the position to show the object on top*/
if (((STRUCTURE_STATS*)psResearch->psStat)->pIMD[0]->nconnectors &&
getStructureStatHeight((STRUCTURE_STATS*)psResearch->psStat) > TOWER_HEIGHT)
{
Position.y -= 30;
}
}
else if(basePlateSize == 2)
{
scale = RESEARCH_COMPONENT_SCALE / 4;
}
else
{
scale = RESEARCH_COMPONENT_SCALE / 5;
}
}
else
{
compID = StatIsComponent(psResearch->psStat);
if (compID != COMP_NUMCOMPONENTS)
{
//this defines how the button is drawn
IMDType = IMDTYPE_COMPONENT;
psResGraphic = psResearch->psStat;
// FIXME: Another kludge to deal with the superTransport to make it "fit" the display.
if (psResearch->id.compare("R-SuperTransport") == 0)
{
scale = RESEARCH_COMPONENT_SCALE / 3;
}
else if (psResearch->id.compare("R-Cyborg-Transport") == 0)
{
scale = RESEARCH_COMPONENT_SCALE / 2;
}
else
{
scale = RESEARCH_COMPONENT_SCALE;
}
}
else
{
ASSERT( false, "intDisplayMessageButton: invalid stat" );
IMDType = IMDTYPE_RESEARCH;
psResGraphic = (BASE_STATS *)psResearch;
}
}
}
else
{
//no Stat for this research topic so use the research topic to define what is drawn
psResGraphic = (BASE_STATS *)psResearch;
IMDType = IMDTYPE_RESEARCH;
}
//scale the research according to size of IMD
if (IMDType == IMDTYPE_RESEARCH)
{
Radius = getResearchRadius((BASE_STATS*)psResGraphic);
if(Radius <= 100)
{
scale = RESEARCH_COMPONENT_SCALE / 2;
}
else if(Radius <= 128)
{
scale = RESEARCH_COMPONENT_SCALE / 3;
}
else if(Radius <= 256)
{
scale = RESEARCH_COMPONENT_SCALE / 4;
}
else
{
scale = RESEARCH_COMPONENT_SCALE / 5;
}
}
/* display the IMDs */
if(IMDType == IMDTYPE_COMPONENT)
{
displayComponentButton(psResGraphic, &Rotation, &Position, scale);
}
else if(IMDType == IMDTYPE_RESEARCH)
{
displayResearchButton(psResGraphic, &Rotation, &Position, scale);
}
else if(IMDType == IMDTYPE_STRUCTURESTAT)
{
displayStructureStatButton((STRUCTURE_STATS *)psResGraphic, &Rotation, &Position, scale);
}
else
{
ASSERT( false, "Unknown PIEType" );
}
// close matrix context
pie_MatEnd();
}