Adding an experience system
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()
{
	
}
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:
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.
virtual void CHL2MP_Player::SetMaxSpeed( float flMaxSpeed )
{
	BaseClass::SetMaxSpeed( flMaxSpeed + 10.0f*(GetLevel()-1) );
}
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!