Adding an experience system

From Valve Developer Community
Jump to navigation Jump to search

This tutorial will cover adding a basic experience (XP) system to your mod. It will assume you are modding hl2mp, but can be easily modified to work with the "from scratch" or singleplayer sdk, by simply changing the player files used.

Basic setup

Server-side variables

Without any ado, lets begin by adding two variables to the player class, one for their current XP, and one for their current level. Open hl2mp_player.h, and add these two lines to the private section of the CHL2MPPlayer class, around line 155:

CNetworkVar(int, m_iExp);
CNetworkVar(int, m_iLevel);

We will now need some functions to control and use these variables. In a public section of the same class (same file, perhaps around line 150), add the following:

int GetXP() { return m_iExp; }
void AddXP(int add=1) { m_iExp += add; CheckLevel(); }

int GetLevel() { return m_iLevel; }
void CheckLevel();
void LevelUp();

void ResetXP() { m_iExp = 0; m_iLevel = 1; LevelUp(); } // calling LevelUp will reset max health, etc

We will need to set default values, so open hl2mp_player.cpp, and look for the constructor:

CHL2MP_Player::CHL2MP_Player() : m_PlayerAnimState( this )

In this function, add:

m_iExp = 0;
m_iLevel = 1;
LevelUp(); // sets default values

We will create bodies for CheckLevel and LevelUp shortly. It seems likely that XP may affect things related to prediction, such as player movement speed, so we will need to be able to check the player's level on the client.

Client-side variables

Open up chl2mp_player.h, as we're going to put the variables & accessors in there also. In a private section:

int m_iExp, m_iLevel;

And in a public section:

int GetXP() { return m_iExp; }
int GetLevel() { return m_iLevel; } 

We don't need to be able to change these variables from the client, as they will be updated by the server.

Linking client & server

In order to have the client m_iExp and m_iLevel keep the same values as they have on the server, we will need to add them to the player's network table. Near the top of hl2mp_player.cpp, you should find the send table:

IMPLEMENT_SERVERCLASS_ST(CHL2MP_Player, DT_HL2MP_Player)
	SendPropBlahBlah( ... )
	SendPropYaddaYadda( ... )
END_SEND_TABLE()

At the top of this send table (the line immediately after IMPLEMENT_SERVERCLASS_ST), add our own variables like so:

SendPropInt( SENDINFO( m_iExp ) ),
SendPropInt( SENDINFO( m_iLevel ) ),

Now we need to modify the client's receive table to read these.

Near the top of c_hl2mp_player.cpp, you'll find the matching receive table:

IMPLEMENT_CLIENTCLASS_DT(C_HL2MP_Player, DT_HL2MP_Player, CHL2MP_Player)

At the top of this, receive our variables like so:

RecvPropInt( RECVINFO( m_iExp ) ),
RecvPropInt( RECVINFO( m_iLevel ) ),

And thats them fully networked! Straightforward enough, right?

Leveling up

That won't yet fully compile, because we still have to add function bodies for CheckLevel and LevelUp. In order to do so, we will need a way to decide when a player is ready to level up. I'm going to assume only 5 levels for now, you can of course add many more, and adjust the xp requirements as you see fit. I'm also going to assume that the player starts out as level 1, and not 0!

Level XP limits

Somewhere in hl2mp_player.cpp, add the CheckLevel function, and be sure to add all these defines in front of it

#define XP_FOR_LEVEL_2	5
#define XP_FOR_LEVEL_3	12
#define XP_FOR_LEVEL_4	22
#define XP_FOR_LEVEL_5	35

// We have gained XP; decide if we should level up, and do it if needed
// Note: the XP limits here could be in an array, but that would be less clear
void CHL2MP_Player::CheckLevel()
{
	bool bShouldLevel = false;
	if ( GetLevel() == 1 )
	{
		if ( GetXP() >= XP_FOR_LEVEL_2 )
			bShouldLevel = true;
	}
	else if ( GetLevel() == 2 )
	{
		if ( GetXP() >= XP_FOR_LEVEL_3 )
			bShouldLevel = true;
	}
	else if ( GetLevel() == 3 )
	{
		if ( GetXP() >= XP_FOR_LEVEL_4 )
			bShouldLevel = true;
	}
	else if ( GetLevel() == 4 )
	{
		if ( GetXP() >= XP_FOR_LEVEL_5 )
			bShouldLevel = true;
	}

	if ( bShouldLevel )
	{
		m_iLevel ++; // actually increment player's level
		LevelUp(); // and then adjust their settings (speed, health, damage) to reflect the change
	}
}

As we currently have no level-related features (different health, speed, etc), LevelUp will be empty for now. Add it underneath CheckLevel

void CHL2MP_Player::LevelUp()
{
	
}

Giving XP for kills

This should be quite straightforward. Goto hl2mp_gamerules.cpp, and find the PlayerKilled function. Add a little bit onto the end: BaseClass::PlayerKilled( pVictim, info );

