//======== (C) Copyright 2001 Charles G. Cleveland All rights reserved. ========= // // The copyright to the contents herein is the property of Charles G. Cleveland. // The contents may be used and/or copied only with the written permission of // Charles G. Cleveland, or in accordance with the terms and conditions stipulated in // the agreement/contract under which the contents have been supplied. // // Purpose: All graphical HUD operations // // $Workfile: AvHHudRender.cpp $ // $Date: 2002/10/24 21:31:13 $ // //------------------------------------------------------------------------------- // $Log: AvHHudRender.cpp,v $ // Revision 1.25 2002/10/24 21:31:13 Flayra // - Removed some parts of energy drawing (unneeded clutter) // - New marine HUD artwork (mainhud.spr) // - Reworked upgrade drawing to only draw one of a type of upgrade (this was the last thing for v1.0, so it's a bit yucky) // // Revision 1.24 2002/10/18 22:20:14 Flayra // - Fixed alien mass crash when a player left the server // // Revision 1.23 2002/10/16 00:59:33 Flayra // - Added titles for umbra and primal scream // - Don't draw building circles for entities on the other team (not even for commander) // - Tried drawing building circles for ghost buildings, but it was confusing // // Revision 1.22 2002/10/03 18:56:10 Flayra // - Moved alien energy to fuser3 // - Changed limits for energy and resources // - Draw order icons centered around order position // - Don't draw health rings for opposing teams // // Revision 1.21 2002/09/25 20:48:37 Flayra // - Allow more UI to draw when gestating // - Only draw text for blip closest to reticle // - Don't draw stuff when dead // - Changed order blinking // // Revision 1.20 2002/09/23 22:19:58 Flayra // - Added "paralyzed" indicator // - HUD element repositioning and refactoring // - Added alien build circles // - Added visible motion-tracking sprite to marine HUD // - Removed special siege sprite // // Revision 1.19 2002/09/09 19:57:33 Flayra // - Fixed black edges in D3D // - Added blinking "webbed" indicator // - Refactored UI constants // - Fixed help icons // - Added hive info indicator // - Draw more info as spectator // // Revision 1.18 2002/08/31 18:01:01 Flayra // - Work at VALVe // // Revision 1.17 2002/08/16 02:38:44 Flayra // - Draw "webbed" message // - Draw health for buildings and players // - Removed old overwatch code // // Revision 1.16 2002/08/09 01:03:36 Flayra // - Started refactoring for moving variable sprite hole drawing into TriAPI // // Revision 1.15 2002/08/02 21:56:54 Flayra // - Added reticle help, new tooltip system and much nicer order drawing! Also changed jetpack label to use font, and refactored some sprite names. // // Revision 1.14 2002/07/26 23:05:06 Flayra // - Generate numerical feedback for damage events // // Revision 1.13 2002/07/23 17:09:41 Flayra // - Add ability to centered, translated strings, visually-smooth energy level, new hive sight info, draw parasited message, draw marine upgrades, new method of drawing alien upgrades (overlapping and offset) // // Revision 1.12 2002/07/10 14:42:26 Flayra // - Removed cl_particleinfo drawing differences // // Revision 1.11 2002/07/08 17:07:56 Flayra // - Started to add display of marine upgrade sprite, fixed bug where building indicators aren't displayed after a map change, info_location drawing changes, primal scream color tweak, removed old hive drawing code // // Revision 1.10 2002/07/01 21:36:23 Flayra // - Added primal scream effect, added building ranges for ghost buildings, removed outdated code, removed mapping build sprite, call vidinit() on hive sight sprites // // Revision 1.9 2002/06/25 18:03:09 Flayra // - Added info_locations, removed old weapon help system, added smooth resource swelling, lots of alien UI usability changes, fixed problem with ghost building // // Revision 1.8 2002/06/10 19:57:01 Flayra // - Allow drawing just a portion of a texture when scaling it, draw team hierarchy for soldiers // // Revision 1.7 2002/06/03 16:49:20 Flayra // - Help sprites moved into one animated sprite // // Revision 1.6 2002/05/28 17:49:06 Flayra // - Hive sight sprite changes // // Revision 1.5 2002/05/23 02:33:42 Flayra // - Post-crash checkin. Restored @Backup from around 4/16. Contains changes for last four weeks of development. // //=============================================================================== #include "AvHHud.h" #include "cl_dll/hud.h" #include "cl_dll/cl_util.h" #include "AvHConstants.h" #include "AvHClientVariables.h" #include "AvHSpecials.h" #include "../common/cl_entity.h" #include "AvHTitles.h" #include "../pm_shared/pm_debug.h" #include "../util/MathUtil.h" #include "../common/r_efx.h" #include "cl_dll/eventscripts.h" #include "AvHSprites.h" #include "ui/UIUtil.h" #include "../types.h" #include #include "../common/com_model.h" #include "cl_dll/studio_util.h" #include "cl_dll/r_studioint.h" #include "AvHMiniMap.h" #include "AvHActionButtons.h" #include "../util/STLUtil.h" #include "AvHSharedUtil.h" #include "../common/event_api.h" #include "AvHScriptManager.h" #include #include #include "AvHParticleSystemManager.h" #include "AvHTeamHierarchy.h" #include "AvHClientUtil.h" #include "AvHTooltip.h" #include "cl_dll/demo.h" #include "../common/demo_api.h" #include "AvHHudConstants.h" #include "AvHPlayerUpgrade.h" #include "AvHCommanderModeHandler.h" #include "../common/ref_params.h" #include "AvHTechImpulsePanel.h" #include "AvHServerVariables.h" #include "AvHSpriteAPI.h" #include "AvHParticleEditorHandler.h" #include "AvHAlienAbilityConstants.h" #include #include "../common/entity_types.h" void IN_GetMousePos( int *mx, int *my ); // Externs extern int g_iVisibleMouse; extern playermove_t* pmove; extern engine_studio_api_t IEngineStudio; extern "C" Vector gPredictedPlayerOrigin; extern vec3_t v_origin; extern vec3_t v_angles; //extern vec3_t gPlayerOrigin; //extern vec3_t gPlayerAngles; extern ref_params_s* pDemoRefParams; extern vec3_t gPlaybackViewOrigin; extern AvHCommanderModeHandler gCommanderHandler; extern AvHParticleEditorHandler gParticleEditorHandler; float kD3DErrorValue = 0.01f; vec3_t GetViewOrigin() { vec3_t theOrigin = v_origin; //cl_entity_t* theViewEntity = gEngfuncs.GetLocalPlayer();//gEngfuncs.GetViewModel(); //if(theViewEntity && /*pDemoRefParams &&*/ gEngfuncs.pDemoAPI->IsPlayingback()) //{ // //theOrigin = pDemoRefParams->vieworg; // theOrigin = theViewEntity->origin; //} if(gEngfuncs.pDemoAPI->IsPlayingback()) { theOrigin = gPlaybackViewOrigin; } return theOrigin; // return v_origin; } void BuildLerpedPoint(float inXPercentage, float inYPercentage, const Vector& inUpperLeft, const Vector& inUpperRight, const Vector& inLowerLeft, Vector& outPoint) { ASSERT(inXPercentage >= 0.0f); ASSERT(inXPercentage <= 1.0f); ASSERT(inYPercentage >= 0.0f); ASSERT(inYPercentage <= 1.0f); Vector theUpperLeftToUpperRight; VectorSubtract(inUpperRight, inUpperLeft, theUpperLeftToUpperRight); Vector theUpperLeftToLowerLeft; VectorSubtract(inLowerLeft, inUpperLeft, theUpperLeftToLowerLeft); Vector theXComp; VectorScale(theUpperLeftToUpperRight, inXPercentage, theXComp); Vector theYComp; VectorScale(theUpperLeftToLowerLeft, inYPercentage, theYComp); outPoint = inUpperLeft + theXComp + theYComp; //float theXPercentage = (outPoint.x - inUpperLeft.x)/(theXComp.x + theYComp.x); //float theYPercentage = (outPoint.y - inUpperLeft.y)/(theXComp.y + theYComp.y); } void CalculatePlaneInfo(const Vector& inUpperLeft, const Vector& inUpperRight, const Vector& inLowerLeft, float& outD, Vector& outPlaneNormal) { // Cross two vectors for plane normal Vector theUpperRightToUpperLeft; VectorSubtract(inUpperLeft, inUpperRight, theUpperRightToUpperLeft); theUpperRightToUpperLeft = theUpperRightToUpperLeft.Normalize(); Vector theUpperLeftToLowerLeft; VectorSubtract(inLowerLeft, inUpperLeft, theUpperLeftToLowerLeft); theUpperLeftToLowerLeft = theUpperLeftToLowerLeft.Normalize(); // Calculate plane normal CrossProduct(theUpperRightToUpperLeft, theUpperLeftToLowerLeft, outPlaneNormal); // Plug in one of the points for D (Ax + By + Cz = -D) outD = -(outPlaneNormal.x*inUpperLeft.x + outPlaneNormal.y*inUpperLeft.y + outPlaneNormal.z*inUpperLeft.z); } void CalculatePointOnPlane(int inXPos, int inYPos, const Vector& inOrigin, const Vector& inPlaneNormal, float inPlaneD, Vector& outPoint) { Vector theRay; CreatePickingRay(inXPos, inYPos, theRay); // Solve for parametric t float thePlaneA = inPlaneNormal.x; float thePlaneB = inPlaneNormal.y; float thePlaneC = inPlaneNormal.z; float theT = -(thePlaneA*inOrigin.x + thePlaneB*inOrigin.y + thePlaneC*inOrigin.z + inPlaneD)/(thePlaneA*theRay.x + thePlaneB*theRay.y + thePlaneC*theRay.z); // Now we have t, solve for the endpoint outPoint.x = inOrigin.x + theT*theRay.x; outPoint.y = inOrigin.y + theT*theRay.y; outPoint.z = inOrigin.z + theT*theRay.z; } void ProjectPointFromViewOrigin(int inDistanceToProject, int inScreenX, int inScreenY, Vector& outResult) { Vector theRay; CreatePickingRay(inScreenX, inScreenY, theRay); // project default distance along picking ray VectorMA(GetViewOrigin(), inDistanceToProject, theRay, outResult); } void AvHHud::DrawTranslatedString(int inX, int inY, const char* inStringToTranslate, bool inCentered, bool inIgnoreUpgrades, bool inTrimExtraInfo, float alpha) { // Translate string theTranslatedText; LocalizeString(inStringToTranslate, theTranslatedText); if(theTranslatedText[0] == '#') { theTranslatedText = theTranslatedText.substr(1, theTranslatedText.size()); } if(inTrimExtraInfo) { AvHCUTrimExtraneousLocationText(theTranslatedText); } // Draw it if(theTranslatedText != "") { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, inIgnoreUpgrades, false); char theCharBuffer[512]; sprintf(theCharBuffer, "%s", theTranslatedText.c_str()); theR *= alpha; theB *= alpha; theG *= alpha; if(inCentered) { this->DrawHudStringCentered(inX, inY, ScreenWidth(), theCharBuffer, theR, theG, theB); } else { this->DrawHudString(inX, inY, ScreenWidth(), theCharBuffer, theR, theG, theB); } } } bool gWarpHUDSprites = false; float gWarpXAmount = 0.0f; float gWarpYAmount = 0.0f; float gWarpXSpeed = 0.0f; float gWarpYSpeed = 0.0f; void SetWarpHUDSprites(bool inMode, float inWarpXAmount = 0.0f, float inWarpYAmount = 0.0f, float inWarpXSpeed = 0.0f, float inWarpYSpeed = 0.0f) { gWarpHUDSprites = inMode; gWarpXAmount = inWarpXAmount; gWarpYAmount = inWarpYAmount; gWarpXSpeed = inWarpXSpeed; gWarpYSpeed = inWarpYSpeed; } void DrawScaledHUDSprite(AVHHSPRITE inSpriteHandle, int inMode, int inRowsInSprite = 1, int inX = 0, int inY = 0, int inWidth = ScreenWidth(), int inHeight = ScreenHeight(), int inForceSpriteFrame = -1, float inStartU = 0.0f, float inStartV = 0.0f, float inEndU = 1.0f, float inEndV = 1.0f, float inRotateUVRadians = 0.0f, bool inUVWrapsOverFrames = false) { // Count number of frames int theNumFrames = SPR_Frames(inSpriteHandle); //int theSpriteWidth = SPR_Width(inSpriteHandle, 0); //int theSpriteHeight = SPR_Height(inSpriteHandle, 0); //float theAspectRatio = theSpriteWidth/theSpriteHeight; // ASSERT that the the number of rows makes sense for the number of frames ASSERT(theNumFrames > 0); float theFloatNumCols = (float)theNumFrames/inRowsInSprite; int theNumCols = (int)theFloatNumCols; ASSERT(theNumCols == theFloatNumCols); //int theNumRows = theNumFrames/theNumCols; int theNumRows = inRowsInSprite; // Allow scaling of one frame, without tiling if(inForceSpriteFrame != -1) { theNumRows = theNumCols = 1; } // Make sure coords are bounded to allowable ranges inStartU = min(max(inStartU, kD3DErrorValue), 1.0f - kD3DErrorValue); inStartV = min(max(inStartV, kD3DErrorValue), 1.0f - kD3DErrorValue); inEndU = min(max(inEndU, kD3DErrorValue), 1.0f - kD3DErrorValue); inEndV = min(max(inEndV, kD3DErrorValue), 1.0f - kD3DErrorValue); if(inRotateUVRadians != 0.0f) { // Rotate all the UV coords vec3_t theAngles(0.0f, 0.0f, inRotateUVRadians); float theMatrix[3][4]; AngleMatrix(theAngles, theMatrix); float theRotatedValues[3]; float theStartValues[3] = {inStartU, inStartV, 0.0f}; VectorRotate(theStartValues, theMatrix, theRotatedValues); inStartU = theRotatedValues[0]; inStartV = theRotatedValues[1]; float theEndValues[3] = {inEndU, inEndV, 0.0f}; VectorRotate(theEndValues, theMatrix, theRotatedValues); inEndU = theRotatedValues[0]; inEndV = theRotatedValues[1]; } // Calculate width and height of each quad int theQuadScreenWidth = inWidth/theNumCols; int theQuadScreenHeight = inHeight/theNumRows; //Vector thePickRay; //int theHalfWidth = ScreenWidth/2; //int theHalfHeight = ScreenHeight/2; //CreatePickingRay(theHalfWidth, theHalfHeight, thePickRay); //char gDebugMessage[256]; //sprintf(gDebugMessage, "(%d, %d): %f, %f, %f", theHalfWidth, theHalfHeight, thePickRay.x, thePickRay.y, thePickRay.z); //CenterPrint(gDebugMessage); //float theRenderOrigin[3]; //pVector theUp; //pVector theRight; //pVector theNormal; //IEngineStudio.GetViewInfo(theRenderOrigin, (float*)&theUp, (float*)&theRight, (float*)&theNormal); //Demo_WriteVector(TYPE_VIEWANGLES, v_angles); //Demo_WriteVector(TYPE_VIEWORIGIN, v_origin); vec3_t theRenderOrigin; VectorCopy(GetViewOrigin(), theRenderOrigin); vec3_t theForward, theRight, theUp; AngleVectors(v_angles, theForward, theRight, theUp); Vector theRenderOriginVector; theRenderOriginVector.x = theRenderOrigin[0]; theRenderOriginVector.y = theRenderOrigin[1]; theRenderOriginVector.z = theRenderOrigin[2]; // This is the smallest value that displays and it still clips with the view model a little const int kDistanceToProject = 10; // create picking ray for upper left of quad Vector theUpperLeftRay; //CreatePickingRay(inX, inY, theUpperLeftRay); CreatePickingRay(0, 0, theUpperLeftRay); // create picking ray for lower left of quad Vector theLowerLeftRay; //CreatePickingRay(inX, inY+inHeight, theLowerLeftRay); CreatePickingRay(0, ScreenHeight(), theLowerLeftRay); // create picking ray for upper right of quad Vector theUpperRightRay; //CreatePickingRay(inX+inWidth, inY, theUpperRightRay); CreatePickingRay(ScreenWidth(), 0, theUpperRightRay); // project default distance along picking ray Vector theUpperLeftWorldPos; VectorMA(theRenderOrigin, kDistanceToProject, theUpperLeftRay, theUpperLeftWorldPos); Vector theUpperRightWorldPos; VectorMA(theRenderOrigin, kDistanceToProject, theUpperRightRay, theUpperRightWorldPos); Vector theLowerLeftWorldPos; VectorMA(theRenderOrigin, kDistanceToProject, theLowerLeftRay, theLowerLeftWorldPos); // Create formula for plane float thePlaneD; Vector thePlaneNormal; CalculatePlaneInfo(theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePlaneD, thePlaneNormal); // loop through screen height int theCurrentFrame = 0; // Allow drawing of just one frame if(inForceSpriteFrame != -1) { theCurrentFrame = inForceSpriteFrame; } for(int i = 0; i < theNumRows; i++) { // loop through screen width for(int j = 0; j < theNumCols; j++) { // draw quad gEngfuncs.pTriAPI->RenderMode(inMode); gEngfuncs.pTriAPI->CullFace(TRI_NONE); //gEngfuncs.pTriAPI->Brightness(1); if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s*)gEngfuncs.GetSpritePointer(inSpriteHandle), theCurrentFrame)) { gEngfuncs.pTriAPI->Begin( TRI_TRIANGLE_STRIP ); //gEngfuncs.pTriAPI->Color4f(1.0f, 1.0f, 1.0f, 1.0f); bool theIsFirstRow = (i == 0); bool theIsFirstCol = (j == 0); bool theIsLastRow = (i == (theNumRows-1)); bool theIsLastCol = (j == (theNumCols-1)); float theStartU = inStartU; float theStartV = inStartV; float theEndU = inEndU; float theEndV = inEndV; if(inUVWrapsOverFrames) { if((theNumCols > 1) && !theIsFirstCol) { theStartU = 0.0f; } if((theNumRows > 1) && !theIsFirstRow) { theStartV = 0.0f; } if((theNumCols > 1) && !theIsLastCol) { theEndU = 1.0f; } if((theNumRows > 1) && !theIsLastRow) { theEndV = 1.0f; } } // Optionally warp XY coords using current time int theWarpXStartAmount = 0; int theWarpYStartAmount = 0; int theWarpXEndAmount = 0; int theWarpYEndAmount = 0; if(gWarpHUDSprites) { float theCurrentTime = gHUD.GetTimeOfLastUpdate(); float theNormXAmount = theCurrentTime*gWarpXSpeed - (int)(theCurrentTime*gWarpXSpeed); // Get fractional part of second float theNormYAmount = theCurrentTime*gWarpYSpeed - (int)(theCurrentTime*gWarpYSpeed); float theSinusoidalNormXAmount = cos(theNormXAmount*2.0f*M_PI); float theSinusoidalNormYAmount = sin(theNormYAmount*2.0f*M_PI); float theXAmount = theSinusoidalNormXAmount*gWarpXAmount;// - gWarpUAmount/2.0f; float theYAmount = theSinusoidalNormYAmount*gWarpYAmount;// - gWarpVAmount/2.0f; if(!theIsFirstCol) { theWarpXStartAmount = theXAmount*ScreenWidth(); } if(!theIsLastCol) { theWarpXEndAmount = theXAmount*ScreenWidth(); } if(!theIsFirstRow) { theWarpYStartAmount = theYAmount*ScreenHeight(); } if(!theIsLastRow) { theWarpYEndAmount = theYAmount*ScreenHeight(); } } // Compensate for gamma float theGammaSlope = gHUD.GetGammaSlope(); float theColorComponent = 1.0f/theGammaSlope; gEngfuncs.pTriAPI->Color4f(theColorComponent, theColorComponent, theColorComponent, 1.0f); Vector thePoint; //float theMinXPercentage = (float)j/theNumCols; //float theMaxXPercentage = (float)(j+1)/theNumCols; //float theMinYPercentage = (float)i/theNumRows; //float theMaxYPercentage = (float)(i+1)/theNumRows; int theMinXPos = inX + ((float)j/theNumCols)*inWidth + theWarpXStartAmount; int theMinYPos = inY + ((float)i/theNumRows)*inHeight + theWarpYStartAmount; int theMaxXPos = inX + ((float)(j+1)/theNumCols)*inWidth + theWarpXEndAmount; int theMaxYPos = inY + ((float)(i+1)/theNumRows)*inHeight + theWarpYEndAmount; // Lower left Vector thePointOne; //BuildLerpedPoint(theMinXPercentage, theMaxYPercentage, theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePointOne); CalculatePointOnPlane(theMinXPos, theMaxYPos, theRenderOriginVector, thePlaneNormal, thePlaneD, thePointOne); float theU = theStartU; float theV = theEndV; gEngfuncs.pTriAPI->TexCoord2f(theU, theV);// 0,1 gEngfuncs.pTriAPI->Vertex3fv((float*)&thePointOne); // Upper left Vector thePointTwo; //BuildLerpedPoint(theMinXPercentage, theMinYPercentage, theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePointTwo); CalculatePointOnPlane(theMinXPos, theMinYPos, theRenderOriginVector, thePlaneNormal, thePlaneD, thePointTwo); theU = theStartU; theV = theStartV; //gEngfuncs.pTriAPI->TexCoord2f(inStartU, inStartV); // 0,0 gEngfuncs.pTriAPI->TexCoord2f(theU, theV);// 0,1 gEngfuncs.pTriAPI->Vertex3fv((float*)&thePointTwo); // Lower right Vector thePointFour; //BuildLerpedPoint(theMaxXPercentage, theMaxYPercentage, theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePointFour); CalculatePointOnPlane(theMaxXPos, theMaxYPos, theRenderOriginVector, thePlaneNormal, thePlaneD, thePointFour); theU = theEndU; theV = theEndV; //gEngfuncs.pTriAPI->TexCoord2f(inEndU, inEndV);// 1,1 gEngfuncs.pTriAPI->TexCoord2f(theU, theV);// 0,1 gEngfuncs.pTriAPI->Vertex3fv((float*)&thePointFour); // Upper right Vector thePointThree; //BuildLerpedPoint(theMaxXPercentage, theMinYPercentage, theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePointThree); CalculatePointOnPlane(theMaxXPos, theMinYPos, theRenderOriginVector, thePlaneNormal, thePlaneD, thePointThree); theU = theEndU; theV = theStartV; //gEngfuncs.pTriAPI->TexCoord2f(inEndU, inStartV); // 1,0 gEngfuncs.pTriAPI->TexCoord2f(theU, theV);// 0,1 gEngfuncs.pTriAPI->Vertex3fv((float*)&thePointThree); gEngfuncs.pTriAPI->End(); } // Increment frame theCurrentFrame++; // Allow drawing of just one frame if(inForceSpriteFrame != -1) { theCurrentFrame = inForceSpriteFrame; } // increment current x and y //theCurrentScreenX += theQuadScreenWidth; } //theCurrentScreenX = 0; //theCurrentScreenY += theQuadScreenHeight; } } void DrawVariableScaledHUDSprite(float inFactor, AVHHSPRITE inSpriteHandle, int inMode, int inX, int inY, int inWidth, int inHeight) { // Draw as two scaled sprites, one for the level and one for the "empty" level // Assumes that sprite has two frames, with the empty level being frame 0 and the full frame being frame 1 int theWidth = inWidth; float theStartU = 0.0f; float theEndU = 1.0f; int theHeight = inFactor*inHeight; float theStartV = 1.0f - inFactor; float theEndV = 1.0f; int theX = inX; int theY = inY + inHeight - theHeight; DrawScaledHUDSprite(inSpriteHandle, inMode, 1, theX, theY, theWidth, theHeight, 1, theStartU, theStartV, theEndU, theEndV); // Draw background theHeight = (1.0f - inFactor)*inHeight; theY = inY; theStartV = 0.0f; theEndV = 1.0f - inFactor; DrawScaledHUDSprite(inSpriteHandle, inMode, 1, theX, theY, theWidth, theHeight, 0, theStartU, theStartV, theEndU, theEndV); } void DrawSpriteOnGroundAtPoint(vec3_t inOrigin, int inRadius, AVHHSPRITE inSprite, int inRenderMode = kRenderNormal, int inFrame = 0, float inAlpha = 1.0f) { if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s *)gEngfuncs.GetSpritePointer(inSprite), inFrame)) { gEngfuncs.pTriAPI->CullFace(TRI_NONE); gEngfuncs.pTriAPI->RenderMode(inRenderMode); //gEngfuncs.pTriAPI->RenderMode( kRenderTransTexture ); //gEngfuncs.pTriAPI->Color4f(inR, inG, inB, inA); gEngfuncs.pTriAPI->Begin(TRI_TRIANGLE_STRIP); // Draw one quad vec3_t thePoint = inOrigin; float theGammaSlope = gHUD.GetGammaSlope(); ASSERT(theGammaSlope > 0.0f); float theColorComponent = 1.0f/theGammaSlope; gEngfuncs.pTriAPI->Color4f(theColorComponent, theColorComponent, theColorComponent, inAlpha); gEngfuncs.pTriAPI->Brightness(1.6f); thePoint[0] = inOrigin[0] - inRadius; thePoint[1] = inOrigin[1] - inRadius; gEngfuncs.pTriAPI->TexCoord2f(0, 0); gEngfuncs.pTriAPI->Vertex3f(thePoint[0], thePoint[1], thePoint[2]); thePoint[0] = inOrigin[0] - inRadius; thePoint[1] = inOrigin[1] + inRadius; gEngfuncs.pTriAPI->TexCoord2f(0, 1); gEngfuncs.pTriAPI->Vertex3f(thePoint[0], thePoint[1], thePoint[2]); thePoint[0] = inOrigin[0] + inRadius; thePoint[1] = inOrigin[1] - inRadius; gEngfuncs.pTriAPI->TexCoord2f(1, 0); gEngfuncs.pTriAPI->Vertex3f(thePoint[0], thePoint[1], thePoint[2]); thePoint[0] = inOrigin[0] + inRadius; thePoint[1] = inOrigin[1] + inRadius; gEngfuncs.pTriAPI->TexCoord2f(1, 1); gEngfuncs.pTriAPI->Vertex3f(thePoint[0], thePoint[1], thePoint[2]); gEngfuncs.pTriAPI->End(); gEngfuncs.pTriAPI->RenderMode( kRenderNormal ); } } void AvHHud::DrawPlayerNames() { bool inReadyRoom = false;//(this->GetPlayMode() == PLAYMODE_READYROOM); bool inTopDownMode = this->GetInTopDownMode(); if(inTopDownMode || inReadyRoom /*&& !gEngfuncs.pDemoAPI->IsPlayingback()*/) { cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer(); if(theLocalPlayer) { int theLocalPlayerIndex = theLocalPlayer->index; // Loop through all players for(int i = 1; i <= kMaxPlayers; i++) { if((i != theLocalPlayerIndex) && AvHCUGetIsEntityInPVSAndVisible(i)) { cl_entity_s* theCurrentPlayer = gEngfuncs.GetEntityByIndex(i); bool theDrawEntity = (inReadyRoom || (theCurrentPlayer && theCurrentPlayer->player && (AvHTeamNumber(theCurrentPlayer->curstate.team) == this->GetHUDTeam()) && (i != theLocalPlayerIndex)) ); if(theDrawEntity) { string theEntityName; bool theIsEnemy; if(this->GetEntityInfoString(i, theEntityName, theIsEnemy)) { vec3_t theEntityOrigin; VectorCopy(theCurrentPlayer->curstate.origin, theEntityOrigin); theEntityOrigin.z += AvHCUGetIconHeightForPlayer((AvHUser3)theCurrentPlayer->curstate.iuser3); // If they are on screen Vector theScreenPos; if(AvHCUWorldToScreen(theEntityOrigin, (float*)&theScreenPos)) { // Set color int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); // If selected or parasited, draw in different color if(inTopDownMode) { if (GetHasUpgrade(theCurrentPlayer->curstate.iuser4, MASK_PARASITED)) { ////Old NS 3.2 method. Now in AvHHud::GetEntityInfoString. //string thePrePendString; //LocalizeString(kParasited, thePrePendString); //theEntityName = string(theEntityName + " (" + thePrePendString + ")"); // Set color to parasited color UnpackRGB(theR, theG, theB, RGB_MARINE_PARASITED); } bool theIsSelected = (std::find(this->mSelected.begin(), this->mSelected.end(), i) != this->mSelected.end()); if (theIsSelected) { // Selected color UnpackRGB(theR, theG, theB, RGB_MARINE_SELECTED); } } // Set text color draw in different color this->mTopDownPlayerNameMessage.SetRGB(theR, theG, theB); this->mTopDownPlayerNameMessage.SetIgnoreFadeForLifetime(true); // Set the message info and draw it this->mTopDownPlayerNameMessage.SetText(theEntityName); // Set position Vector theNormPos; float theNormX = theScreenPos.x/ScreenWidth(); int theBoxHeight = this->mTopDownPlayerNameMessage.GetScreenHeight(); float theNormY = (theScreenPos.y - theBoxHeight)/ScreenHeight(); if((inTopDownMode && !this->GetIsRegionBlockedByUI(theNormX, theNormY)) || inReadyRoom) { this->mTopDownPlayerNameMessage.SetNormalizedScreenX(theNormX); this->mTopDownPlayerNameMessage.SetNormalizedScreenY(theNormY); this->mTopDownPlayerNameMessage.SetCentered(true); this->mTopDownPlayerNameMessage.SetNormalizedMaxWidth(kReticleMaxWidth); this->mTopDownPlayerNameMessage.Draw(); } } } } } } } } } //bool AvHHud::ChopStringOfMaxScreenWidth(int inMaxScreenWidth, string& ioBaseString, string& outChoppedString) //{ // // Loop backwards through the string, until we get a string that fits in this screen width // int theCurrentLength = ioBaseString.length(); // int theMaxLength = ioBaseString.length(); // bool theSuccess = false; // // while(!theSuccess) // { // string theCurrentString = ioBaseString.substr(0, theCurrentLength); // int theCurrentStringScreenWidth = this->GetHudStringWidth(theCurrentString.c_str()); // if(theCurrentStringScreenWidth <= inMaxScreenWidth) // { // // Look for a word to break the line // while((theCurrentLength > 0) && !theSuccess) // { // char theCurrentChar = ioBaseString[theCurrentLength-1]; // if((theCurrentChar == ' ') || (theCurrentLength == theMaxLength)) // { // outChoppedString = ioBaseString.substr(0, theCurrentLength); // ioBaseString = ioBaseString.substr(theCurrentLength, ioBaseString.length() - theCurrentLength); // theSuccess = true; // break; // } // else // { // theCurrentLength--; // } // } // } // else // { // theCurrentLength--; // } // } // // return theSuccess; //} void AvHHud::DrawReticleInfo() { this->mReticleMessage.Draw(); // if(this->mReticleInfoText != "") // { // const float kMaxWidth = .3f; // int kMaxScreenWidth = kMaxWidth*ScreenWidth; // // StringList theStringList; // string theHelpString = this->mReticleInfoText; // // do // { // string theNewString; // if(ChopStringOfMaxScreenWidth(kMaxScreenWidth, theHelpString, theNewString)) // { // theStringList.push_back(theNewString); // } // else // { // theHelpString = ""; // } // } // while(theHelpString != ""); // // // For each line, if the line contains any special markers, move them to their own lines // // // Compute max width of all the strings, add some extra for a frame // int theBoxWidth = 0; // StringList::iterator theStringListIter; // for(theStringListIter = theStringList.begin(); theStringListIter != theStringList.end(); theStringListIter++) // { // int theCurrentScreenWidth = this->GetHudStringWidth(theStringListIter->c_str()); // theBoxWidth = max(theBoxWidth, theCurrentScreenWidth); // } // int theLineHSpacing = .01f*ScreenWidth; // theBoxWidth += 2*theLineHSpacing; // // // Compute max height needed to contain all the strings, add some extra for a frame // int theLineVSpacing = .01f*ScreenHeight(); // int theLineHeight = this->GetHudStringHeight(); // int theBoxHeight = 2*theLineVSpacing + (theStringList.size()*theLineHeight); // // int theFillStartX = this->mReticleInfoScreenX; // int theFillStartY = this->mReticleInfoScreenY; // // theFillStartX -= theBoxWidth/2; // theFillStartY -= theBoxHeight/2; // // // Draw nice border and shaded background // const float kReticleInfoMaxAlpha = 25; // float theNormalizedAlpha = this->mReticleInfoColorA/255; // int theAlphaComponent = theNormalizedAlpha*kReticleInfoMaxAlpha; // // FillRGBA(theFillStartX, theFillStartY, theBoxWidth, theBoxHeight, this->mReticleInfoColorR, this->mReticleInfoColorG, this->mReticleInfoColorB, theAlphaComponent); // vguiSimpleBox(theFillStartX, theFillStartY, theFillStartX + theBoxWidth, theFillStartY + theBoxHeight, this->mReticleInfoColorR, this->mReticleInfoColorG, this->mReticleInfoColorB, theAlphaComponent); // // // Now draw each line, non-centered, left-aligned // int theLineNumber = 0; // for(theStringListIter = theStringList.begin(); theStringListIter != theStringList.end(); theStringListIter++) // { // int theR = this->mReticleInfoColorR; // int theG = this->mReticleInfoColorG; // int theB = this->mReticleInfoColorB; // // // If the line starts with a marker, draw it in a special color // //string theDamageMarker(kDamageMarker); // //if(theStringListIter->substr(0, theDamageMarker.length()) == theDamageMarker) // //{ // // // Draw string in yellow // // theR = theG = 255; // // theB = 25; // //} // // int theBaseY = theFillStartY + theLineVSpacing + theLineNumber*theLineHeight; // // // Draw message (DrawHudStringCentered only centers in x) // this->DrawHudString(theFillStartX + theLineHSpacing, theBaseY /*- theLineHeight/2*/, ScreenWidth, theStringListIter->c_str(), theR*theNormalizedAlpha, theG*theNormalizedAlpha, theB*theNormalizedAlpha); // // theLineNumber++; // } // } } void AvHHud::DrawToolTips() { if(!gEngfuncs.pDemoAPI->IsPlayingback()) { this->mHelpMessage.Draw(); // Draw each one for(AvHTooltipListType::iterator theIter = this->mTooltips.begin(); theIter != this->mTooltips.end(); theIter++) { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); theIter->SetRGB(theR, theG, theB); theIter->Draw(); } } } void AvHHud::DrawWorldSprite(AVHHSPRITE inSpriteHandle, int inRenderMode, vec3_t inWorldPosition, int inFrame, float inWorldSize, float inAlpha) // : added inAlpha { vec3_t theUpperLeft; vec3_t theLowerRight; vec3_t theForward, theRight, theUp; AngleVectors(v_angles, theForward, theRight, theUp); vec3_t theToUpperLeft; VectorAdd(-theRight, theUp, theToUpperLeft); VectorNormalize(theToUpperLeft); VectorMA(inWorldPosition, inWorldSize, theToUpperLeft, theUpperLeft); vec3_t theToLowerRight; VectorAdd(theRight, -theUp, theToLowerRight); VectorNormalize(theToLowerRight); VectorMA(inWorldPosition, inWorldSize, theToLowerRight, theLowerRight); vec3_t theScreenUpperLeft; vec3_t theScreenLowerRight; // World to screen returns true if the world pos is behind the viewer if(!gEngfuncs.pTriAPI->WorldToScreen((float*)theUpperLeft, (float*)theScreenUpperLeft)) { if(!gEngfuncs.pTriAPI->WorldToScreen((float*)theLowerRight, (float*)theScreenLowerRight)) { // If the sprite is behind you, push it to the bottom or top of the screen // cl_entity_t* theLocalPlayer = gEngfuncs.GetLocalPlayer(); // ASSERT(theLocalPlayer); // // vec3_t theDirectionToOrder; // VectorSubtract(inWorldPosition, theLocalPlayer->origin, theDirectionToOrder); // float theDotProduct = DotProduct(theDirectionToOrder, theLocalPlayer->angles); // if(theDotProduct < 0) // { // if(theWorldPos.z < theLocalPlayer->origin.z) // { // theY = theScreenHeight - theSpriteHeight - theScreenBorder; // } // else // { // theY = theScreenBorder; // } // vec3_t theCrossProduct; // theCrossProduct = CrossProduct(theLocalPlayer->angles, theDirectionToOrder); // // // It's to our right // if(theCrossProduct.z > 0) // { // theX = theScreenWidth - theSpriteWidth - theScreenBorder; // } // else // { // theX = theScreenBorder; // } // } // float theDistanceToLocation = (float)VectorDistance(inWorldPosition, theLocalPlayer->origin); // const theMaxDistance = 1500; // float theEffectiveDistance = min(theDistanceToLocation, theMaxDistance); // const int theMinColorComponent = 100; // int theColorComponent = max(theMinColorComponent, 255 - ((theEffectiveDistance/theMaxDistance)*255)); //SPR_Set(inSpriteHandle, theColorComponent, theColorComponent, theColorComponent); ////SPR_DrawHoles((int)0, theX, theY, NULL); //if(inRenderMode == kRenderNormal) //{ // SPR_Draw(inFrame, theX, theY, NULL); //} //else if(inRenderMode == kRenderTransAdd) //{ // SPR_DrawAdditive(inFrame, theX, theY, NULL); //} float theScreenX = XPROJECT(theScreenUpperLeft.x); float theScreenY = YPROJECT(theScreenUpperLeft.y); float theWidth = XPROJECT(theScreenLowerRight.x) - theScreenX; float theHeight = YPROJECT(theScreenLowerRight.y) - theScreenY; //DrawScaledHUDSprite(inSpriteHandle, inRenderMode, 1, theScreenX, theScreenY, theWidth, theHeight, inFrame); AvHSpriteSetColor(1, 1, 1, inAlpha); AvHSpriteSetRenderMode(inRenderMode); AvHSpriteDraw(inSpriteHandle, inFrame, theScreenX, theScreenY, theScreenX + theWidth, theScreenY + theHeight, 0, 0, 1, 1); } } } void AvHHud::DrawOrderIcon(const AvHOrder& inOrder) { if(this->mOrderSprite) { int theNumFrames = SPR_Frames(this->mOrderSprite); int theCurrentFrame = this->GetFrameForOrderType(inOrder.GetOrderType()); if((theCurrentFrame >= 0) && (theCurrentFrame < theNumFrames)) { vec3_t theWorldPos; inOrder.GetLocation(theWorldPos); if ( inOrder.GetOrderType() == ORDERTYPET_ATTACK ) { theWorldPos[2]+=kAttackOrderZOffset; } // Draw icon above pos, text below theWorldPos.z += BALANCE_VAR(kOrderIconDrawSize); this->DrawWorldSprite(this->mOrderSprite, kRenderTransAdd, theWorldPos, theCurrentFrame, BALANCE_VAR(kOrderIconDrawSize)); // If the order is our own order, draw the order indicator around it if((this->GetHUDPlayMode() == PLAYMODE_PLAYING) && this->GetIsMarine() && !this->GetInTopDownMode()) { this->DrawWorldSprite(this->mMarineOrderIndicator, kRenderTransAdd, theWorldPos, 0, BALANCE_VAR(kOrderIconDrawSize)); //DrawScaledHUDSprite(theSpriteHandle, kRenderNormal, 1, thePosX, thePosY, theWidth, theHeight, theFrame, theStartU, theStartV, theEndU, theEndV); } vec3_t orderDir; inOrder.GetLocation(orderDir); this->GetOrderDirection(orderDir, 2); } } } void AvHHud::DrawOrderText(const AvHOrder& inOrder) { int theIndex = (int)(inOrder.GetOrderType()); // Now draw text describing the waypoint string theTitle; sprintf(theTitle, "Order%d", theIndex); string theLocalizedTitle(" "); LocalizeString(theTitle.c_str(), theLocalizedTitle); if((theIndex == ORDERTYPET_BUILD) || (theIndex == ORDERTYPET_GUARD) || (theIndex == ORDERTYPET_GET)) { AvHUser3 theUser3 = inOrder.GetTargetUser3Type(); string theUser3Name; if(this->GetTranslatedUser3Name(theUser3, theUser3Name)) { // "Get %s" -> "Get heavy machine gun" // "Guard %s -> "Guard soldier" // "Build %s" -> "Build sentry turret" string theTitleWithTarget; sprintf(theTitleWithTarget, theLocalizedTitle.c_str(), theUser3Name.c_str()); theLocalizedTitle = theTitleWithTarget; } } vec3_t theOrderLocation; inOrder.GetLocation(theOrderLocation); // Because the commander may not have information about the players heading to this waypoint (outside of his PVS), we // can't draw a range for the commander string theRangeDisplayString; if(!this->GetInTopDownMode()) { float theDistanceToWaypoint = VectorDistance(gPredictedPlayerOrigin, theOrderLocation); int theVisibleDistance = max(1, (int)theDistanceToWaypoint/100); string theVisibleUnits; if(LocalizeString("Range", theVisibleUnits)) { sprintf(theRangeDisplayString, theVisibleUnits.c_str(), theVisibleDistance); } } string theLocationOfOrder; theLocationOfOrder = this->GetNameOfLocation(theOrderLocation); // It's OK if this fails, as only official maps will have these translated string theTranslatedLocation = theLocationOfOrder; LocalizeString(theLocationOfOrder.c_str(), theTranslatedLocation); // : 0000992 string theFirstLine = theLocalizedTitle; if(theRangeDisplayString != "") { theFirstLine += string(" : ") + theRangeDisplayString; } // : Vector theScreenPos; if(AvHCUWorldToScreen((float*)theOrderLocation, (float*)&theScreenPos)) { float theNormX = theScreenPos.x/ScreenWidth(); float theNormY = theScreenPos.y/ScreenHeight(); if(!this->GetIsRegionBlockedByUI(theNormX, theNormY)) { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); string theFirstLine = theLocalizedTitle; if(theRangeDisplayString != "") { theFirstLine += string(" : ") + theRangeDisplayString; } // Draw order (icon above world position, text below it) int theBaseX = theScreenPos.x; int theBaseY = theScreenPos.y; int theStringHeight = this->GetHudStringHeight(); this->DrawHudStringCentered(theBaseX, theBaseY + theStringHeight, ScreenWidth(), theFirstLine.c_str(), theR, theG, theB); // Draw location below it this->DrawHudStringCentered(theBaseX, theBaseY + 2*theStringHeight, ScreenWidth(), theTranslatedLocation.c_str(), theR, theG, theB); } } // : 0000992 if (this->mDisplayOrderType == 2) { // this->mDisplayOrderText1 = "The commander issued an order:"; this->mDisplayOrderText1 = theFirstLine.c_str(); this->mDisplayOrderText2 = theTranslatedLocation.c_str(); } // : } // : #define CENTER_TEXT_LENGTH 10 #define CENTER_TEXT_FADEOUT 2 void AvHHud::DrawCenterText() { if ((this->mCenterTextTime > -1) && (this->mTimeOfLastUpdate < this->mCenterTextTime + CENTER_TEXT_LENGTH + CENTER_TEXT_FADEOUT)) { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); if (this->mTimeOfLastUpdate > this->mCenterTextTime + CENTER_TEXT_LENGTH) { float fraction = this->mTimeOfLastUpdate - (this->mCenterTextTime + CENTER_TEXT_LENGTH); fraction = 1 - fraction / CENTER_TEXT_FADEOUT; theR *= fraction; theG *= fraction; theB *= fraction; } int posX = 0.5 * ScreenWidth() - this->mFont.GetStringWidth(this->mCenterText.c_str()) / 2; int posY = 0.4 * ScreenHeight(); this->mFont.DrawString(posX, posY, this->mCenterText.c_str(), theR, theG, theB); } } // : // : 0000992 void AvHHud::SetDisplayOrder(int inOrderType, int inOrderIndex, string inText1, string inText2, string inText3) { this->mDisplayOrderTime = this->mTimeOfLastUpdate; this->mDisplayOrderType = inOrderType; this->mDisplayOrderIndex = inOrderIndex; this->mDisplayOrderText1 = inText1; this->mDisplayOrderText2 = inText2; this->mDisplayOrderText3 = inText3; } void AvHHud::DrawDisplayOrder() { const float flashLength = 1.0f; const float fadeLimit = 6.0f; const float fadeEnd = 2.0f; if ((this->mDisplayOrderType > 0) && (this->mDisplayOrderTime + fadeLimit + fadeEnd) > this->mTimeOfLastUpdate && (this->GetInTopDownMode() == false)) { float theFade = 1.0f; if ((this->mDisplayOrderTime + fadeLimit) < this->mTimeOfLastUpdate) { theFade = 1.0f - (this->mTimeOfLastUpdate - (this->mDisplayOrderTime + fadeLimit)) / fadeEnd; if(theFade < 0.0f) { this->mDisplayOrderType = 0; return; } } // flash the icon for the first second if ((this->mDisplayOrderTime + flashLength) > this->mTimeOfLastUpdate) { if (((int)((this->mTimeOfLastUpdate - this->mDisplayOrderTime) * 8)) % 2) { theFade = 0.0f; } } // draw the panel // int sprite = SPR_Load(kWhiteSprite); int r, g, b; GetPrimaryHudColor(r, g, b, true, false); int theStringHeight = this->GetHudStringHeight(); float mIconX1 = 0.47f * ScreenWidth(); float mIconY1 = 0.10f * ScreenHeight(); float mIconX2 = mIconX1 + 0.06f * ScreenWidth(); float mIconY2 = mIconY1 + 0.06f * ScreenWidth(); float mLeftX = mIconX1 - 0.06f * ScreenWidth(); float mRightX = mIconX2 + 0.06f * ScreenWidth(); float mTextX1 = 0.50f * ScreenWidth(); AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteSetDrawMode(kSpriteDrawModeFilled); AvHSpriteSetColor(1, 1, 1, 1 * theFade); int theTeamAdd = 0; if (this->GetIsAlien()) theTeamAdd = 2; if (this->mDisplayOrderDirection == 1) AvHSpriteDraw(this->mTeammateOrderSprite, TEAMMATE_MARINE_LEFT_ARROW + theTeamAdd, mLeftX, mIconY1, mIconX1, mIconY2, 0, 0, 1, 1); // Left else if (this->mDisplayOrderDirection == 2) AvHSpriteDraw(this->mTeammateOrderSprite, TEAMMATE_MARINE_RIGHT_ARROW + theTeamAdd, mIconX2, mIconY1, mRightX, mIconY2, 0, 0, 1, 1); // Right if (this->mDisplayOrderType == 1) { AvHSpriteDraw(this->mTeammateOrderSprite, this->mDisplayOrderIndex + 8, mIconX1, mIconY1, mIconX2, mIconY2, 0, 0, 1, 1); this->DrawHudStringCentered(mTextX1, mIconY2, ScreenWidth(), this->mDisplayOrderText1.c_str(), r, g, b); } else if (this->mDisplayOrderType == 2) { AvHSpriteDraw(this->mOrderSprite, this->mDisplayOrderIndex, mIconX1, mIconY1, mIconX2, mIconY2, 0, 0, 1, 1); this->DrawHudStringCentered(mTextX1, mIconY2, ScreenWidth(), this->mDisplayOrderText1.c_str(), r, g, b); this->DrawHudStringCentered(mTextX1, mIconY2 + theStringHeight, ScreenWidth(), this->mDisplayOrderText2.c_str(), r, g, b); } // float mTextX1 = mIconX2 + 0.02 * ScreenWidth(); // this->DrawHudString(mTextX1, mIconY1, ScreenWidth(), this->mDisplayOrderText1.c_str(), r, g, b); // this->DrawHudString(mTextX1, mIconY1 + theStringHeight, ScreenWidth(), this->mDisplayOrderText2.c_str(), r, g, b); // this->DrawHudString(mTextX1, mIconY1 + theStringHeight * 2, ScreenWidth(), this->mDisplayOrderText3.c_str(), r, g, b); } } // : // : 0000971 void AvHHud::GetOrderDirection(vec3_t inTarget, int inOrderType) { if (this->mDisplayOrderType == inOrderType) { // find left/right/none direction for the order popup notificator vec3_t theForward, theRight, theUp, theDir; AngleVectors(v_angles, theForward, theRight, theUp); VectorSubtract(inTarget, v_origin, theDir); theForward[2] = theDir[2] = 0; VectorNormalize(theForward); VectorNormalize(theDir); this->mDisplayOrderDirection = 0; // if it is too close to screen center, ignore it if (DotProduct(theForward, theDir) < 0.9f) { // Determine left and right vec3_t theCrossProd = CrossProduct(theForward, theDir); if (theCrossProd[2] > 0.0f) this->mDisplayOrderDirection = 1; // Left else if (theCrossProd[2] < 0.0f) this->mDisplayOrderDirection = 2; // Right } } } void AvHHud::DrawTeammateOrders() { TeammateOrderListType::iterator toErase = this->mTeammateOrder.end();// to fix 2014 not sure wether it works correctly or not was NULL BEFORE cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer(); const float flashLength = 1.0f; const float fadeLimit = 6.0f; const float fadeEnd = 2.0f; for(TeammateOrderListType::iterator theIter = this->mTeammateOrder.begin(); theIter != this->mTeammateOrder.end(); theIter++) { float lastOrder = 0; TeammateOrderType theOrder = (*theIter).second; int theEntIndex = (*theIter).first; float theFade = 1.0f; // remove the order if it has expired if((theOrder.second + fadeEnd + fadeLimit) < this->mTimeOfLastUpdate) { toErase = theIter; continue; } // draw the order fading away else if((theOrder.second + fadeLimit) < this->mTimeOfLastUpdate) { theFade = 1.0f - (this->mTimeOfLastUpdate - (theOrder.second + fadeLimit)) / fadeEnd; if(theFade < 0.0f) theFade = 0.0f; } // else, draw the order normally cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(theEntIndex); if (theEntity && (theEntIndex < MAX_PLAYERS && theEntIndex >= 0) && (theEntity->index != theLocalPlayer->index)) { if (AvHTraceLineAgainstWorld(theLocalPlayer->origin, theEntity->origin) == 1.0f) { vec3_t theVec; VectorCopy(theEntity->origin, theVec); theVec[2] += AvHCUGetIconHeightForPlayer((AvHUser3)theEntity->curstate.iuser3); this->DrawWorldSprite(this->mTeammateOrderSprite, kRenderTransAdd, theVec, theOrder.first, kHelpIconDrawSize, theFade); if (lastOrder < theOrder.second) { lastOrder = theOrder.second; this->GetOrderDirection(theVec, 1); } } } } if (toErase != this->mTeammateOrder.end()) // to fix 2014 not sure wether it works correctly or not was NULL BEFORE this->mTeammateOrder.erase(toErase); // flash target player if (((this->mCurrentOrderTime + flashLength) > this->mTimeOfLastUpdate) && (this->mCurrentOrderTarget > 0)) { if (((int)((this->mTimeOfLastUpdate - (this->mCurrentOrderTime + flashLength)) * 8)) % 2) { cl_entity_s* theTargetEntity = gEngfuncs.GetEntityByIndex(this->mCurrentOrderTarget); vec3_t theVec; VectorCopy(theTargetEntity->origin, theVec); theVec[2] += AvHCUGetIconHeightForPlayer((AvHUser3)theTargetEntity->curstate.iuser3); this->DrawWorldSprite(this->mTeammateOrderSprite, kRenderTransAdd, theVec, this->mCurrentOrderType, kHelpIconDrawSize, 1.0f); } } } // : void AvHHud::DrawOrders() { if(1/*!this->mIsRenderingSelectionView*/) { // Draw them blinking for soldiers, but always for commanders float theFractionalLastUpdate = this->mTimeOfLastUpdate - (int)this->mTimeOfLastUpdate; if((theFractionalLastUpdate > .25f) || (this->GetHUDUser3() == AVH_USER3_COMMANDER_PLAYER)) { OrderListType theOrders = this->GetOrderList(); EntityListType theDrawPlayerList = this->GetDrawPlayerOrders(); // Run through the order list type for(OrderListType::iterator theIter = theOrders.begin(); theIter != theOrders.end(); theIter++) { // For each one, if the order is for a player in the theDrawPlayerList, draw it vec3_t theOrderLocation; theIter->GetLocation(theOrderLocation); if(theIter->GetOrderTargetType() == ORDERTARGETTYPE_TARGET) { int theTargetIndex = theIter->GetTargetIndex(); cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(theTargetIndex); if(theEntity) { //: dont follow if they are cloaked, leave the waypoint active so they have a clue where they may be at, the wp should snap back to the baddy //once they are spotted again. if(theEntity->curstate.rendermode != kRenderTransTexture && theEntity->curstate.renderamt > 128) VectorCopy(theEntity->origin, theOrderLocation); } } // Draw the order if the order is for any plays that are in our draw player list bool theDrawWaypoint = false; EntityInfo theReceiverPlayer = theIter->GetReceiver(); EntityListType::iterator theSearchIter = std::find(theDrawPlayerList.begin(), theDrawPlayerList.end(), theReceiverPlayer); if(theSearchIter != theDrawPlayerList.end() && *theSearchIter == theReceiverPlayer) { //gEngfuncs.pEfxAPI->R_ParticleLine((float*)theEntity->origin, (float*)theOrderLocation, 0, 255, 0, .05f); theDrawWaypoint = true; } if(theDrawWaypoint) { this->DrawOrderIcon(*theIter); this->DrawOrderText(*theIter); } } } } } int AvHHud::GetHelpIconFrameFromUser3(AvHUser3 inUser3) { int theFrame = -1; switch(inUser3) { case AVH_USER3_WELD: theFrame = 0; break; case AVH_USER3_MARINEITEM: theFrame = 1; break; case AVH_USER3_HIVE: theFrame = 2; break; //case AVH_USER3_USEABLE: // theFrame = 3; // break; case AVH_USER3_COMMANDER_STATION: theFrame = 4; break; //case AVH_USER3_BREAKABLE: // theFrame = 5; // break; } return theFrame; } AVHHSPRITE AvHHud::GetHelpSprite() const { return this->mHelpSprite; } void AvHHud::DrawHelpIcons() { // Lookup table // Only draw if enabled //if(gEngfuncs.pfnGetCvarFloat(kvAutoHelp)) //{ // Iterate through help icons, drawing each one if we have an sprite for it for(HelpIconListType::iterator theIter = this->mHelpIcons.begin(); theIter != this->mHelpIcons.end(); theIter++) { int theUser3 = theIter->second; int theFrame = GetHelpIconFrameFromUser3(AvHUser3(theUser3)); // Lookup sprite to see if it's loaded // if(this->mHelpSprites[theUser3] == 0) // { // string theIconName = string(kHelpIconPrefix) + MakeStringFromInt(theUser3) + ".spr"; // this->mHelpSprites[theUser3] = SPR_Load(theIconName.c_str()); // } // // int theSpriteHandle = this->mHelpSprites[theUser3]; // if(theSpriteHandle > 0) // { // // Draw icon at entity center // this->DrawWorldSprite(theSpriteHandle, kRenderTransAdd, theIter->first, 0); // } if((theFrame >= 0) && this->mHelpSprite) { this->DrawWorldSprite(this->mHelpSprite, kRenderTransAdd, theIter->first, theFrame, kHelpIconDrawSize); } } //} } // inDrawMode determines if we're drawing text or sprites void AvHHud::DrawHUDStructureNotification() { const float kHUDStructureNotificationStartX = .01f; const float kHUDStructureNotificationStartY = .08f; const float kHUDStructureNotificationIconWidth = .03f; const float kHUDStructureNotificationIconHeight = kHUDStructureNotificationIconWidth*1.333f; const float kHUDStructureNotificationIconHorizontalSpacing = .01f; const float kHUDStructureNotificationIconVerticalSpacing = kHUDStructureNotificationIconHorizontalSpacing*1.333f; //const float kHUDStructureNotificationMaxTextWidth = .2f; const float kTextHeightCenteringFactor = 0.25f; // Draw them all in order if(this->GetIsAlive() && CVAR_GET_FLOAT(kvBuildMessages)) { // Get starting coords float theCurrentX = kHUDStructureNotificationStartX; float theCurrentY = kHUDStructureNotificationStartY; bool isMarine = GetIsMarine(); float kSmallScaleFactor; //Don't make building icons smaller if alien. (isMarine) ? kSmallScaleFactor = 0.75f : kSmallScaleFactor = 1.0f; const float kIconWidthSmall = kHUDStructureNotificationIconWidth * kSmallScaleFactor; const float kIconHeightSmall = kHUDStructureNotificationIconHeight * kSmallScaleFactor; if (isMarine) { for (ResearchInfoListType::iterator theIter = this->mResearchInfoList.begin(); theIter != this->mResearchInfoList.end(); theIter++) { // Draw icon AvHMessageID theIconTech = theIter->mResearch; int theFrame = 0; int theStartX = (theCurrentX + kHUDStructureNotificationIconWidth + kHUDStructureNotificationIconHorizontalSpacing)*ScreenWidth(); string theResearchTimerText; ActionButton::GetLabelForMessage(theIconTech, theResearchTimerText); int timeLeft = theIter->mTimeResearchDone - this->mTimeOfCurrentUpdate; int theMinutesLeft = timeLeft / 60; int theSecondsLeft = timeLeft % 60; if (theMinutesLeft) theResearchTimerText += " - " + MakeStringFromInt(theMinutesLeft) + "m " + MakeStringFromInt(theSecondsLeft) + "s"; else theResearchTimerText += " - " + MakeStringFromInt(theSecondsLeft) + "s"; int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); char theCharBuffer[512]; sprintf(theCharBuffer, "%s", theResearchTimerText.c_str()); float theCurrentTextY = theCurrentY + (kHUDStructureNotificationIconHeight * kTextHeightCenteringFactor); this->DrawTechTreeSprite(theIconTech, theCurrentX*ScreenWidth(), theCurrentY*ScreenHeight(), kHUDStructureNotificationIconWidth*ScreenWidth(), kHUDStructureNotificationIconHeight*ScreenHeight(), theFrame); this->DrawHudString(theStartX, theCurrentTextY*ScreenHeight(), ScreenWidth(), theCharBuffer, theR, theG, theB); theCurrentY += (kHUDStructureNotificationIconHeight + kHUDStructureNotificationIconVerticalSpacing); } } for(StructureHUDNotificationListType::iterator theIter = this->mStructureNotificationList.begin(); theIter != this->mStructureNotificationList.end(); theIter++) { // Draw icon AvHMessageID theIconTech = theIter->mStructureID; int theFrame = 0; string theLocationName = this->GetNameOfLocation(theIter->mLocation); int theStartX = (theCurrentX + kHUDStructureNotificationIconWidth + kHUDStructureNotificationIconHorizontalSpacing)*ScreenWidth(); bool isResearch = AvHSHUGetIsResearchTech(theIconTech); if (!isResearch) { this->DrawTechTreeSprite(theIconTech, theCurrentX*ScreenWidth(), theCurrentY*ScreenHeight(), kIconWidthSmall*ScreenWidth(), kIconHeightSmall*ScreenHeight(), theFrame); if (theLocationName != "") { int theStartXsmall = theStartX * kSmallScaleFactor; float theCurrentTextY = theCurrentY + (kIconHeightSmall * kTextHeightCenteringFactor); this->DrawTranslatedString(theStartXsmall, theCurrentTextY*ScreenHeight(), theLocationName.c_str(), false, true); } // Increment coords theCurrentY += (kIconHeightSmall + kHUDStructureNotificationIconVerticalSpacing); } } } } void AvHHud::DrawInfoLocationText() { string theTimeLeftText; // Get drawing color and position int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); // Get position of health and draw to the right of it (must be left-justified) int theX = 0; int theY = 0; if(this->GetInTopDownMode()) { const float kLeftInset = .4f; theX = mViewport[0] + kLeftInset*ScreenWidth(); theY = mViewport[1] + mViewport[3] - (1-.78f)*ScreenHeight(); } // Draw info location text the same for aliens and marines else if(this->GetHUDUser3() != AVH_USER3_ALIEN_EMBRYO) { const float kLeftInset = .08f; theX = mViewport[0] + kLeftInset*ScreenWidth(); theY = mViewport[1] + mViewport[3] - (1-.88f)*ScreenHeight(); } int theLeftEdge = theX; // Draw location text, translation if possible string theLocalizedLocationText; if(this->mLocationText != "") { LocalizeString(this->mLocationText.c_str(), theLocalizedLocationText); if(theLocalizedLocationText[0] == '#') { theLocalizedLocationText = theLocalizedLocationText.substr(1, theLocalizedLocationText.size()); } // Draw handicap text as well int theHandicap = (int)this->GetHUDHandicap(); if(theHandicap < 100) { // draw "(handicap 70%)" string theHandicapString; if(LocalizeString(kHandicap, theHandicapString)) { char theFormattedHandicapString[256]; sprintf(theFormattedHandicapString, " (%d%% %s)", theHandicap, theHandicapString.c_str()); theLocalizedLocationText += string(theFormattedHandicapString); } } // Draw info location if(theLocalizedLocationText != "") { char theCharArray[512]; sprintf(theCharArray, "%s", theLocalizedLocationText.c_str()); this->DrawHudString(theX, theY, ScreenWidth(), theCharArray, theR, theG, theB); } } // Don't draw time when playing back, it isn't right and not worth breaking backward-compatibility over. // TODO: Draw time next time demo version changes if(!gEngfuncs.pDemoAPI->IsPlayingback() && this->GetHUDUser3() != AVH_USER3_ALIEN_EMBRYO) { // Whether we draw first line or not, increment for second line below theY += this->GetHudStringHeight(); // Optionally draw time left below it int theTimeLimitSeconds = (int)this->GetGameTimeLimit(); int theTimeElapsed = this->GetGameTime(); int theMinutesElapsed = theTimeElapsed/60; int theSecondsElapsed = theTimeElapsed%60; int theTimeLeftSeconds = theTimeLimitSeconds - theTimeElapsed; int theMinutesLeft = theTimeLeftSeconds/60; int theSecondsLeft = theTimeLeftSeconds%60; int theMinutesLimit = theTimeLimitSeconds/60; int theSecondsLimit = theTimeLimitSeconds%60; // If we're in tournament mode or playing Combat, draw the timelimit bool theTournyMode = (gHUD.GetServerVariableFloat(kvTournamentMode) > 0); bool theDisplayTimeLimit = theTournyMode || gHUD.GetIsCombatMode(); string theGameTimeText; if(LocalizeString(kGameTime, theGameTimeText)) { bool theTimeVisible = true; // Flash time when we're almost out of time if((theMinutesLeft < 1) && this->GetGameStarted() && theDisplayTimeLimit) { float theTime = gHUD.GetTimeOfLastUpdate(); float theTimeFraction = theTime - floor(theTime); if(theTimeFraction < .5f) { theTimeVisible = false; } } // Game time - 12:43 char theGameTimeCStr[256]; sprintf(theGameTimeCStr, " - %02d:%02d", theMinutesElapsed, theSecondsElapsed); theGameTimeText += string(theGameTimeCStr); if(theTimeVisible) { this->DrawHudString(theX, theY, ScreenWidth(), theGameTimeText.c_str(), theR, theG, theB); } // Increment X so time limit is drawn properly theX += this->GetHudStringWidth(theGameTimeText.c_str()); } if(theDisplayTimeLimit) { string theTimeLimitString; if(LocalizeString(kTimeLimit, theTimeLimitString)) { // Format: Time limit - 60:00 char theTimeLimitCStr[256]; sprintf(theTimeLimitCStr, " %s - %02d:%02d", theTimeLimitString.c_str(), theMinutesLimit, theSecondsLimit); string theTimeLimitText = string(theTimeLimitCStr); this->DrawHudString(theX, theY, ScreenWidth(), theTimeLimitText.c_str(), theR, theG, theB); } if(gHUD.GetIsCombatMode()) { theY += this->GetHudStringHeight(); // Draw "attacking" or "defending" AvHTeamNumber theTeamNumber = this->GetHUDTeam(); string theDisplayString = ""; if(theTeamNumber != TEAM_IND) { if(theTeamNumber == this->GetCombatAttackingTeamNumber()) { LocalizeString(kAttacking, theDisplayString); } else { LocalizeString(kDefending, theDisplayString); } this->DrawHudString(theLeftEdge, theY, ScreenWidth(), theDisplayString.c_str(), theR, theG, theB); } } } } } void AvHHud::DrawMouseCursor(int inBaseX, int inBaseY) { if ( g_iVisibleMouse && !(this->GetInTopDownMode() && gEngfuncs.pDemoAPI->IsPlayingback()) ) { AVHHSPRITE theCursorSprite; int theCursorFrame; GetCursor(theCursorSprite, theCursorFrame); if (theCursorSprite > 0) { float theGammaSlope = this->GetGammaSlope(); ASSERT(theGammaSlope > 0.0f); /* int theColorComponent = 255/theGammaSlope; SPR_Set(this->mCursorSprite, theColorComponent, theColorComponent, theColorComponent); // Draw the logo at 20 fps //SPR_DrawAdditive( 0, this->mMouseCursorX - inBaseX, this->mMouseCursorY - inBaseY, NULL ); const int kMouseHotSpotX = -1; const int kMouseHotSpotY = -1; SPR_DrawHoles(this->mCurrentCursorFrame, this->mMouseCursorX - inBaseX - kMouseHotSpotX, this->mMouseCursorY - inBaseY - kMouseHotSpotY, NULL ); */ // Draw the mouse cursor. const int kMouseHotSpotX = -1; const int kMouseHotSpotY = -1; AvHSpriteBeginFrame(); AvHSpriteEnableVGUI(true); AvHSpriteSetVGUIOffset(inBaseX, inBaseY); float x = this->mMouseCursorX - kMouseHotSpotX; float y = this->mMouseCursorY - kMouseHotSpotY; float w = SPR_Width(theCursorSprite, theCursorFrame); float h = SPR_Height(theCursorSprite, theCursorFrame); AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(theCursorSprite, theCursorFrame, x, y, x + w, y + h, 0, 0, 1, 1); // Draw the marquee if it's visible. if (mSelectionBoxVisible) { AVHHSPRITE sprite = SPR_Load(kWhiteSprite); int r, g, b; GetPrimaryHudColor(r, g, b, true, false); AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteSetColor(r / 255.0, g / 255.0, b / 255.0, 0.3); AvHSpriteSetDrawMode(kSpriteDrawModeFilled); AvHSpriteDraw(sprite, 0, mSelectionBoxX1 + 1, mSelectionBoxY1 + 1, mSelectionBoxX2 - 1, mSelectionBoxY2 - 1, 0, 0, 1, 1); AvHSpriteSetRenderMode(kRenderNormal); AvHSpriteSetColor(1, 1, 1, 1); AvHSpriteSetDrawMode(kSpriteDrawModeBorder); AvHSpriteDraw(sprite, 0, mSelectionBoxX1, mSelectionBoxY1, mSelectionBoxX2, mSelectionBoxY2, 0, 0, 1, 1); } AvHSpriteEndFrame(); } } } void AvHHud::DrawTopDownBG() { // If we're in top down mode, draw a black background float theDrawBackgroundHeight = -1;//cl_drawbg->value; // Draw the bottom plane at the map's min view height, unless overridden by cl_drawbg (this variable is for for testing purposes only) if(theDrawBackgroundHeight == -1) { if(this->mMapExtents.GetDrawMapBG()) { theDrawBackgroundHeight = this->mMapExtents.GetMinViewHeight(); } } if(theDrawBackgroundHeight != -1) { if(this->mBackgroundSprite) { // Build three points on plane described by max world dimensions at the draw background height Vector theUpperLeftWorldPos(-kMaxMapDimension, kMaxMapDimension, theDrawBackgroundHeight); Vector theUpperRightWorldPos(kMaxMapDimension, kMaxMapDimension, theDrawBackgroundHeight); Vector theLowerLeftWorldPos(-kMaxMapDimension, -kMaxMapDimension, theDrawBackgroundHeight); // Calculate plane info float thePlaneD; Vector thePlaneNormal; CalculatePlaneInfo(theUpperLeftWorldPos, theUpperRightWorldPos, theLowerLeftWorldPos, thePlaneD, thePlaneNormal); gEngfuncs.pTriAPI->RenderMode( kRenderNormal ); gEngfuncs.pTriAPI->CullFace( TRI_NONE ); gEngfuncs.pTriAPI->Brightness( 1 ); if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s *)gEngfuncs.GetSpritePointer(this->mBackgroundSprite), 0)) { gEngfuncs.pTriAPI->Begin( TRI_TRIANGLE_STRIP ); gEngfuncs.pTriAPI->Color4f(1.0f, 1.0f, 1.0f, 1.0f); Vector thePoint; CalculatePointOnPlane(0, ScreenHeight(), GetViewOrigin(), thePlaneNormal, thePlaneD, thePoint); gEngfuncs.pTriAPI->TexCoord2f(0, 1); gEngfuncs.pTriAPI->Vertex3fv((float*)&thePoint); CalculatePointOnPlane(0, 0, GetViewOrigin(), thePlaneNormal, thePlaneD, thePoint); gEngfuncs.pTriAPI->TexCoord2f(0, 0); gEngfuncs.pTriAPI->Vertex3fv((float*)&thePoint); CalculatePointOnPlane(ScreenWidth(), ScreenHeight(), GetViewOrigin(), thePlaneNormal, thePlaneD, thePoint); gEngfuncs.pTriAPI->TexCoord2f(1, 1); gEngfuncs.pTriAPI->Vertex3fv((float*)&thePoint); CalculatePointOnPlane(ScreenWidth(), 0, GetViewOrigin(), thePlaneNormal, thePlaneD, thePoint); gEngfuncs.pTriAPI->TexCoord2f(1, 0); gEngfuncs.pTriAPI->Vertex3fv((float*)&thePoint); // Add in some buffer because of perspective // pVector ver; // ver.z = theDrawBackgroundHeight; // // // Lower left // gEngfuncs.pTriAPI->TexCoord2f(0,1); // ver.x = -kMaxMapDimension; // ver.y = -kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Upper left // gEngfuncs.pTriAPI->TexCoord2f(0,0); // ver.x = -kMaxMapDimension; // ver.y = kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Upper right // gEngfuncs.pTriAPI->TexCoord2f(1,0); // ver.x = kMaxMapDimension; // ver.y = kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Lower right // gEngfuncs.pTriAPI->TexCoord2f(1,1); // ver.x = kMaxMapDimension; // ver.y = -kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); gEngfuncs.pTriAPI->End(); } } } } void AvHHud::PostModelRender(char* inModelName) { // Get our view model name cl_entity_t* theViewEntity = gEngfuncs.GetViewModel(); if(theViewEntity) { // If this is our view model, that means we can now render our own stuff in screen space without z-buffering on top of everything else if(theViewEntity->model) { if(!strcmp(theViewEntity->model->name, inModelName)) { this->RenderNoZBuffering(); } } } } float AvHHud::GetHUDExperience() const { vec3_t theVUser4; theVUser4.z = 0.0f; cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer(); if(theLocalPlayer) { theVUser4 = theLocalPlayer->curstate.vuser4; } if(g_iUser1 == OBS_IN_EYE) { cl_entity_t* theEnt = gEngfuncs.GetEntityByIndex(g_iUser2); if(theEnt) { theVUser4 = theEnt->curstate.vuser4; } } float theExperience = theVUser4.z/kNumericNetworkConstant; return theExperience; } int AvHHud::GetHUDExperienceLevel() const { float theExperience = this->GetHUDExperience(); int theLevel = AvHPlayerUpgrade::GetPlayerLevel(theExperience); return theLevel; } float AvHHud::GetHUDHandicap() const { float theHandicap = 100.0f; AvHTeamNumber theTeamNumber = this->GetHUDTeam(); switch(theTeamNumber) { case TEAM_ONE: theHandicap = this->GetServerVariableFloat(kvTeam1DamagePercent); break; case TEAM_TWO: theHandicap = this->GetServerVariableFloat(kvTeam2DamagePercent); break; case TEAM_THREE: theHandicap = this->GetServerVariableFloat(kvTeam3DamagePercent); break; case TEAM_FOUR: theHandicap = this->GetServerVariableFloat(kvTeam4DamagePercent); break; } return theHandicap; } AvHUser3 AvHHud::GetHUDUser3() const { AvHUser3 theUser3 = AVH_USER3_NONE; cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer(); if(theLocalPlayer) { theUser3 = (AvHUser3)(theLocalPlayer->curstate.iuser3); } if(g_iUser1 == OBS_IN_EYE) { cl_entity_t* theEnt = gEngfuncs.GetEntityByIndex(g_iUser2); if(theEnt) { theUser3 = (AvHUser3)(theEnt->curstate.iuser3); } } return theUser3; } AvHTeamNumber AvHHud::GetHUDTeam() const { AvHTeamNumber theTeamNumber = TEAM_IND; cl_entity_s* thePlayer = this->GetVisiblePlayer(); if(thePlayer) { theTeamNumber = AvHTeamNumber(thePlayer->curstate.team); } return theTeamNumber; } int AvHHud::GetHUDUpgrades() const { int theUpgrades = 0; cl_entity_s* thePlayer = this->GetVisiblePlayer(); if(thePlayer) { theUpgrades = thePlayer->curstate.iuser4; } return theUpgrades; } int AvHHud::GetHUDMaxArmor() const { int theHUDUpgrades = this->GetHUDUpgrades(); AvHUser3 theUser3 = this->GetHUDUser3(); int theMaxArmor = AvHPlayerUpgrade::GetMaxArmorLevel(theHUDUpgrades, theUser3); return theMaxArmor; } int AvHHud::GetHUDMaxHealth() const { int theHUDUpgrades = this->GetHUDUpgrades(); AvHUser3 theUser3 = this->GetHUDUser3(); int theHUDExperienceLevel = this->GetHUDExperienceLevel(); int theMaxHealth = AvHPlayerUpgrade::GetMaxHealth(theHUDUpgrades, theUser3, theHUDExperienceLevel); return theMaxHealth; } void AvHHud::GetPrimaryHudColor(int& outR, int& outG, int& outB, bool inIgnoreUpgrades, bool gammaCorrect) const { if(this->GetIsMarine()) { UnpackRGB(outR, outG, outB, RGB_MARINE_BLUE); } else { // HUD turns green when we're frenzying //if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_PRIMALSCREAM) && !inIgnoreUpgrades) //{ // UnpackRGB(outR, outG, outB, RGB_GREENISH); //} //else //{ UnpackRGB(outR, outG, outB, RGB_YELLOWISH); //} } if (gammaCorrect) { // Take into account current gamma? float theGammaSlope = this->GetGammaSlope(); outR /= theGammaSlope; outG /= theGammaSlope; outB /= theGammaSlope; } } void AvHHud::HandleFog() { float theFogColor[3]; theFogColor[0] = this->mFogColor.x; theFogColor[1] = this->mFogColor.y; theFogColor[2] = this->mFogColor.z; gEngfuncs.pTriAPI->Fog(theFogColor, this->mFogStart, this->mFogEnd, this->mFogActive); } void AvHHud::PreRenderFrame() { bool theRenderForTopDown = this->mInTopDownMode /*&& !this->mIsRenderingSelectionView*/; if(!theRenderForTopDown) { this->HandleFog(); } else { this->DrawTopDownBG(); // Now draw commander HUD scaled //void CreatePickingRay( int mousex, int mousey, Vector& outVecPickingRay ) // static int theCommanderHUDSprite = 0; // if(!theCommanderHUDSprite) // { // theCommanderHUDSprite = SPR_Load("sprites/.spr"); // } // // if(theCommanderHUDSprite) // { // gEngfuncs.pTriAPI->RenderMode( kRenderNormal ); // gEngfuncs.pTriAPI->CullFace( TRI_NONE ); // gEngfuncs.pTriAPI->Brightness( 1 ); // // if(gEngfuncs.pTriAPI->SpriteTexture((struct model_s *)gEngfuncs.GetSpritePointer(theCommanderHUDSprite), 0)) // { // gEngfuncs.pTriAPI->Begin( TRI_QUADS ); // // gEngfuncs.pTriAPI->Color4f(1.0f, 1.0f, 1.0f, .5f); // // pVector ver; // // // Lower left // const float kMaxMapDimension = 4096; // //ver.z = -kMaxMapDimension; // ver.z = theDrawBackgroundHeight; // // gEngfuncs.pTriAPI->TexCoord2f(0,1); // ver.x = -kMaxMapDimension; // ver.y = -kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Upper left // gEngfuncs.pTriAPI->TexCoord2f(0,0); // ver.x = -kMaxMapDimension; // ver.y = kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Upper right // gEngfuncs.pTriAPI->TexCoord2f(1,0); // ver.x = kMaxMapDimension; // ver.y = kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // // Lower right // gEngfuncs.pTriAPI->TexCoord2f(1,1); // ver.x = kMaxMapDimension; // ver.y = -kMaxMapDimension; // gEngfuncs.pTriAPI->Vertex3fv((float*)&ver); // // gEngfuncs.pTriAPI->End(); // } // } } AvHScriptManager::Instance()->DrawNormal(); } void AvHHud::PostRenderFrame() { // Restore player angles and view position // v_origin = gPlayerOrigin; // v_angles = gPlayerAngles; } void AvHHud::DrawActionButtons() { // Look up AvHActionButtons component AvHActionButtons* theActionButtons = NULL; if(this->GetManager().GetVGUIComponentNamed(kActionButtonsComponents, theActionButtons)) { int theNumCols = theActionButtons->GetNumCols(); int theNumRows = theActionButtons->GetNumRows(); // Iterate through num cols for(int theCol = 0; theCol < theNumCols; theCol++) { // Iterate through num rows for(int theRow = 0; theRow < theNumRows; theRow++) { // Get current ActionButton ActionButton* theActionButton = theActionButtons->GetActionButtonAtPos(theCol, theRow); ASSERT(theActionButton); // Get message ID for button AvHMessageID theMessageID = theActionButton->GetMessageID(); // Find the group that it belongs to (20, 30, 40, etc.) int theMessageNumber = (int)theMessageID - (theMessageID % 10); // // Load sprite if not loaded // bool theSpriteLoaded = (this->mActionButtonSprites[theMessageNumber] > 0); // if(!theSpriteLoaded) // { // // Build sprite name for message ID // char theMessageNumberString[16]; // sprintf(theMessageNumberString, "%d", (int)theMessageNumber); // //string theSpriteName = kTechTreeSpriteDirectory + string("/") + kTechTreeSpritePrefix + string(theMessageIDString) + string(".spr"); // string theSpriteName = kTechTreeSpriteDirectory + string("/") + kTechTreeSpritePrefix + string(theMessageNumberString) + string("s.spr"); // int theSpriteHandle = SPR_Load(theSpriteName.c_str()); // // // Sprite handle can be 0, as I don't have sprites for all tech yet // this->mActionButtonSprites[theMessageNumber] = theSpriteHandle; // } // Get pos and size for component int thePosX, thePosY; theActionButton->getPos(thePosX, thePosY); int theWidth, theHeight; theActionButton->getSize(theWidth, theHeight); // Set sprite frame depending if button is available, active bool theTechEnabled = theActionButton->GetTechEnabled(); bool theMouseOver = theActionButton->GetMouseOver(); bool theCostMet = theActionButton->GetCostMet(); // 0 = default // 1 = highlighted // 2 = dark // 3 = red int theFrame = 2; // If it's enabled, or if it's an icon in the "menu navigation" category, allow it to be highlighted if(theTechEnabled || (theMessageNumber == 80)) { if(theCostMet) { if(theMouseOver) { theFrame = 1; } else { theFrame = 0; } } else { theFrame = 3; } } // Also highlight menu category headings for open menus, and for active nodes just pressed if(gCommanderHandler.GetDisplayMenuMessageID() == theMessageID) { theFrame = 1; } else { AvHMessageID theMessageJustActivated = MESSAGE_NULL; if(gCommanderHandler.GetAndClearTechNodePressed(theMessageJustActivated, false)) { if(theMessageJustActivated == theMessageID) { if(theTechEnabled) { theFrame = 1; } } } } this->DrawTechTreeSprite(theMessageID, thePosX, thePosY, theWidth, theHeight, theFrame); } } } } AVHHSPRITE AvHHud::GetTechTreeSprite(AvHMessageID inMessageID) { // Find the group that it belongs to (20, 30, 40, etc.) int theMessageNumber = (int)inMessageID - (inMessageID % 10); // Load sprite if not loaded bool theSpriteLoaded = (this->mActionButtonSprites[theMessageNumber] > 0); if(!theSpriteLoaded && (theMessageNumber != 0)) { // Build sprite name for message ID char theMessageNumberString[16]; sprintf(theMessageNumberString, "%d", (int)theMessageNumber); //string theSpriteName = kTechTreeSpriteDirectory + string("/") + kTechTreeSpritePrefix + string(theMessageIDString) + string(".spr"); string theSpriteName; if (CVAR_GET_FLOAT("hud_style") == 2.0f) theSpriteName = kTechTreeSpriteDirectoryNL + string("/") + kTechTreeSpritePrefix + string(theMessageNumberString) + string("s.spr"); else theSpriteName = kTechTreeSpriteDirectory + string("/") + kTechTreeSpritePrefix + string(theMessageNumberString) + string("s.spr"); AVHHSPRITE theSpriteHandle = SPR_Load(theSpriteName.c_str()); // Sprite handle can be 0, as I don't have sprites for all tech yet this->mActionButtonSprites[theMessageNumber] = theSpriteHandle; } // Fetch sprite handle AVHHSPRITE theSpriteHandle = this->mActionButtonSprites[theMessageNumber]; return theSpriteHandle; } void AvHHud::DrawTechTreeSprite(AvHMessageID inMessageID, int inPosX, int inPosY, int inWidth, int inHeight, int inFrame) { if(inMessageID != MESSAGE_NULL) { // Check for alien sprites bool theIsAlienSprite = false; AVHHSPRITE theSpriteHandle = 0; int theRenderMode = kRenderTransAlpha; // kRenderNormal switch(inMessageID) { case ALIEN_BUILD_DEFENSE_CHAMBER: theIsAlienSprite = true; inFrame = 0; break; case ALIEN_BUILD_OFFENSE_CHAMBER: theIsAlienSprite = true; inFrame = 1; break; case ALIEN_BUILD_MOVEMENT_CHAMBER: theIsAlienSprite = true; inFrame = 2; break; case ALIEN_BUILD_SENSORY_CHAMBER: theIsAlienSprite = true; inFrame = 3; break; case ALIEN_BUILD_RESOURCES: theIsAlienSprite = true; inFrame = 4; break; case ALIEN_BUILD_HIVE: theIsAlienSprite = true; inFrame = 5; break; } float theStartU = 0.0f; float theStartV = 0.0f; float theEndU = 1.0f; float theEndV = 1.0f; if(theIsAlienSprite) { theRenderMode = kRenderTransAlpha; theSpriteHandle = this->mAlienUIUpgradeCategories; } else { // Fetch sprite handle theSpriteHandle = this->GetTechTreeSprite(inMessageID); int theIndex = inMessageID % 10; int theXIndex = theIndex % 4; int theYIndex = (theIndex / 4); theStartU = theXIndex*.25f; theStartV = theYIndex*.25f; theEndU = (theXIndex+1)*.25f; theEndV = (theYIndex+1)*.25f; } if(theSpriteHandle > 0) { // Draw sprite scaled here AvHSpriteSetRenderMode(theRenderMode); AvHSpriteDraw(theSpriteHandle, inFrame, inPosX, inPosY, inPosX + inWidth, inPosY + inHeight, theStartU, theStartV, theEndU, theEndV); } } } void AvHHud::DrawHotgroups() { AvHMessageID theHotgroups[kNumHotkeyGroups] = {GROUP_SELECT_1, GROUP_SELECT_2, GROUP_SELECT_3, GROUP_SELECT_4, GROUP_SELECT_5}; // Run through list of hotgroups, drawing them if they exist for(int i = 0; i < kNumHotkeyGroups; i++) { EntityListType& theHotgroup = this->mGroups[i]; if(theHotgroup.size() > 0) { // Get message ID to draw AvHUser3 theGroupTypeUser3 = this->mGroupTypes[i]; // Default is marine face AvHMessageID theHotgroupIcon = MENU_SOLDIER_FACE; AvHSHUUser3ToMessageID(theGroupTypeUser3, theHotgroupIcon); // Build component name ("PendingImpulseXPanel", where X is the impulse) AvHMessageID theHotgroupMessageID = theHotgroups[i]; char theComponentName[256]; sprintf(theComponentName, kPendingImpulseSpecifier, (int)theHotgroupMessageID); AvHTechImpulsePanel* theTechImpulsePanel = NULL; if(this->GetManager().GetVGUIComponentNamed(theComponentName, theTechImpulsePanel)) { int thePosX, thePosY; theTechImpulsePanel->getPos(thePosX, thePosY); int theWidth, theHeight; theTechImpulsePanel->getSize(theWidth, theHeight); // Highlight if mouse is in region or if it's our current selection bool theIsHighlighted = this->GetIsMouseInRegion(thePosX, thePosY, theWidth, theHeight) || (this->mSelected == theHotgroup); int theFrame = theIsHighlighted ? 1 : 0; int theSecondOfLastUpdate = (int)this->mTimeOfLastUpdate; if((this->mGroupAlerts[i] == ALERT_UNDER_ATTACK) && (theSecondOfLastUpdate % 2)) { // Play blinking red frame when being attacked theFrame = 3; } this->DrawTechTreeSprite(theHotgroupIcon, thePosX, thePosY, theWidth, theHeight, theFrame); theTechImpulsePanel->SetVisualNumber((int)theHotgroup.size()); } } } } // Draw pending requests void AvHHud::DrawPendingRequests() { // Run through list of requests and delete any entries with PendingRequestListType::iterator theIterator; for(theIterator = this->mPendingRequests.begin(); theIterator != this->mPendingRequests.end(); /* nothing */) { if(theIterator->second == 0) { PendingRequestListType::iterator theSavedIter = theIterator; theSavedIter++; this->mPendingRequests.erase(theIterator); theIterator = theSavedIter; } else { theIterator++; } } int theNumRequestsToDraw = (int)this->mPendingRequests.size(); int theCounter = 0; for(theIterator = this->mPendingRequests.begin(); theIterator != this->mPendingRequests.end(); theIterator++) { // Draw each one above the minimap AvHMessageID theMessageID = theIterator->first; int theNumRequests = theIterator->second; // Build component name ("PendingImpulseXPanel", where X is the impulse) char theComponentName[256]; sprintf(theComponentName, kPendingImpulseSpecifier, (int)theMessageID); AvHTechImpulsePanel* theTechImpulsePanel = NULL; if(this->GetManager().GetVGUIComponentNamed(theComponentName, theTechImpulsePanel)) { // Get size of panel and draw tech sprite there int thePosX, thePosY; theTechImpulsePanel->getPos(thePosX, thePosY); int theWidth, theHeight; theTechImpulsePanel->getSize(theWidth, theHeight); AvHMessageID theMessageToDraw = theMessageID; switch(theMessageID) { case COMMANDER_NEXTIDLE: theMessageToDraw = MENU_SOLDIER_FACE; break; case COMMANDER_NEXTAMMO: theMessageToDraw = BUILD_AMMO; break; case COMMANDER_NEXTHEALTH: theMessageToDraw = BUILD_HEALTH; break; } int theFrame = this->GetIsMouseInRegion(thePosX, thePosY, theWidth, theHeight) ? 1 : 0; this->DrawTechTreeSprite(theMessageToDraw, thePosX, thePosY, theWidth, theHeight, theFrame); theTechImpulsePanel->SetVisualNumber(theNumRequests); if(theNumRequests > 0) { theTechImpulsePanel->setVisible(true); } else { theTechImpulsePanel->setVisible(false); } //this->DrawHUDNumber(thePosX + kTileWidth, thePosY + kTileHeight, DHN_2DIGITS, theNumRequests); } // Draw indicator on sprite showing how many requests there are theCounter++; } } // : 0000988 void AvHHud::DrawBuildHealthEffectsForEntity(int inEntityIndex, float inAlpha) // : { if ( this->GetHUDPlayMode() == PLAYMODE_READYROOM ) return; // Get entity int theUser3 = 0; int theUser4 = 0; float theFuser1 = 0.0f; int theEntityTeam = 0; bool theIsOnOurTeam=false; vec3_t theOrigin; vec3_t theMins; vec3_t theMaxs; bool theContinue = false; float theHealthPercentage = 0.0f; double theDistanceToEntity = 0; cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(inEntityIndex); bool theEntityIsPlayer = ((inEntityIndex > 0) && (inEntityIndex <= gEngfuncs.GetMaxClients())); if(theEntity) { theUser3 = theEntity->curstate.iuser3; theUser4 = theEntity->curstate.iuser4; theFuser1 = theEntity->curstate.fuser1; theEntityTeam = theEntity->curstate.team; theIsOnOurTeam = (theEntityTeam == (int)this->GetHUDTeam()); //theOrigin = theEntity->curstate.origin; theOrigin = AvHSHUGetRealLocation(theEntity->origin, theEntity->curstate.mins, theEntity->curstate.maxs); if(theEntity->player) { // Subtract half-height to be at feet float theHeight = theEntity->curstate.maxs[2] - theEntity->curstate.mins[2]; theOrigin[2] -= theHeight/2.0f; } theDistanceToEntity = VectorDistance(gPredictedPlayerOrigin, theOrigin); theMins = theEntity->curstate.mins; theMaxs = theEntity->curstate.maxs; theHealthPercentage = theEntity->curstate.fuser2/kNormalizationNetworkFactor; // : 991 transmit armour and health for marines if ( GetIsMarine() && theEntityIsPlayer && theIsOnOurTeam ) { int tmpPercent=theEntity->curstate.fuser2; if ( GetInTopDownMode() ) { theHealthPercentage = (float)(tmpPercent&0x7F)/100; } else { theHealthPercentage = (float)(tmpPercent >> 7)/100; } } // : theContinue = true; // 991: // Do not display health rings for enemy players unless we are commander if ( !GetInTopDownMode() && theEntityIsPlayer && !theIsOnOurTeam ) { theContinue=false; } } // Get local player cl_entity_s* theLocalPlayer = gEngfuncs.GetLocalPlayer(); bool theDrewBuildInProgress = false; float theOversizeScalar = 1.0f; if(AvHSHUGetDrawRingsForUser3((AvHUser3)theUser3, theOversizeScalar)) { if(theContinue && theLocalPlayer) { const int kDrawEnemyBuildingDistance = 200; bool healthLowEnough = theHealthPercentage < (CVAR_GET_FLOAT("hud_teamhealthalert") * 0.01f); // Draw effects if we are in top-down mode OR if( this->GetInTopDownMode() || // It's an unfriendly building that's very close OR (!theEntityIsPlayer && (theDistanceToEntity < kDrawEnemyBuildingDistance)) || // It's a friendly building and we're a builder OR //(theIsOnOurTeam && (this->GetHUDUser3() == AVH_USER3_ALIEN_PLAYER2)) || (theIsOnOurTeam && !theEntityIsPlayer && (this->GetHUDUser3() == AVH_USER3_ALIEN_PLAYER2)) || // It's a friendly player with <95% armor/health and we have a welder in our inventory or are gorge OR (theIsOnOurTeam && theEntityIsPlayer && healthLowEnough && (this->mHasWelder || this->GetHUDUser3() == AVH_USER3_ALIEN_PLAYER2)) || // welder/healing spray is selected (this->mCurrentWeaponID == AVH_WEAPON_WELDER || this->mCurrentWeaponID == AVH_WEAPON_HEALINGSPRAY) ) { // If they're not on opposite teams //if((theEntityTeam == theLocalPlayer->curstate.team) || (theEntityTeam == 0)) //{ // Get entity build/research state bool theIsBuilding = false; bool theIsResearching = false; float theNormalizedPercentage = 0.0f; // Calculate percentage full AvHSHUGetBuildResearchState(theUser3, theUser4, theFuser1, theIsBuilding, theIsResearching, theNormalizedPercentage); bool theDrawHealth = true; AVHHSPRITE theSpriteToUse = this->GetIsAlien() ? this->mAlienHealthSprite : this->mMarineHealthSprite; bool theDrawAsRecyling = (GetHasUpgrade(theUser4, MASK_RECYCLING) && theIsOnOurTeam); if((theIsOnOurTeam && theIsBuilding && (GetHasUpgrade(theUser4, MASK_BUILDABLE))) || theDrawAsRecyling) { theSpriteToUse = this->GetIsAlien() ? this->mAlienBuildSprite : this->mMarineBuildSprite; theDrawHealth = false; // Draw progress inverted, as we're "undoing" the structure if(theDrawAsRecyling) { theNormalizedPercentage = (1.0f - theNormalizedPercentage); } } else { theNormalizedPercentage = theHealthPercentage; } if(theDrawHealth || (theEntityTeam == theLocalPlayer->curstate.team)) { // Draw it int theNumFrames = SPR_Frames(theSpriteToUse); ASSERT(theNumFrames > 0); int theCurrentFrame; if ((theHealthPercentage < 1.0f) || theDrawAsRecyling) { theCurrentFrame = theNormalizedPercentage * (theNumFrames - 1); // : 0000893 // quick hack to eliminate 1 red bar shown for dead players if (theEntity->player) theCurrentFrame = min(max(0, theCurrentFrame), theNumFrames - 1); else theCurrentFrame = min(max(1, theCurrentFrame), theNumFrames - 1); // : } else { theCurrentFrame = theNumFrames - 1; } // Get real position of entity vec3_t thePosition; thePosition = AvHSHUGetRealLocation(theOrigin, theMins, theMaxs); float theMaxX = theMaxs[0] - theMins[0]; float theMaxY = theMaxs[1] - theMins[1]; float theRadius = max(theMaxX, theMaxY)*theOversizeScalar; // Bring position off ground just a little to prevent z-fighting thePosition.z += kZFightingOffset; //GetHasUpgrade(this->GetHUDUpgrades(), MASK_PARASITED) DrawSpriteOnGroundAtPoint(thePosition, theRadius, theSpriteToUse, kRenderTransAdd, theCurrentFrame, inAlpha); theDrewBuildInProgress = true; } } } } } void AvHHud::DrawHUDNumber(int inX, int inY, int inFlags, int inNumber) { // Draw number on HUD, in our HUD color int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); int theGammaSlope = this->GetGammaSlope(); theR /= theGammaSlope; theG /= theGammaSlope; theB /= theGammaSlope; this->DrawHudNumber(inX, inY, inFlags, inNumber, theR, theG, theB); } void AvHHud::DrawSelectionAndBuildEffects() { // : 0000988 list theSelectedList; // : // Draw build effects for(SelectionListType::iterator theSelectIter = this->mSelectionEffects.begin(); theSelectIter != this->mSelectionEffects.end(); theSelectIter++) { // Draw selection effect around the entity int theEntIndex = theSelectIter->mEntIndex; this->DrawBuildHealthEffectsForEntity(theEntIndex); // : 0000988 theSelectedList.push_back(theEntIndex); // : } bool theDrawBuildingEffect = false; for(EntityListType::iterator theBuildingIter = this->mBuildingEffectsEntityList.begin(); theBuildingIter != this->mBuildingEffectsEntityList.end(); theBuildingIter++) { int theEntIndex = *theBuildingIter; this->DrawBuildHealthEffectsForEntity(theEntIndex); // : 0000988 theSelectedList.push_back(theEntIndex); // : } // : 0000988 & 0000991 bool maintanceWeaponSelected = (this->mCurrentWeaponID == AVH_WEAPON_WELDER || this->mCurrentWeaponID == AVH_WEAPON_HEALINGSPRAY); bool hasWelder = this->mHasWelder; bool isGorge = this->GetHUDUser3() == AVH_USER3_ALIEN_PLAYER2; bool isCommander = this->GetInTopDownMode(); //if (isCommander || maintanceWeaponSelected) { if (isCommander || hasWelder || isGorge) { gEngfuncs.pEventAPI->EV_SetUpPlayerPrediction( false, true ); gEngfuncs.pEventAPI->EV_PushPMStates(); gEngfuncs.pEventAPI->EV_SetSolidPlayers(-1); int localPlayerIndex = gEngfuncs.GetLocalPlayer()->index; int currentteam = this->GetHUDTeam(); int maxclients = gEngfuncs.GetMaxClients(); int theNumEnts = pmove->numphysent; physent_t* theEntity = NULL; for (int i = 0; i < theNumEnts; i++) { theEntity = gEngfuncs.pEventAPI->EV_GetPhysent(i); if(theEntity) { if (localPlayerIndex != theEntity->info) { int theEntityIndex = theEntity->info; list::iterator theSelectedIterator = find(theSelectedList.begin(), theSelectedList.end(), theEntityIndex); if (theSelectedIterator == theSelectedList.end()) { bool theIsPlayer = ((theEntityIndex >= 1) && (theEntityIndex <= maxclients)); bool theSameTeam = (theEntity->team == currentteam ); if (isCommander && (theIsPlayer || theSameTeam)) { this->DrawBuildHealthEffectsForEntity(theEntityIndex, 0.2); } //else if (maintanceWeaponSelected && theSameTeam && !theIsPlayer) else if ((maintanceWeaponSelected && theSameTeam && !theIsPlayer) || ((isGorge || hasWelder) && theSameTeam && theIsPlayer)) { if (AvHTraceLineAgainstWorld(gEngfuncs.GetLocalPlayer()->origin, theEntity->origin) == 1.0f) { this->DrawBuildHealthEffectsForEntity(theEntityIndex, 0.3); } } } } } } gEngfuncs.pEventAPI->EV_PopPMStates(); } // : } void AvHHud::Render() { if (!IEngineStudio.IsHardware()) { // Don't show anything in software mode. const char* theMessage = "Software mode is not supported by Natural Selection"; int theWidth; int theHeight; float gammaScale = 1.0f / GetGammaSlope(); gEngfuncs.pfnDrawSetTextColor(0, gammaScale, 0); gEngfuncs.pfnDrawConsoleStringLen(theMessage, &theWidth, &theHeight); gEngfuncs.pfnDrawConsoleString((ScreenWidth() - theWidth) / 2, (ScreenHeight() - theHeight) / 2, (char*)theMessage); return; } if (m_Spectator.IsInOverviewMode()) { AvHSpriteSetViewport(mSpecialViewport[0], mSpecialViewport[1], mSpecialViewport[0] + mSpecialViewport[2], mSpecialViewport[1] + mSpecialViewport[3]); } AvHSpriteBeginFrame(); if (mSteamUIActive) { // Use the old 2D method if we're in the Steam UI. AvHSpriteEnableVGUI(false); } else { AvHSpriteEnableVGUI(true); AvHSpriteSetVGUIOffset(0, 0); } #ifdef DEBUG if ( CVAR_GET_FLOAT( "hud_hideview" ) != 0 ) { // Black out the screen int sprite = SPR_Load(kWhiteSprite); AvHSpriteSetColor(0, 0, 0, 1); AvHSpriteDraw(sprite, 0, 0, 0, ScreenWidth(), ScreenHeight(), 0, 0, 1, 1); AvHSpriteSetColor(1, 1, 1, 1); } #endif // Don't draw the HUD while being digested. if (GetIsBeingDigested()) { if (this->mDigestingSprite) { AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderNormal); DrawWarpedOverlaySprite(mDigestingSprite, 4, 3, .02, .02, .3, .15); } RenderProgressBar(kProgressBarSprite); } else { if (this->GetHUDPlayMode() == PLAYMODE_PLAYING || this->GetHUDPlayMode() == PLAYMODE_READYROOM) { if (!mSteamUIActive) { this->DrawReticleInfo(); this->DrawPlayerNames(); this->DrawToolTips(); this->DrawCenterText(); } } if (this->GetHUDPlayMode() == PLAYMODE_PLAYING) { RenderCommonUI(); if (GetIsMarine()) { if (GetInTopDownMode()) { RenderCommanderUI(); } else { RenderMarineUI(); } } else if (GetIsAlien()) { RenderAlienUI(); } RenderProgressBar(kProgressBarSprite); } if (this->GetHUDPlayMode() >= 1 && this->GetHUDPlayMode() <= 5) { RenderShowSpeed(); } } AvHSpriteEndFrame(); AvHSpriteClearViewport(); } void AvHHud::RenderShowSpeed() { static bool speedMeasured = false; if (!mSteamUIActive) { static int maxSpeed = 0, maxGroundSpeed = 0, maxClimb = 0, maxDive = 0; if (CVAR_GET_FLOAT("cl_showspeed") != 0) { // Draw the speedometer. int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); extern playermove_s* pmove; char buffer[1024]; maxClimb = max(maxClimb, (int)pmove->velocity[2]); maxDive = min(maxDive, (int)pmove->velocity[2]); int speed = (int)Length(pmove->velocity); maxSpeed = max(speed, maxSpeed); sprintf(buffer, "Speed = %d (%d)", speed, maxSpeed/*, maxClimb, maxDive*/); mFont.DrawString(10, 10, buffer, theR, theG, theB); float theGroundSpeed = sqrtf(pmove->velocity[0] * pmove->velocity[0] + pmove->velocity[1] * pmove->velocity[1]); maxGroundSpeed = max(theGroundSpeed, maxGroundSpeed); sprintf(buffer, "Ground speed = %d (%d)", (int)theGroundSpeed, maxGroundSpeed); mFont.DrawString(10, 12 + mFont.GetStringHeight(), buffer, theR, theG, theB); //sprintf(buffer, "vangle0= %f vangle1= %f vangle2= %f)", pmove->angles[0], pmove->angles[1], pmove->angles[2]); //mFont.DrawString(10, 12 + mFont.GetStringHeight() * 2, buffer, theR, theG, theB); speedMeasured = true; } else if (speedMeasured == true) { char msg[256]; sprintf(msg, "Current Speed(%d)\tCurrent Ground Speed(%d) Max Speed(%d)\t Max Ground Speed (%d)\tMax Climb (%d)\tMax Dive(%d)\n", (int)Length(pmove->velocity), (int)sqrtf(pmove->velocity[0] * pmove->velocity[0] + pmove->velocity[1] * pmove->velocity[1]), maxSpeed, maxGroundSpeed, maxClimb, maxDive); ConsolePrint(msg); maxSpeed = 0, maxGroundSpeed = 0, maxClimb = 0, maxDive = 0; speedMeasured = false; } } } void AvHHud::RenderCommonUI() { if (!mSteamUIActive) { DrawInfoLocationText(); DrawHUDStructureNotification(); this->DrawOrders(); this->DrawHelpIcons(); // : 0000971 this->DrawTeammateOrders(); // : 0000992 this->DrawDisplayOrder(); // : if (this->GetIsCombatMode()) { // If we're in combat mode, draw our rank const float kTitleYInset = (kPlayerStatusVerticalInset+5*kPlayerStatusStatusSpacing); string theTitleText = this->GetRankTitle(true); char theCharArray[512]; sprintf(theCharArray, theTitleText.c_str()); // Draw it int theX = mViewport[0] + kPlayerStatusHorizontalCenteredInset*(mViewport[2] - mViewport[0]); int theY = mViewport[1] + mViewport[3] - (1-kTitleYInset)*ScreenHeight(); int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); this->DrawHudStringCentered(theX, theY, ScreenWidth(), theCharArray, theR, theG, theB); } // If we're parasited, draw a message as such if (this->GetIsAlive()) { string theText; char theCharArray[512]; int theR, theG, theB; //UnpackRGB(theR, theG, theB, RGB_YELLOWISH); this->GetPrimaryHudColor(theR, theG, theB, true, false); int theSecondOfLastUpdate = (int)this->mTimeOfLastUpdate; // Draw blinking primal scream message if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_BUFFED) && (theSecondOfLastUpdate % 2)) { if(this->GetIsAlien()) { LocalizeString(kPrimalScreamed, theText); } else if(this->GetIsMarine()) { LocalizeString(kCatalysted, theText); } // Draw it sprintf(theCharArray, "%s", theText.c_str()); this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), kPlayerStatusVerticalInset*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // Draw parasited indicator if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_PARASITED)) { LocalizeString(kParasited, theText); sprintf(theCharArray, "%s", theText.c_str()); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // Draw umbraed indicator if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_UMBRA)) { LocalizeString(kUmbraed, theText); sprintf(theCharArray, "%s", theText.c_str()); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+2*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // Draw blinking "webbed" message if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_ENSNARED) && (theSecondOfLastUpdate % 2)) { LocalizeString(kWebbed, theText); sprintf(theCharArray, "%s", theText.c_str()); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+3*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // Draw "stunned" message (it's so fast, try not blinking it) if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_PLAYER_STUNNED) /*&& (theSecondOfLastUpdate % 2)*/) { LocalizeString(kStunned, theText); sprintf(theCharArray, "%s", theText.c_str()); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+4*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // Draw "being digested" or "digesting" message if(this->GetIsBeingDigested() && (theSecondOfLastUpdate % 2)) { LocalizeString(kDigested, theText); sprintf(theCharArray, "%s", theText.c_str()); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+5*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } else if(this->GetIsDigesting() && (theSecondOfLastUpdate % 2)) { // Look up player in progress entity if((this->mProgressBarEntityIndex >= 1) && (this->mProgressBarEntityIndex <= gEngfuncs.GetMaxClients())) { hud_player_info_t thePlayerInfo; gEngfuncs.pfnGetPlayerInfo(this->mProgressBarEntityIndex, &thePlayerInfo); char* thePlayerName = thePlayerInfo.name; if(thePlayerName) { LocalizeString(kDigestingPlayer, theText); sprintf(theCharArray, theText.c_str(), thePlayerName); } } // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+5*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } // If we are in a squad, draw it on our HUD if(this->mCurrentSquad > 0) { string theSquadIndicator; if(LocalizeString(kSquadIndicator, theSquadIndicator)) { sprintf(theCharArray, theSquadIndicator.c_str(), this->mCurrentSquad); // Draw it this->DrawHudStringCentered(kPlayerStatusHorizontalCenteredInset*ScreenWidth(), (kPlayerStatusVerticalInset+6*kPlayerStatusStatusSpacing)*ScreenHeight(), ScreenWidth(), theCharArray, theR, theG, theB); } } } // Draw the numerical effects. if(this->GetIsAlive()) { for(NumericalInfoEffectListType::iterator theIter = this->mNumericalInfoEffects.begin(); theIter != this->mNumericalInfoEffects.end(); theIter++) { // Draw it Vector theScreenPos; float theWorldPosition[3]; theIter->GetPosition(theWorldPosition); if(AvHCUWorldToScreen(theWorldPosition, (float*)&theScreenPos)) { float theNormX = theScreenPos.x/ScreenWidth(); float theNormY = theScreenPos.y/ScreenHeight(); if(!this->GetInTopDownMode() || !this->GetIsRegionBlockedByUI(theNormX, theNormY)) { float theNumber = theIter->GetNumber(); string thePrependString("+"); int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, true, false); if(theNumber < 0) { // Red, minus for negatives theR = 255; theG = 0; theB = 0; // The "-" is prepended by MakeStringFromInt when negative thePrependString = string(""); } const char* thePostPendStringToTranslate = NULL; string thePostPendString; switch(theIter->GetEventType()) { case kNumericalInfoResourcesEvent: thePostPendStringToTranslate = kNumericalEventResources; break; case kNumericalInfoHealthEvent: thePostPendStringToTranslate = kNumericalEventHealth; break; case kNumericalInfoResourcesDonatedEvent: thePostPendStringToTranslate = kNumericalEventResourcesDonated; break; case kNumericalInfoAmmoEvent: thePostPendStringToTranslate = kNumericalEventAmmo; break; } if(thePostPendStringToTranslate) { LocalizeString(thePostPendStringToTranslate, thePostPendString); } // Don't draw 0, so we can have other messages string theNumberString; if(theNumber != 0) { theNumberString = thePrependString + MakeStringFromFloat(theNumber) + " "; } string theDisplayString = theNumberString + thePostPendString; char theCharBuffer[512]; sprintf(theCharBuffer, "%s", theDisplayString.c_str()); this->DrawHudStringCentered(theScreenPos.x, theScreenPos.y, ScreenWidth(), theCharBuffer, theR, theG, theB); } } } } } // Draw the combat HUD. if (this->GetIsCombatMode()) { // Now draw our current experience level, so people know how close they are to the next level // Load alien resource and energy sprites string theSpriteName = UINameToSprite(kCombatExperienceSprite, ScreenWidth()); AVHHSPRITE theExperienceSprite = SPR_Load(theSpriteName.c_str()); if(theExperienceSprite) { const float kNormalizedWidth = .1f; const float kNormalizedYInset = .96f; const float kNormalizedHeight = .025f; const int kBaseIndex = this->GetIsMarine() ? 2 : 0; // Draw full background const int kXStart = mViewport[0] + (.5f - kNormalizedWidth/2.0f)*(mViewport[2] - mViewport[0]); const int kYStart = mViewport[1] + mViewport[3] - (1 - kNormalizedYInset)*ScreenHeight(); AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(theExperienceSprite, kBaseIndex + 1, kXStart, kYStart, kXStart + kNormalizedWidth*ScreenWidth(), kYStart + kNormalizedHeight*ScreenHeight(), 0, 0, 1, 1); // Draw overlay showing amount of experience float thePercentToNextLevel = AvHPlayerUpgrade::GetPercentToNextLevel(this->GetHUDExperience()); if((thePercentToNextLevel >= 0.0f) && (thePercentToNextLevel <= 1.0f)) { AvHSpriteDraw(theExperienceSprite, kBaseIndex, kXStart, kYStart, kXStart + thePercentToNextLevel*kNormalizedWidth*ScreenWidth(), kYStart + kNormalizedHeight*ScreenHeight(), 0, 0, thePercentToNextLevel, 1.0f); } } if(this->GetIsAlive(false)) { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); this->mCombatUpgradeMenu.SetR(theR); this->mCombatUpgradeMenu.SetG(theG); this->mCombatUpgradeMenu.SetB(theB); this->mCombatUpgradeMenu.Draw(); } } // Draw top down help if(this->GetInTopDownMode()) { this->mTopDownActionButtonHelp.Draw(); } } void AvHHud::RenderProgressBar(char *spriteName) { // Draw the progress bars const float progressBarStayTime = 0.2f; if (this->mProgressBarLastDrawn + progressBarStayTime > this->GetTimeOfLastUpdate()) { AVHHSPRITE currentSprite=0; if ( spriteName && ( strcmp(spriteName, kExperienceBarSprite) == 0 ) ) { currentSprite=this->mExperienceBarSprite; } if ( spriteName && ( strcmp(spriteName, kProgressBarSprite) == 0 ) ) { currentSprite=this->mProgressBarSprite; } if (currentSprite) { const float kNormalizedWidth = .1f; const float kNormalizedYInset = .89f; const float kNormalizedHeight = .025f; // Draw full background const int kXStart = mViewport[0] + (.5f - kNormalizedWidth/2.0f)*(mViewport[2] - mViewport[0]); const int kYStart = mViewport[1] + mViewport[3] - (1 - kNormalizedYInset)*ScreenHeight(); AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(currentSprite, this->mProgressBarDrawframe + 1, kXStart, kYStart, kXStart + kNormalizedWidth*ScreenWidth(), kYStart + kNormalizedHeight*ScreenHeight(), 0, 0, 1, 1); // Draw overlay showing progress float theProgress = this->mProgressBarStatus; if((theProgress >= 0.0f) && (theProgress <= 1.0f)) { AvHSpriteDraw(currentSprite, this->mProgressBarDrawframe, kXStart, kYStart, kXStart + theProgress*kNormalizedWidth*ScreenWidth(), kYStart + kNormalizedHeight*ScreenHeight(), 0, 0, theProgress, 1.0f); } } } } void AvHHud::RenderMiniMap(int inX, int inY, int inWidth, int inHeight) { AvHOverviewMap& theOverviewMap = gHUD.GetOverviewMap(); float hudMinimap=CVAR_GET_FLOAT(kvHudMapZoom); hudMinimap=min(3, max(1, hudMinimap)); float zoomScale=(3.0f-hudMinimap); AvHOverviewMap::DrawInfo theDrawInfo; theDrawInfo.mX = inX; theDrawInfo.mY = inY; theDrawInfo.mWidth = inWidth; theDrawInfo.mHeight = inHeight; theDrawInfo.mFullScreen = false; float worldViewWidth = 800 + 400.0f*zoomScale; theDrawInfo.mZoomScale = 1-(0.25f * zoomScale ); float aspectRatio = (float)(theDrawInfo.mHeight) / theDrawInfo.mWidth; float thePlayerX; float thePlayerY; theOverviewMap.GetWorldPosition(thePlayerX, thePlayerY); theDrawInfo.mViewWorldMinX = thePlayerX - worldViewWidth; theDrawInfo.mViewWorldMinY = thePlayerY - worldViewWidth * aspectRatio; theDrawInfo.mViewWorldMaxX = thePlayerX + worldViewWidth; theDrawInfo.mViewWorldMaxY = thePlayerY + worldViewWidth * aspectRatio; theOverviewMap.Draw(theDrawInfo); } void AvHHud::RenderMarineUI() { int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); if (mMarineTopSprite && (this->mMapMode == MAP_MODE_NS)) { float theX; float theY; float theWidth = .75f*ScreenWidth(); float theHeight = ((float)256/768)*ScreenHeight(); AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteSetColor(1,1,1); int hudMinimap=CVAR_GET_FLOAT(kvHudMapZoom); for ( int i=0; i<3; i++ ) { float width=theWidth/3.0f; int frame=i; if ( hudMinimap == 0 && i == 2 ) frame=i+1; AvHSpriteDraw(mMarineTopSprite, frame, mViewport[2] - width*(3-i) + mViewport[0], mViewport[1], mViewport[2] - width*(2-i) + mViewport[0], mViewport[1] + theHeight, 0, 0, 1, 1); } // Draw the minimap. if ( hudMinimap > 0 ) { int theMiniMapX = mViewport[2] - 0.21f * ScreenWidth() + mViewport[0]; int theMiniMapY = mViewport[1] + 0.013 * ScreenHeight(); int theMiniMapWidth = 0.200 * ScreenWidth(); int theMiniMapHeight = 0.202 * ScreenHeight(); RenderMiniMap(theMiniMapX, theMiniMapY, theMiniMapWidth, theMiniMapHeight); } // Draw the resource label. theHeight = ScreenHeight() * 0.038; std::string theResourceText; char theResourceBuffer[64]; LocalizeString(kMarineResourcePrefix, theResourceText); _snprintf(theResourceBuffer, 64, "%s %d", theResourceText.c_str(), this->mVisualResources); theX = mViewport[0] + 0.3 * ScreenWidth(); theY = mViewport[1] + 0.007 * ScreenHeight() + (theHeight - GetSmallFont().GetStringHeight()) / 2; GetSmallFont().DrawString(theX, theY, theResourceBuffer, theR, theG, theB); // Draw the commander status label. std::string theCommanderName; bool theCommanderNameVisible = true; if (!GetCommanderLabelText(theCommanderName)) { // Blink if there is no commander. if ((int)this->mTimeOfLastUpdate % 2) { theCommanderNameVisible = false; } } if (theCommanderNameVisible) { theX = mViewport[0] + 0.50 * ScreenWidth(); theY = mViewport[1] + 0.006 * ScreenHeight() + (theHeight - GetSmallFont().GetStringHeight()) / 2; GetSmallFont().DrawString(theX, theY, theCommanderName.c_str(), theR, theG, theB); } } AvHSpriteEnableClippingRect(false); // TODO: Draw various marine upgrades (assumes 256x256 sprite, with each upgrade being 64x64, listed right to left, top to bottom, armor upgrades first, // weapon upgrades next, then motion-tracking, then jetpacks, then exoskeleton) // Do we have a jetpack? if(GetHasUpgrade(this->GetHUDUpgrades(), MASK_UPGRADE_7)) { // Draw jetpack energy indicator if(this->mMarineUIJetpackSprite) { float theFactor = 1 - pmove->fuser3/kNormalizationNetworkFactor; int theFrame = 0; string theOutputMessage; if(LocalizeString(kJetpackEnergyHUDText, theOutputMessage)) { float theX = .9f*ScreenWidth(); float theY = .8f*ScreenHeight(); float theWidth = .02f*ScreenWidth(); float theHeight = .1f*ScreenHeight(); AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteSetColor(1,1,1); AvHSpriteDraw(mMarineUIJetpackSprite, 0, theX, theY, theX + theWidth, theY + theHeight * theFactor, 0, 0, 1, theFactor); AvHSpriteDraw(mMarineUIJetpackSprite, 1, theX, theY + theHeight * theFactor, theX + theWidth, theY + theHeight, 0, theFactor, 1, 1); } } } const int kNumUpgradesToDraw = 4; AvHUpgradeMask kUpgradeToMaskTable[kNumUpgradesToDraw] = {MASK_UPGRADE_4, MASK_UPGRADE_1, MASK_UPGRADE_8, MASK_UPGRADE_7}; const float kAspectRatio = ScreenWidth()/ScreenHeight(); const float kNormalizedSpacing = 0.01f; const int kBaseRightOffset = kNormalizedSpacing*ScreenWidth(); const int kVerticalUpgradeSpacing = kNormalizedSpacing*kAspectRatio*ScreenHeight(); int theUpgradeVar = this->GetHUDUpgrades(); const int kUpgradeFrame = 0; const float kUpgradeSize = 0.05; int theUpgradeWidth = kUpgradeSize*ScreenWidth(); int theUpgradeHeight = kUpgradeSize*kAspectRatio*ScreenHeight(); for(int i = 0; i < kNumUpgradesToDraw; i++) { AvHUpgradeMask theUpgradeMask = kUpgradeToMaskTable[i]; // Draw highest ammo upgrade level if(theUpgradeMask == MASK_UPGRADE_4) { if(GetHasUpgrade(theUpgradeVar, MASK_UPGRADE_5)) { theUpgradeMask = MASK_UPGRADE_5; if(GetHasUpgrade(theUpgradeVar, MASK_UPGRADE_6)) { theUpgradeMask = MASK_UPGRADE_6; } } } // Draw highest armor level if(theUpgradeMask == MASK_UPGRADE_1) { if(GetHasUpgrade(theUpgradeVar, MASK_UPGRADE_2)) { theUpgradeMask = MASK_UPGRADE_2; if(GetHasUpgrade(theUpgradeVar, MASK_UPGRADE_3)) { theUpgradeMask = MASK_UPGRADE_3; } } } // Draw heavy armor instead of jetpacks if they have it if(theUpgradeMask == MASK_UPGRADE_7) { if(GetHasUpgrade(theUpgradeVar, MASK_UPGRADE_13)) { theUpgradeMask = MASK_UPGRADE_13; } } if(GetHasUpgrade(theUpgradeVar, theUpgradeMask)) { int theFrame = 0; switch(theUpgradeMask) { case MASK_UPGRADE_4: theFrame = 0; break; case MASK_UPGRADE_5: theFrame = 1; break; case MASK_UPGRADE_6: theFrame = 2; break; case MASK_UPGRADE_1: theFrame = 3; break; case MASK_UPGRADE_2: theFrame = 4; break; case MASK_UPGRADE_3: theFrame = 5; break; case MASK_UPGRADE_8: theFrame = 6; break; case MASK_UPGRADE_7: theFrame = 7; break; case MASK_UPGRADE_13: theFrame = 8; break; } float theStartU = (theFrame % 4)*.25f; float theStartV = (theFrame / 4)*.333f; float theEndU = theStartU + .25f; float theEndV = theStartV + .333f; const int kBaseX = ScreenWidth() - .05*ScreenWidth(); const int kBaseY = .75*ScreenHeight(); const int kIconWidth = .05*ScreenWidth(); const int kIconHeight = .05*ScreenHeight(); float x1 = kBaseX; float y1 = kBaseY + i*kIconHeight; float x2 = x1 + kIconWidth; float y2 = y1 + kIconHeight; AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteEnableClippingRect(false); AvHSpriteSetColor(1, 1, 1); AvHSpriteDraw(mMarineUpgradesSprite, 0, x1, y1, x2, y2, theStartU, theStartV, theEndU, theEndV); } } bool frames[3] = { false, false, false}; if ( this->mHasGrenades ) frames[0]=true; if ( this->mHasMines ) frames[1]=true; if ( this->mHasWelder ) frames[2]=true; for(int i = 0; i < 3; i++) { int theFrame=i+9; if ( frames[i] == true ) { const int kIconWidth = .05*ScreenWidth(); const int kIconHeight = .05*ScreenHeight(); const int kBaseX = ScreenWidth() - .05*ScreenWidth(); const int kBaseY = .75*ScreenHeight(); float theStartU = (theFrame % 4)*.25f; float theStartV = (theFrame / 4)*.333f; float theEndU = theStartU + .25f; float theEndV = theStartV + .333f; float x1 = kBaseX; float y1 = kBaseY - (i+1)*kIconHeight; float x2 = x1 + kIconWidth; float y2 = y1 + kIconHeight; AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteEnableClippingRect(false); AvHSpriteSetColor(1, 1, 1); AvHSpriteDraw(mMarineUpgradesSprite, theFrame, x1, y1, x2, y2, theStartU, theStartV, theEndU, theEndV); } } } void AvHHud::RenderCommanderUI() { RenderStructureRanges(); // Draw command button if(this->mCommandButtonSprite) { // Animate the button unless the mouse is over it (?) int kNumFrames = 16; // Last frame is highlighted, first frame seems off const float kFramesPerSecond = 10; const float kCycleTime = (1.0f/kFramesPerSecond)*kNumFrames; float theNumCycles = this->mTimeOfLastUpdate/kCycleTime; int theFrame = 1 + (theNumCycles - (int)theNumCycles)*kNumFrames; const int kStartX = 890; int theX = (kStartX/1024.0f)*ScreenWidth(); int theY = (525/768.0f)*ScreenHeight(); float theWidth = ((1024 - kStartX)/1024.0f)*ScreenWidth(); float theHeight = (38/768.0f)*ScreenHeight(); float x1 = theX; float y1 = theY; float x2 = x1 + theWidth; float y2 = y1 + theHeight; AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(mCommandButtonSprite, theFrame, x1, y2, x2, y2, 0, 0, 1, 1); } // Draw "select all" button if(this->mSelectAllSprite) { AvHTechImpulsePanel* theTechImpulsePanel = NULL; if(this->GetManager().GetVGUIComponentNamed(kSelectAllImpulsePanel, theTechImpulsePanel)) { if(theTechImpulsePanel->isVisible()) { int theX, theY; theTechImpulsePanel->getPos(theX, theY); int theWidth, theHeight; theTechImpulsePanel->getSize(theWidth, theHeight); // Draw unhighlighted, highlighted, or active int theFrame = 0; // If mouse is over, use frame 1 if(this->GetIsMouseInRegion(theX, theY, theWidth, theHeight)) { theFrame = 1; } // If all troops selected, use frame 2 if((this->mSelected.size()) > 0 && (this->mSelected == this->mSelectAllGroup)) { theFrame = 2; } AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(mSelectAllSprite, theFrame, theX, theY, theX + theWidth, theY + theHeight, 0, 0, 1, 1); } } } // Draw blinking "under attack" indicator if(this->mCommandStatusSprite) { int theFrame = 0; int theX = (725/1024.0f)*ScreenWidth(); int theY = (702/768.0f)*ScreenHeight(); float theWidth = (61.5f/1024.0f)*ScreenWidth(); float theHeight = (66/768.0f)*ScreenHeight(); // Blink button when an alert is set! if(this->mBlinkingAlertType) { int theSecondOfLastUpdate = (int)this->mTimeOfLastUpdate; if(theSecondOfLastUpdate % 2) { theFrame = this->mBlinkingAlertType; } } AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(mCommandStatusSprite, theFrame, theX, theY, theX + theWidth, theY + theHeight, 0, 0, 1, 1); } if(this->mTopDownTopSprite) { float theHeight = ((float)128/768)*ScreenHeight(); AvHSpriteDrawTiles(mTopDownTopSprite, 4, 1, 0, 0, ScreenWidth(), theHeight, 0, 0, 1, 1); } if(this->mTopDownBottomSprite) { // The artwork is three 128s high float theHeight = ((float)256/768)*ScreenHeight(); AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDrawTiles(mTopDownBottomSprite, 4, 1, 0, ScreenHeight()-theHeight, ScreenWidth(), ScreenHeight(), 0, 0, 1, 1); } // Draw scaled sprites for all ActionButtons this->DrawActionButtons(); // Draw hotgroups this->DrawHotgroups(); // Draw pending requests this->DrawPendingRequests(); // Draw logout button if(this->mLogoutSprite) { int theFrame = 0; int theX = .785*ScreenWidth(); int theY = .013*ScreenHeight(); float theWidth = .197f*ScreenWidth(); float theHeight = .083f*ScreenHeight(); // Detect mouseover if(this->GetIsMouseInRegion(theX, theY, theWidth, theHeight)) { theFrame = 1; } AvHSpriteSetRenderMode(kRenderTransAlpha); AvHSpriteDraw(mLogoutSprite, theFrame, theX, theY, theX + theWidth, theY + theHeight, 0, 0, 1, 1); } } void AvHHud::RenderStructureRanges() { float theRangeR = 0; float theRangeG = 1; float theRangeB = 0; float theRangeA = 0.2f; // Draw range for current ghost building, if applicable EntityListType theEntities; IntList theDistanceRequirements; float theZAdjustment = 0.0f; bool theSnapToLocation = false; bool theDrewGhostRegions = false; AvHSHUGetBuildRegions(this->mGhostBuilding, theEntities, theDistanceRequirements, theZAdjustment, theSnapToLocation); // For each entity to draw int theDistanceCounter = 0; for(EntityListType::iterator theRangeIter = theEntities.begin(); theRangeIter != theEntities.end(); theRangeIter++, theDistanceCounter++) { //cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(*theRangeIter); physent_t* theEntity = gEngfuncs.pEventAPI->EV_GetPhysent(*theRangeIter); if(theEntity) { vec3_t thePosition; thePosition = AvHSHUGetRealLocation(theEntity->origin, theEntity->mins, theEntity->maxs); //int theSprite = (theEntity->iuser3 == AVH_USER3_SIEGETURRET) ? this->mSiegeTurretSprite : this->mBuildCircleSprite; AVHHSPRITE theSprite = this->mBuildCircleSprite; int theDistanceRequirement = theDistanceRequirements[theDistanceCounter]; RenderStructureRange(thePosition, theDistanceRequirement, theSprite, kRenderTransAdd, 0, theRangeR, theRangeG, theRangeB, theRangeA); theDrewGhostRegions = true; } } // Draw selection as well for(EntityListType::iterator theSelectedIter = this->mSelected.begin(); theSelectedIter != this->mSelected.end(); theSelectedIter++) { cl_entity_s* theEntity = gEngfuncs.GetEntityByIndex(*theSelectedIter); //physent_t* theEntity = gEngfuncs.pEventAPI->EV_GetPhysent(*theSelectedIter); if(theEntity) { vec3_t thePosition; thePosition = AvHSHUGetRealLocation(theEntity->curstate.origin, theEntity->curstate.mins, theEntity->curstate.maxs); AvHUser3 theUser3 = (AvHUser3)(theEntity->curstate.iuser3); int theRange = AvHSHUGetDrawRangeForUser3(theUser3); if(theRange > 0) { // Don't draw structures that are recycling and therefore inactive if(!GetHasUpgrade(theEntity->curstate.iuser4, MASK_RECYCLING)) { //int theSprite = (theEntity->curstate.iuser3 == AVH_USER3_SIEGETURRET) ? this->mSiegeTurretSprite : this->mBuildCircleSprite; AVHHSPRITE theSprite = this->mBuildCircleSprite; RenderStructureRange(thePosition, theRange, theSprite, kRenderTransAdd, 0, theRangeR, theRangeG, theRangeB, theRangeA); } } } } // Draw the minimum build radius violations. EntityListType theBuildViolations; AvHSHUGetMinBuildRadiusViolations(mGhostBuilding, mGhostWorldLocation, theBuildViolations); for (EntityListType::iterator theBuildViolationsIter = theBuildViolations.begin(); theBuildViolationsIter != theBuildViolations.end(); ++theBuildViolationsIter) { gEngfuncs.pEventAPI->EV_SetUpPlayerPrediction( false, true ); // Store off the old count gEngfuncs.pEventAPI->EV_PushPMStates(); // Now add in all of the players. gEngfuncs.pEventAPI->EV_SetSolidPlayers (-1); physent_t* theEntity = gEngfuncs.pEventAPI->EV_GetPhysent(*theBuildViolationsIter); gEngfuncs.pEventAPI->EV_PopPMStates(); if(theEntity) { vec3_t thePosition; thePosition = AvHSHUGetRealLocation(theEntity->origin, theEntity->mins, theEntity->maxs); AvHUser3 theUser3 = (AvHUser3)(theEntity->iuser3); vec3_t theMinSize, theMaxSize; AvHSHUGetSizeForUser3(theUser3, theMinSize, theMaxSize); float theMaxRadius2 = max(max(theMinSize.x, theMaxSize.x), max(theMinSize.y, theMaxSize.y)); AVHHSPRITE theSprite = this->mBuildCircleSprite; // : 0000291 // It's possible to place "on" marines if you're offset a little from center. This code and // associated changes above and in AvHSharedUtil.cpp is to enforce a build distance around marines, // in the same way as structures, to prevent this exploit. float theMinMarineBuildDistance; if (theUser3 == AVH_USER3_MARINE_PLAYER) { theMinMarineBuildDistance = BALANCE_VAR(kMinMarinePlayerBuildDistance); } else { theMinMarineBuildDistance = BALANCE_VAR(kMinMarineBuildDistance); } // : RenderStructureRange(thePosition, theMinMarineBuildDistance + theMaxRadius2, theSprite, kRenderTransAdd, 0, 1, 0, 0, 0.3f); } } } void AvHHud::RenderStructureRange(vec3_t inOrigin, int inRadius, AVHHSPRITE inSprite, int inRenderMode, int inFrame, float inR, float inG, float inB, float inAlpha) { vec3_t w1; w1[0] = inOrigin[0] - inRadius; w1[1] = inOrigin[1] - inRadius; w1[2] = inOrigin[2]; vec3_t s1; gEngfuncs.pTriAPI->WorldToScreen(w1, s1); vec3_t w2; w2[0] = inOrigin[0] + inRadius; w2[1] = inOrigin[1] + inRadius; w2[2] = inOrigin[2]; vec3_t s2; gEngfuncs.pTriAPI->WorldToScreen(w2, s2); float x1 = XPROJECT(s1[0]); float y1 = YPROJECT(s1[1]); float x2 = XPROJECT(s2[0]); float y2 = YPROJECT(s2[1]); AvHSpriteSetColor(inR, inG, inB, inAlpha); AvHSpriteSetRenderMode(inRenderMode); AvHSpriteDraw(inSprite, inFrame, x1, y1, x2, y2, 0, 0, 1, 1); AvHSpriteSetColor(1, 1, 1, 1); } void AvHHud::RenderAlienUI() { const float kAspectRatio = ScreenWidth()/ScreenHeight(); // Draw the embryo overlay if the player is gestating. if (GetHUDUser3() == AVH_USER3_ALIEN_EMBRYO) { if (mMembraneSprite) { AvHSpriteSetRenderMode(kRenderTransAdd); AvHSpriteSetColor(1,1,1); DrawWarpedOverlaySprite(mMembraneSprite, 4, 3, .007 * 2, .002 * 2, .15, .3); } return; } AvHSpriteSetRenderMode(kRenderTransAlpha); int theWidth = kResourceEnergyBarWidth*ScreenWidth(); int theHeight = kResourceEnergyBarHeight*ScreenHeight(); int theX = mViewport[0]; int theY = mViewport[3] - theHeight + mViewport[1]; // Draw resource text label. if (!mSteamUIActive) { const float kTextInset = kResourceEnergyBarWidth*.5f; const int kNumericYOffset = 1.5*this->GetHudStringHeight(); // : 0000989 // moved resource label a bit down //int theResourceLabelX = mViewport[0] + kTextInset*ScreenWidth(); //int theResourceLabelY = theY - + .05f * ScreenHeight(); int theResourceLabelX = 10; int theResourceLabelY = .68f * ScreenHeight(); // : if(this->mMapMode == MAP_MODE_NS) { string theLocalizedResourceDescription; if(LocalizeString(kAlienResourcePrefix, theLocalizedResourceDescription)) { int theStringWidth = this->GetHudStringWidth(theLocalizedResourceDescription.c_str()); int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); this->DrawHudString(theResourceLabelX, theResourceLabelY, ScreenWidth(), theLocalizedResourceDescription.c_str(), theR, theG, theB); } // Draw amount we have char theBuffer[128]; sprintf(theBuffer, "(%d/%d)", this->mVisualResources, this->GetMaxAlienResources()); this->DrawTranslatedString(theResourceLabelX, theResourceLabelY + kNumericYOffset, theBuffer); } } // Draw energy bar. if (mAlienUIEnergySprite) { theX = mViewport[2] - theWidth + mViewport[0]; float theFactor = 1 - this->mVisualEnergyLevel; AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransTexture); AvHSpriteDraw(mAlienUIEnergySprite, 0, theX, theY, theX + theWidth, theY + theHeight * theFactor, 0, 0, 1, theFactor); AvHSpriteDraw(mAlienUIEnergySprite, 1, theX, theY + theHeight * theFactor, theX + theWidth, theY + theHeight, 0, theFactor, 1, 1); } if (mAlienUICloakSprite ) { cl_entity_s* theLocalPlayer = GetVisiblePlayer(); if(theLocalPlayer ) { theX = mViewport[2] - theWidth + mViewport[0]; int amount=0; int range=255-kAlienSelfCloakingBaseOpacity; if ( theLocalPlayer->curstate.renderamt > 0 ) { amount=theLocalPlayer->curstate.renderamt-kAlienSelfCloakingBaseOpacity; amount=max(0, min(range, amount)); } float theFactor = 1; if ( theLocalPlayer->curstate.rendermode == kRenderTransTexture ) theFactor=(float)amount/(float)range; AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransTexture); AvHSpriteDraw(mAlienUICloakSprite, 0, theX, theY, theX + theWidth, theY + theHeight * theFactor, 0, 0, 1, theFactor); AvHSpriteDraw(mAlienUICloakSprite, 1, theX, theY + theHeight * theFactor, theX + theWidth, theY + theHeight, 0, theFactor, 1, 1); // } // else { // int a=0; // } } } // Draw hive indicators. if (mHiveInfoSprite) // && (this->mMapMode == MAP_MODE_NS)) // removed check to enable hive with health to be shown in combat { int theHiveIndex = 0; for (HiveInfoListType::iterator theIter = this->mHiveInfoList.begin(); theIter != this->mHiveInfoList.end(); theIter++, theHiveIndex++) { // For this hive, figure out which frame to draw int theFrame = theIter->mStatus; // If under attack, make it blink red int theSecondOfLastUpdate = (int)this->mTimeOfLastUpdate; if(theIter->mUnderAttack && (theSecondOfLastUpdate % 2)) { theFrame = kHiveInfoStatusUnderAttack; } int theHiveWidth = kHiveNormScreenWidth*ScreenWidth(); // Draw hive //float theNormX = theHiveWidth; float theNormY = kHiveNormScreenY + kHiveNormScreenVerticalSpacing*theHiveIndex; //DrawScaledHUDSprite(this->mHiveInfoSprite, kRenderTransAlpha, 1, theNormX*ScreenWidth, theNormY*ScreenHeight(), kHiveNormScreenWidth*ScreenWidth, kHiveNormScreenHeight*ScreenHeight(), theFrame); float x1 = mViewport[0] + mViewport[2] - theHiveWidth; float y1 = mViewport[1] + theNormY*ScreenHeight(); float x2 = x1 + theHiveWidth; float y2 = y1 + kHiveNormScreenHeight*ScreenHeight(); AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransTexture); AvHSpriteDraw(mHiveInfoSprite, theFrame, x1, y1, x2, y2, 0, 0, 1, 1); // AvHSpriteSetColor(1,1,1); // AvHSpriteSetRenderMode(kRenderTransTexture); // AvHSpriteDraw(mHiveHealthSprite, 0, x1, y1, x1+100, y1+100, 0, 0, 1, 1); // use the hive sprite height for the hivehealth sprite height and width float theHealthSpriteHeight = (float)(kHiveNormScreenHeight) *ScreenHeight(); float theHealthSpriteWidth = theHealthSpriteHeight; AvHSpriteSetColor(1,1,1); AvHSpriteSetRenderMode(kRenderTransTexture); // adjust the position for the sprite -- to the left float w1 = mViewport[0] + mViewport[2] - (kHiveNormScreenWidth + 0.02f)*ScreenWidth(); AvHSpriteDraw(mHiveHealthSprite, 0, w1, y1, w1 + theHealthSpriteWidth, y1 + theHealthSpriteHeight, 0, 0, 1, 1); if (theIter->mStatus > 0) { if (theIter->mHealthPercentage < 100) { AvHSpriteDraw(mHiveHealthSprite, 0, w1, y1, w1 + theHealthSpriteWidth, y1 + theHealthSpriteHeight, 0, 0, 1, 1); AvHSpriteDraw(mHiveHealthSprite, 1, w1, y1 + theHealthSpriteHeight - theHealthSpriteHeight * ((float)(theIter->mHealthPercentage) * 0.92f / 100), w1 + theHealthSpriteWidth, y1 + theHealthSpriteHeight, 0, 1 - ((float)(theIter->mHealthPercentage) * 0.92f / 100), 1, 1); } else { AvHSpriteDraw(mHiveHealthSprite, 1, w1, y1, w1 + theHealthSpriteWidth, y1 + theHealthSpriteHeight, 0, 0, 1, 1); } } // Draw technology it's supporting AvHMessageID theTechnology = (AvHMessageID)theIter->mTechnology; theFrame = -1; switch(theTechnology) { case ALIEN_BUILD_DEFENSE_CHAMBER: theFrame = 0; break; case ALIEN_BUILD_SENSORY_CHAMBER: theFrame = 3; break; case ALIEN_BUILD_MOVEMENT_CHAMBER: theFrame = 2; break; } // Draw it inset if(theFrame >= 0) { float x1 = mViewport[0] + mViewport[2] - theHiveWidth + (kHiveTechnologyInsetXScalar*kHiveNormScreenWidth)*ScreenWidth(); float y1 = mViewport[1] + (theNormY + kHiveTechnologyInsetYScalar*kHiveNormScreenHeight)*ScreenHeight(); float x2 = x1 + kHiveTechnologyInsetWidthScalar*kHiveNormScreenWidth*ScreenWidth(); float y2 = y1 + kHiveTechnologyInsetHeightScalar*kHiveNormScreenHeight*ScreenHeight(); AvHSpriteDraw(mAlienUIUpgradeCategories, theFrame, x1, y1, x2, y2, 0, 0, 1, 1); } } } // Draw the upgrades. const int kNumberXInset = 2; const int kNumberYInset = 2; const int kBaseYOffset = .1*ScreenHeight(); // Now draw alien upgrades const float kNormalizedSpacing = 0.01f; const int kBaseRightOffset = kNormalizedSpacing*ScreenWidth(); const int kVerticalUpgradeSpacing = kNormalizedSpacing*kAspectRatio*ScreenHeight(); int theUpgradeVar = this->GetHUDUpgrades(); const int kUpgradeFrame = 0; const float kUpgradeSize = 0.04; int theUpgradeWidth = kUpgradeSize*ScreenWidth(); int theUpgradeHeight = kUpgradeSize*kAspectRatio*ScreenHeight(); // In Combat mode, we can have multiple upgrades in each category int theNumDrawnInCategory[ALIEN_UPGRADE_CATEGORY_MAX_PLUS_ONE + 1]; memset(theNumDrawnInCategory, 0, sizeof(int)*(ALIEN_UPGRADE_CATEGORY_MAX_PLUS_ONE + 1)); AvHUpgradeMask theUpgradeMasks[kNumAlienUpgrades] = {MASK_UPGRADE_1, MASK_UPGRADE_2, MASK_UPGRADE_3, MASK_NONE, MASK_NONE, MASK_NONE, MASK_UPGRADE_4, MASK_UPGRADE_5, MASK_UPGRADE_6, MASK_UPGRADE_7, MASK_UPGRADE_8, MASK_UPGRADE_9}; for(int i = 0; i < kNumAlienUpgrades; i++) { AvHUpgradeMask theUpgradeMask = theUpgradeMasks[i]; AvHAlienUpgradeCategory theCategory = ALIEN_UPGRADE_CATEGORY_INVALID; AvHGetAlienUpgradeCategoryFromMask(theUpgradeMask, theCategory); int theCategoryNumber = (int)theCategory; //int theRowNumber = i % (kNumAlienUpgrades/kNumAlienUpgradeCategories); // Quick hack: treat defense as offense for drawing purposes, because offense has been (temporarily?) removed int theCategoryNumberOffset = (int)(((theCategory == ALIEN_UPGRADE_CATEGORY_DEFENSE) ? ALIEN_UPGRADE_CATEGORY_OFFENSE : theCategory)); int theX = ScreenWidth() - theUpgradeWidth - kBaseRightOffset; // Inset drawing of multiple upgrades in same category (needed for Combat mode) theX -= (theNumDrawnInCategory[theCategory]*theUpgradeWidth); int theEnergyHeight = kResourceEnergyBarHeight*ScreenHeight(); int theY = (3*kHiveNormScreenVerticalSpacing)*ScreenHeight() + kVerticalUpgradeSpacing + theCategoryNumberOffset*(kVerticalUpgradeSpacing + theUpgradeHeight); if(GetHasUpgrade(theUpgradeVar, theUpgradeMask)) { theNumDrawnInCategory[theCategory]++; int theLevelOfUpgrade = AvHGetAlienUpgradeLevel(theUpgradeVar, theUpgradeMask); for(int theLevel = theLevelOfUpgrade; theLevel > 0; theLevel--) { // Draw them slightly overlapping const float kOffset = .01f; float x1 = theX - theLevel*(kOffset*ScreenWidth()); float y1 = theY - theLevel*(kOffset*ScreenHeight()); float x2 = x1 + theUpgradeWidth; float y2 = y1 + theUpgradeHeight; AvHSpriteDraw(mAlienUIUpgrades, i, x1, y1, x2, y2, 0, 0, 1, 1); } } else { if(this->GetGameStarted() && !this->GetIsCombatMode()) { // If upgrade is pending, draw it blinking for(int i = 0; i < this->mNumUpgradesAvailable; i++) { if(this->mCurrentUpgradeCategory[i] == theCategory) { int theSecondOfLastUpdate = (int)this->mTimeOfLastUpdate; if(theSecondOfLastUpdate % 2) { int numSprites=1; switch ( theCategory ) { case ALIEN_UPGRADE_CATEGORY_DEFENSE: numSprites=this->mNumDefense; break; case ALIEN_UPGRADE_CATEGORY_SENSORY: numSprites=this->mNumSensory; break; case ALIEN_UPGRADE_CATEGORY_MOVEMENT: numSprites=this->mNumMovement; break; } for ( int j = numSprites; j > 0; j-- ) { const float kOffset = .01f; int theFrame = theCategory-1; float x1 = theX - j*(kOffset*ScreenWidth()); float y1 = theY - j*(kOffset*ScreenHeight()); float x2 = x1 + theUpgradeWidth; float y2 = y1 + theUpgradeHeight; AvHSpriteDraw(mAlienUIUpgradeCategories, theFrame, x1, y1, x2, y2, 0, 0, 1, 1); } } break; } } } } } if(this->GetIsAlien() && (this->GetHUDTeam() != TEAM_IND)) { bool theIsGestating = (this->GetHUDUser3() == AVH_USER3_ALIEN_EMBRYO); bool theIsBeingDigested = this->GetIsBeingDigested(); this->mSelectedNodeResourceCost = -1; // Find the blip nearest our view reticle int theNearestBlip = -1; int theBlip = 0; float theDotProductOfClosestBlip = -1; // Get view vector Vector theViewVector; vec3_t theForward, theRight, theUp; AngleVectors(v_angles, theForward, theRight, theUp); VectorNormalize(theForward); for( theBlip = 0; theBlip < this->mFriendlyBlips.mNumBlips; theBlip++) { // Get vector to current blip Vector theVectorToBlip; VectorSubtract(this->mFriendlyBlips.mBlipPositions[theBlip], gPredictedPlayerOrigin, theVectorToBlip); VectorNormalize(theVectorToBlip); // Dot them float theDotProduct = DotProduct(theForward, theVectorToBlip); // Is dot product closer to 1 then best? if(theDotProduct > theDotProductOfClosestBlip) { // Save new blip and dot product theNearestBlip = theBlip; theDotProductOfClosestBlip = theDotProduct; } } // Draw information about our friendly blips theBlip = theNearestBlip; if(theNearestBlip >= 0) { ASSERT(theNearestBlip < this->mFriendlyBlips.mNumBlips); // Draw location when player under attack int theBlipStatus = this->mFriendlyBlips.mBlipStatus[theBlip]; //if((theBlipStatus == kVulnerableFriendlyBlipStatus) || ((theBlipStatus >= kSayingOne) && (theBlipStatus <= kSayingSix))) //{ // Get location string theLocationName = this->GetNameOfLocation(this->mFriendlyBlips.mBlipPositions[theBlip]); // Get name of entity string theBlipName; int theBlipInfo = this->mFriendlyBlips.mBlipInfo[theBlip]; if(theBlipInfo <= kBaseBlipInfoOffset) { hud_player_info_t thePlayerInfo; gEngfuncs.pfnGetPlayerInfo(theBlipInfo, &thePlayerInfo); if(thePlayerInfo.name) { theBlipName = thePlayerInfo.name; } } else { AvHUser3 theUser3 = AvHUser3(theBlipInfo - kBaseBlipInfoOffset); this->GetTranslatedUser3Name(theUser3, theBlipName); } string theBlipStatusText; char theBlipStatusString[512]; sprintf(theBlipStatusString, "#%s%d", kBlipStatusPrefix, theBlipStatus); LocalizeString(theBlipStatusString, theBlipStatusText); Vector theScreenPos; Vector theMessageWorldPos = this->mFriendlyBlips.mBlipPositions[theBlip]; const float kSpacingScalar = 1.2f; theMessageWorldPos.z -= (kWorldBlipScale/2.0f)*kSpacingScalar; if(AvHCUWorldToScreen(theMessageWorldPos, (float*)&theScreenPos)) { if((theBlipName != "") && (theBlipStatusText != "") && (theLocationName != "") && (CVAR_GET_FLOAT(kvLabelHivesight) == 1)) { // Find alpha for the blip-text based on position on the screen float screenWidth = ScreenWidth(); float screenHeight = ScreenHeight(); float xdiff = fabs(theScreenPos[0] - screenWidth/2); float ydiff = fabs(theScreenPos[1] - screenHeight/2); float quadrance = xdiff * xdiff + ydiff * ydiff; float alpha = max(0.0f, 0.9f - quadrance / (screenHeight * screenHeight)); alpha *= alpha * alpha * alpha; // "MonsieurEvil is under attack" // "Resource tower is under attack" char theFirstLine[512]; sprintf(theFirstLine, "%s %s\n", theBlipName.c_str(), theBlipStatusText.c_str()); this->DrawTranslatedString(theScreenPos[0], theScreenPos[1], theFirstLine, true, true, false, alpha); char theLocationNameCStr[512]; strcpy(theLocationNameCStr, theLocationName.c_str()); this->DrawTranslatedString(theScreenPos[0], theScreenPos[1] + .022f*ScreenHeight(), theLocationNameCStr, true, true, true, alpha); } } //} } // Draw hive locations under each hive indicator if(!theIsGestating && !theIsBeingDigested && (this->mMapMode == MAP_MODE_NS)) { int theHiveIndex = 0; for(HiveInfoListType::iterator theIter = this->mHiveInfoList.begin(); theIter != this->mHiveInfoList.end(); theIter++, theHiveIndex++) { vec3_t theHivePosition(theIter->mPosX, theIter->mPosY, theIter->mPosZ); string theLocationName = this->GetNameOfLocation(theHivePosition); int theHiveWidth = kHiveNormScreenWidth*ScreenWidth(); int theScreenPosX = mViewport[0] + mViewport[2] - theHiveWidth; int theScreenPosY = mViewport[1] + (kHiveNormScreenY + theHiveIndex*kHiveNormScreenVerticalSpacing + kHiveNormScreenHeight)*ScreenHeight(); string theTranslatedLocationName; LocalizeString(theLocationName.c_str(), theTranslatedLocationName); AvHCUTrimExtraneousLocationText(theTranslatedLocationName); int theR, theG, theB; this->GetPrimaryHudColor(theR, theG, theB, false, false); // Draw text in greyscale if hive isn't building or active if(theIter->mStatus == kHiveInfoStatusUnbuilt) { theR = theG = theB = 100; } // If building, draw time until completion. else if (theIter->mStatus > 0 && theIter->mStatus < 6) { const int totalBuildTime = BALANCE_VAR(kHiveBuildTime); if (theIter->mBuildTime > 0 && theIter->mBuildTime < totalBuildTime) { int timeLeft = theIter->mBuildTime; int theMinutesLeft = timeLeft / 60; int theSecondsLeft = timeLeft % 60; if (theMinutesLeft) theTranslatedLocationName = MakeStringFromInt(theMinutesLeft) + "m " + MakeStringFromInt(theSecondsLeft) + "s - " + theTranslatedLocationName; else theTranslatedLocationName = MakeStringFromInt(theSecondsLeft) + "s - " + theTranslatedLocationName; } } this->DrawHudStringReverse(mViewport[0] + mViewport[2] - 5, theScreenPosY, ScreenWidth(), (char*)theTranslatedLocationName.c_str(), theR, theG, theB); } } } } void AvHHud::DrawWarpedOverlaySprite(AVHHSPRITE spriteHandle, int numXFrames, int numYFrames, float inWarpXAmount, float inWarpYAmount, float inWarpXSpeed, float inWarpYSpeed) { float dx = ScreenWidth(); float dy = ScreenHeight(); float theCurrentTime = gHUD.GetTimeOfLastUpdate(); float theNormXAmount = theCurrentTime*inWarpXSpeed - (int)(theCurrentTime*inWarpXSpeed); // Get fractional part of second float theNormYAmount = theCurrentTime*inWarpYSpeed - (int)(theCurrentTime*inWarpYSpeed); float theSinusoidalNormXAmount = cos(theNormXAmount*2.0f*M_PI); float theSinusoidalNormYAmount = sin(theNormYAmount*2.0f*M_PI); float theXAmount = theSinusoidalNormXAmount*inWarpXAmount; float theYAmount = theSinusoidalNormYAmount*inWarpYAmount; for (int frameY = 0; frameY < numYFrames; ++frameY) { for (int frameX = 0; frameX < numXFrames; ++frameX) { float theWarpXStartAmount = 0; float theWarpYStartAmount = 0; float theWarpXEndAmount = 0; float theWarpYEndAmount = 0; int frame = frameX + frameY * numXFrames; float pw = SPR_Width(spriteHandle, frame); float ph = SPR_Height(spriteHandle, frame); float fx1 = ((float)(frameX)) / numXFrames; float fy1 = ((float)(frameY)) / numYFrames; float fx2 = ((float)(frameX + 1)) / numXFrames; float fy2 = ((float)(frameY + 1)) / numYFrames; if(frameX > 0) { theWarpXStartAmount = theXAmount*ScreenWidth(); } if(frameX < numXFrames - 1) { theWarpXEndAmount = theXAmount*ScreenWidth(); } if(frameY > 0) { theWarpYStartAmount = theYAmount*ScreenHeight(); } if(frameY < numYFrames - 1) { theWarpYEndAmount = theYAmount*ScreenHeight(); } AvHSpriteDraw(spriteHandle, frame, dx * fx1 + theWarpXStartAmount, dy * fy1 + theWarpYStartAmount, dx * fx2 + theWarpXEndAmount, dy * fy2 + theWarpYEndAmount, 1 / pw, 1 / ph, 1 - 1 / pw, 1 - 1 / ph); } } } void AvHHud::RenderNoZBuffering() { if (mSteamUIActive) { // Hack since the HUD doesn't get drawn while the Steam UI is open. Render(); } // Don't draw this stuff while being digested if(this->GetIsBeingDigested() || gParticleEditorHandler.GetInEditMode()) { return; } // Temporary, so we can make movies without having the UI flit around (known HL playback bug) if(/*gEngfuncs.pDemoAPI->IsPlayingback() ||*/ gViewPort->IsOptionsMenuVisible()) { return; } const int theDistance = 50; const float kAspectRatio = ScreenWidth()/ScreenHeight(); float m_vRenderOrigin[3]; pVector m_vUp; pVector m_vRight; pVector m_vNormal; //vec3_t theViewAngles; //gEngfuncs.GetViewAngles((float*)theViewAngles) //gEngfuncs.GetView IEngineStudio.GetViewInfo( m_vRenderOrigin, (float*)&m_vUp, (float*)&m_vRight, (float*)&m_vNormal ); //float m_fSoftwareXScale, m_fSoftwareYScale; //IEngineStudio.GetAliasScale( &m_fSoftwareXScale, &m_fSoftwareYScale ); //vec3_t theViewHeight; //gEngfuncs.pEventAPI->EV_LocalPlayerViewheight(theViewHeight); // pVector p; // p.x = m_vRenderOrigin[0] + m_vNormal.x*theDistance; // p.y = m_vRenderOrigin[1] + m_vNormal.y*theDistance; // p.z = m_vRenderOrigin[2] + m_vNormal.z*theDistance; // pVector p; // p.x = m_vRenderOrigin[0]; // p.y = m_vRenderOrigin[1]; // p.z = m_vRenderOrigin[2]; // p.x = v_origin[0]; // p.y = v_origin[1]; // p.z = v_origin[2]; // Allow scripts to perform render AvHScriptManager::Instance()->DrawNoZBuffering(); // if(cl_particleinfo->value) // { // // Draw particles here for debugging // pVector theRealView; // vec3_t angles, up, right, forward; // gEngfuncs.pfnAngleVectors(pmove->angles, forward, right, up); // // theRealView.x = forward[0]; // theRealView.y = forward[1]; // theRealView.z = forward[2]; // // AvHParticleSystemManager::Instance()->Draw(theRealView); // } //if(this->mMappingTechSprite && !this->mIsRenderingSelectionView && !this->mGameStarted) //{ // float theXPercentage = 256/1024.0f; // float theYPercentage = 64/768.0f; // DrawScaledHUDSprite(this->mMappingTechSprite, kRenderTransColor, 1, 0, ScreenHeight/2, theXPercentage*ScreenWidth, theYPercentage*ScreenHeight()); //} // Draw blips, orders, selection and build effects for spectators and demo watchers too int theUpgradeVar = this->GetHUDUpgrades(); bool theHasHiveSightUpgrade = true;//GetHasUpgrade(theUpgradeVar, MASK_ALIEN_UPGRADE_11); if(this->GetIsAlien()) { if(theHasHiveSightUpgrade) { // Draw friendly blips this->mFriendlyBlips.Draw(m_vNormal, kFriendlyBlipStatus); } } // Draw enemy blips this->mEnemyBlips.Draw(m_vNormal, kEnemyBlipStatus); //this->DrawOrders(true); this->DrawSelectionAndBuildEffects(); //this->DrawHelpIcons(); } void AvHHud::InitHUDData( void ) { this->ResetGame(true); } void AvHHud::VidInit(void) { UIHud::VidInit(); mOverviewMap.VidInit(); int theScreenWidth = ScreenWidth(); string theSpriteName; // theSpriteName = UINameToSprite(kEggSprite, theScreenWidth); // this->mAlienUIEggSprite = SPR_Load(theSpriteName.c_str()); // theSpriteName = UINameToSprite(kHiveSprite, theScreenWidth); // this->mAlienUIHiveSprite = SPR_Load(theSpriteName.c_str()); int i = 0; // for(i = 0; i < kNumAlienLifeforms; i++) // { // char theBaseName[128]; // sprintf(theBaseName, "%s%d", kLifeformSprite, i+1); // string theSpriteName = "sprites/level1_hud.spr";//UINameToSprite(theBaseName, theScreenWidth, true); // this->mAlienUILifeforms[i] = SPR_Load(theSpriteName.c_str()); // } if (CVAR_GET_FLOAT("hud_style") == 2.0f) { this->mAlienUIUpgrades = SPR_Load(kAlienUpgradeSpriteNL); this->mAlienUIEnergySprite = SPR_Load(kAlienEnergySpriteNL); this->mAlienUICloakSprite = SPR_Load(kAlienCloakSpriteNL); this->mBackgroundSprite = SPR_Load(kTopDownBGSpriteNL); this->mTopDownTopSprite = SPR_Load(kTopDownTopHUDSpriteNL); this->mTopDownBottomSprite = SPR_Load(kTopDownBottomHUDSpriteNL); this->mMarineTopSprite = SPR_Load(kMarineTopHUDSpriteNL); this->mLogoutSprite = SPR_Load(kLogoutSpriteNL); this->mCommandButtonSprite = SPR_Load(kCommandButtonSpriteNL); this->mCommandStatusSprite = SPR_Load(kCommandStatusSpriteNL); this->mSelectAllSprite = SPR_Load(kSelectAllSpriteNL); this->mOrderSprite = SPR_Load(kOrdersSpriteNL); this->mHiveInfoSprite = SPR_Load(kHiveInfoSpriteNL); this->mHiveHealthSprite = SPR_Load(kHiveHealthSpriteNL); this->mMarineOrderIndicator = SPR_Load(kMarineOrderSpriteNL); this->mMarineUpgradesSprite = SPR_Load(kMarineUpgradesSpriteNL); } else if (CVAR_GET_FLOAT("hud_style") == 1.0f) { char theBaseName[128]; sprintf(theBaseName, "%s", kAlienUpgradeSprite); theSpriteName = UINameToSprite(theBaseName, theScreenWidth); this->mAlienUIUpgrades = SPR_Load(theSpriteName.c_str()); // Load alien energy sprite theSpriteName = UINameToSprite(kAlienEnergySprite, theScreenWidth); this->mAlienUIEnergySprite = SPR_Load(theSpriteName.c_str()); theSpriteName = UINameToSprite(kAlienCloakSprite, theScreenWidth); this->mAlienUICloakSprite = SPR_Load(theSpriteName.c_str()); this->mBackgroundSprite = SPR_Load(kTopDownBGSprite); this->mTopDownTopSprite = SPR_Load(kTopDownTopHUDSpriteMin); this->mTopDownBottomSprite = SPR_Load(kTopDownBottomHUDSpriteMin); this->mMarineTopSprite = SPR_Load(kMarineTopHUDSpriteMin); this->mLogoutSprite = SPR_Load(kLogoutSprite); this->mCommandButtonSprite = SPR_Load(kCommandButtonSprite); this->mCommandStatusSprite = SPR_Load(kCommandStatusSpriteMin); this->mSelectAllSprite = SPR_Load(kSelectAllSpriteMin); // Load order sprite theSpriteName = UINameToSprite(kOrdersSprite, theScreenWidth); this->mOrderSprite = SPR_Load(theSpriteName.c_str()); this->mHiveInfoSprite = SPR_Load(kHiveInfoSprite); this->mHiveHealthSprite = SPR_Load(kHiveHealthSprite); this->mMarineOrderIndicator = SPR_Load(kMarineOrderSprite); this->mMarineUpgradesSprite = SPR_Load(kMarineUpgradesSprite); } else { char theBaseName[128]; sprintf(theBaseName, "%s", kAlienUpgradeSprite); theSpriteName = UINameToSprite(theBaseName, theScreenWidth); this->mAlienUIUpgrades = SPR_Load(theSpriteName.c_str()); // Load alien energy sprite theSpriteName = UINameToSprite(kAlienEnergySprite, theScreenWidth); this->mAlienUIEnergySprite = SPR_Load(theSpriteName.c_str()); theSpriteName = UINameToSprite(kAlienCloakSprite, theScreenWidth); this->mAlienUICloakSprite = SPR_Load(theSpriteName.c_str()); // Load background for topdown mode this->mBackgroundSprite = SPR_Load(kTopDownBGSprite); // Load HUD this->mTopDownTopSprite = SPR_Load(kTopDownTopHUDSprite); this->mTopDownBottomSprite = SPR_Load(kTopDownBottomHUDSprite); this->mMarineTopSprite = SPR_Load(kMarineTopHUDSprite); this->mLogoutSprite = SPR_Load(kLogoutSprite); this->mCommandButtonSprite = SPR_Load(kCommandButtonSprite); this->mCommandStatusSprite = SPR_Load(kCommandStatusSprite); this->mSelectAllSprite = SPR_Load(kSelectAllSprite); // Load order sprite theSpriteName = UINameToSprite(kOrdersSprite, theScreenWidth); this->mOrderSprite = SPR_Load(theSpriteName.c_str()); this->mHiveInfoSprite = SPR_Load(kHiveInfoSprite); this->mHiveHealthSprite = SPR_Load(kHiveHealthSprite); this->mMarineOrderIndicator = SPR_Load(kMarineOrderSprite); this->mMarineUpgradesSprite = SPR_Load(kMarineUpgradesSprite); } char theBaseName[128]; sprintf(theBaseName, "%s", kAlienUpgradeCategory); theSpriteName = UINameToSprite(theBaseName, theScreenWidth); this->mAlienUIUpgradeCategories = SPR_Load(theSpriteName.c_str()); // Load jetpack sprite theSpriteName = UINameToSprite(kJetpackSprite, theScreenWidth); this->mMarineUIJetpackSprite = SPR_Load(theSpriteName.c_str()); //this->mTopDownBottomSprite = SPR_Load("sprites/distorttest.spr"); //this->mTopDownBottomSprite = SPR_Load("sprites/ns.spr"); //this->mTopDownBottomSprite = SPR_Load("sprites/distorttest.spr"); // Load overlays this->mMembraneSprite = SPR_Load(kMembraneSprite); this->mDigestingSprite = SPR_Load(kDigestingSprite); // Load cursor sprite this->mMarineCursor = SPR_Load(kCursorsSprite); this->mAlienCursor = SPR_Load(kAlienCursorSprite); //this->mMappingTechSprite = SPR_Load("sprites/ns.spr"); this->mAlienBuildSprite = SPR_Load(kAlienBuildSprite); this->mMarineBuildSprite = SPR_Load(kMarineBuildSprite); this->mAlienHealthSprite = SPR_Load(kAlienHealthSprite); this->mMarineHealthSprite = SPR_Load(kMarineHealthSprite); this->mBuildCircleSprite = SPR_Load(kBuildCircleSprite); //this->mSiegeTurretSprite = SPR_Load(kSiegeTurretSprite); this->mActionButtonSprites.clear(); //this->mHelpSprites.clear(); string theIconName = string(kHelpIconPrefix) + ".spr"; this->mHelpSprite = SPR_Load(theIconName.c_str()); // : 0000971 this->mTeammateOrderSprite = SPR_Load(kTeammateOrderSprite); // : this->mExperienceBarSprite = SPR_Load(kExperienceBarSprite); this->mProgressBarSprite = SPR_Load(kProgressBarSprite); this->mEnemyBlips.VidInit(); this->mFriendlyBlips.VidInit(); }