CBasePlayer *pScorer = GetDeathScorer( pKiller, pInflictor ); if ( pScorer ) pScorer->AddXP(1);

  1. endif

}


Showing a player's level

It would be a nice feature to show a player's level when you move the mouse over them, so that instead of saying "Enemy: Winston" it will say "Enemy: Winston (level 2)"

Note: This section untested

Open yourmod_english.txt from your mod's resource folder, and replace the following three lines:

"Playerid_sameteam"		"Friend: %s1 Health: %s2"
"Playerid_diffteam"		"Enemy: %s1"
"Playerid_noteam"		"%s1 Health:%s2"

With

"Playerid_sameteam"		"Friend: %s1 Health: %s2 (level %s3)"
"Playerid_diffteam"		"Enemy: %s1 (level %s2)"
"Playerid_noteam"		"%s1 Health:%s2 (level %s3)"

Now go to hl2mp_hud_target_id.cpp, which is the file that controls drawing other player's names when you look at them. Add at line 148: (new line in bold)

wchar_t wszPlayerName[ MAX_PLAYER_NAME_LENGTH ];
wchar_t wszHealthText[ 10 ];
wchar_t wszLevelText[ 10 ];
bool bShowHealth = false;
bool bShowPlayerName = false;

And then at line 177

	if ( bShowHealth )
	{
		_snwprintf( wszHealthText, ARRAYSIZE(wszHealthText) - 1, L"%.0f%%",  ((float)pPlayer->GetHealth() / (float)pPlayer->GetMaxHealth() ) );
		wszHealthText[ ARRAYSIZE(wszHealthText)-1 ] = '\0';
	}
	_snwprintf( wszLevelText, ARRAYSIZE(wszLevelText) - 1, L"%i",  pPlayer->GetLevel() );
	wszLevelText[ ARRAYSIZE(wszLevelText)-1 ] = '\0';
}

Now, from line 182, replace the entire if ( printFormatString ) statement with:

if ( printFormatString )
{
	if ( bShowPlayerName && bShowHealth )
	{
		vgui::localize()->ConstructString( sIDString, sizeof(sIDString),
			vgui::localize()->Find(printFormatString), 3, wszPlayerName, wszHealthText, wszLevelText );
	}
	else if ( bShowPlayerName )
	{
		vgui::localize()->ConstructString( sIDString, sizeof(sIDString),
			vgui::localize()->Find(printFormatString), 2, wszPlayerName, wszLevelText );
	}
	else if ( bShowHealth )
	{
		vgui::localize()->ConstructString( sIDString, sizeof(sIDString),
			vgui::localize()->Find(printFormatString), 2, wszHealthText, wszLevelText );
	}
	else
	{
		vgui::localize()->ConstructString( sIDString, sizeof(sIDString),
			vgui::localize()->Find(printFormatString), 1, wszLevelText );
	}
}

Experience effects

Now, at last, the interesting part. Different things for different levels! Use these effects as a basis for developing your own, unique, level differences for your mod.

Health increase

Put the following into your LevelUp function (in hl2mp_player.cpp):

	switch ( GetLevel() )
	{
	case 1:
		SetHealthMax(100);
		break;
	case 2:
		SetHealthMax(110);
		break;
	case 3:
		SetHealthMax(120);
		break;
	case 4:
		SetHealthMax(130);
		break;
	case 5:
		SetHealthMax(140);
		break;
	}

We will need to declare the SetHealthMax function (somewhere public in hl2mp_player.h)

void SetHealthMax(int h) { m_iHealthMax = h; }

This variable, m_iHealthMax, should be defined in a private section in this file, like so:

int m_iHealthMax;

You needn't worry about setting a default value, as the constructor's call to LevelUp will take care of that.

Speed increase

We'll do this one a bit differently. The player's maximum allowed speed is changed all the time when they press or release the sprint button, so adding a new variable would be fiddly. Instead, we'll just adjust the SetMaxSpeed function. In both hl2mp_player.h and c_hl2mp_player.h (this is needed for prediction), declare:

virtual void	SetMaxSpeed( float flMaxSpeed );

This lets us override the default function. Now open up hl2mp_player_shared.cpp, and put the function there. By putting it in this file, the same code will be executed on the client and the server, ensuring that we don't have any prediction problems.

void CHL2MP_Player::SetMaxSpeed( float flMaxSpeed )
{
	BaseClass::SetMaxSpeed( flMaxSpeed + 10.0f*(GetLevel()-1.0f) );
}

That should add on 10 units per second onto the players movement speed, for each level they have (beyond level 1)

Conclusion

This tutorial has covered the implementation of a very basic experience system. The user is encouraged to develop their own extensions, to affect things like maximum stamina, weapon damage, and armor. Feel free to add any such extensions onto this tutorial!