Magazine style reloads: Difference between revisions
| m (It's not a clip, it's a magazine. They are NOT the same thing and naming this article could cause confusion regarding weapons with TRUE clips (like the M1 Garand)) |  (Did the same thing in the simplier way noted on the page. Even more of a "hack" now, but works better.) | ||
| Line 1: | Line 1: | ||
| This will be a short and simple one! | This will be a short and simple one! | ||
| Say you want to make each shot and each reload count? In CS you can reload anytime you feel like. Some bullets will be deducted and that is all. However, in real life, you will not start to load the half-empty mag, you pull out a new one, and throw away the used one. OK you don't throw it away but anyway. This system can be found in BF2, and some other games. | Say you want to make each shot and each reload count? In CS you can reload anytime you feel like. Some bullets will be deducted and that is all. However, in real life, you will not start to load the half-empty mag, you pull out a new one, and throw away the used one. OK you don't throw it away but anyway. This system can be found in BF2, and some other games. | ||
| == Creating a Boolean Variable == | |||
| So now to the code; look up the '''basecombatweapon_shared.h''' and search for "m_bFiresUnderwater" (line 115) to find this code: | |||
|  CNetworkVar( int, m_iClip2 );	// number of shots left in the secondary weapon clip, -1 it not used | |||
|  bool	m_bFiresUnderwater;	// true if this weapon can fire underwater | |||
| In-between these two lines you create a variable: | |||
|  CNetworkVar( int, m_iClip2 );		// number of shots left in the secondary weapon clip, -1 it not used | |||
|  '''bool	m_bMagazineStyleReloads;	// true if this weapon reloads by removing magazines (remaining bullets)''' | |||
|  bool	m_bFiresUnderwater;		// true if this weapon can fire underwater | |||
| Now go to '''basecombatweapon_shared.cpp''' and search to "m_bFiresUnderwater" (lines 2253 and 2300): | |||
|  DEFINE_FIELD( m_bFiresUnderwater, FIELD_BOOLEAN ), | |||
| Once again you need to define it like so: | |||
|  '''DEFINE_FIELD( m_bMagazineStyleReloads, FIELD_BOOLEAN ),''' | |||
|  DEFINE_FIELD( m_bFiresUnderwater, FIELD_BOOLEAN ), | |||
| Now that you have the variable set up, look into the CBaseCombatWeapon constructor (line 54) where: | |||
|  m_bReloadsSingly	= false; | |||
| And depending on what you want, you either default magazine style reloads to be on or off: | |||
|  m_bReloadsSingly		= false; | |||
|  '''m_bMagazineStyleReloads	= false;''' | |||
| == The Actual Code == | |||
| Now here's the main part of the code. Look at the ''FinishReload'' procedure (line 1986) and you will find: | |||
|   if ( UsesClipsForAmmo1() ) |   if ( UsesClipsForAmmo1() ) | ||
|   { |   { | ||
| Line 11: | Line 39: | ||
|   } |   } | ||
| to look like this: | Change the last two lines to look like this: | ||
|   if ( UsesClipsForAmmo1() ) |   if ( UsesClipsForAmmo1() ) | ||
|   { |   { | ||
|   	int primary = min( GetMaxClip1() - m_iClip1, pOwner->GetAmmoCount(m_iPrimaryAmmoType));	 | |||
|   	'''m_iClip1 = GetMaxClip1();''' |   	'''m_iClip1 = m_bMagazineStyleReloads ? GetMaxClip1() : primary;''' | ||
|   	'''pOwner->RemoveAmmo(  |   	'''pOwner->RemoveAmmo( m_bMagazineStyleReloads ? GetMaxClip1() : primary, m_iPrimaryAmmoType);''' | ||
|   } |   } | ||
| All that has happened is that if the flag is turned on: you make the clip full again, and remove a whole clip out from the total ammo count.   | |||
| To display the amount of magazines you just have to divide the total amount of ammo by a full clip size and round up. | |||
| ''Note: this technique "hacks" a magazine style reload instead of creating a native implementation, thus the player can still receive individual bullets.'' | |||
| ''So you will still receive a full clip even if the amount of your total ammo on reload is less than that... that's why always round up.'' | |||
| == Covering Your Ass == | |||
| Ever noticed, that the game auto-reloads your gun if you holster it? This is not good, especially the way this code is implemented; to remove this go to '''weapon_hl2mpbasehlmpcombatweapon.cpp''' or '''basehlcombatweapon_shared.cpp''' and find the ''ItemHolsterFrame'' procedure. Comment out the  | |||
|  		FinishReload(); | |||
| == An Example == | |||
| Now an example to turn on magazine style reloads for any weapon: | |||
|  CWeaponPistol::CWeaponPistol( void ) | |||
|  { | |||
|  	m_flSoonestPrimaryAttack = gpGlobals->curtime; | |||
|  	m_flAccuracyPenalty = 0.0f; | |||
|  	m_fMinRange1		= 24; | |||
|  	m_fMaxRange1		= 1500; | |||
|  	m_fMinRange2		= 24; | |||
|  	m_fMaxRange2		= 200; | |||
|  	'''m_bMagazineReloads	= true; // Magazine style reloads''' | |||
|  	m_bFiresUnderwater	= true; | |||
|  } | |||
| [[Category:Programming]] | [[Category:Programming]] | ||
| [[Category:Tutorials]] | [[Category:Tutorials]] | ||
Revision as of 22:03, 7 March 2009
This will be a short and simple one! Say you want to make each shot and each reload count? In CS you can reload anytime you feel like. Some bullets will be deducted and that is all. However, in real life, you will not start to load the half-empty mag, you pull out a new one, and throw away the used one. OK you don't throw it away but anyway. This system can be found in BF2, and some other games.
Creating a Boolean Variable
So now to the code; look up the basecombatweapon_shared.h and search for "m_bFiresUnderwater" (line 115) to find this code:
CNetworkVar( int, m_iClip2 ); // number of shots left in the secondary weapon clip, -1 it not used bool m_bFiresUnderwater; // true if this weapon can fire underwater
In-between these two lines you create a variable:
CNetworkVar( int, m_iClip2 ); // number of shots left in the secondary weapon clip, -1 it not used bool m_bMagazineStyleReloads; // true if this weapon reloads by removing magazines (remaining bullets) bool m_bFiresUnderwater; // true if this weapon can fire underwater
Now go to basecombatweapon_shared.cpp and search to "m_bFiresUnderwater" (lines 2253 and 2300):
DEFINE_FIELD( m_bFiresUnderwater, FIELD_BOOLEAN ),
Once again you need to define it like so:
DEFINE_FIELD( m_bMagazineStyleReloads, FIELD_BOOLEAN ), DEFINE_FIELD( m_bFiresUnderwater, FIELD_BOOLEAN ),
Now that you have the variable set up, look into the CBaseCombatWeapon constructor (line 54) where:
m_bReloadsSingly = false;
And depending on what you want, you either default magazine style reloads to be on or off:
m_bReloadsSingly = false; m_bMagazineStyleReloads = false;
The Actual Code
Now here's the main part of the code. Look at the FinishReload procedure (line 1986) and you will find:
if ( UsesClipsForAmmo1() )
{
	int primary = min( GetMaxClip1() - m_iClip1, pOwner->GetAmmoCount(m_iPrimaryAmmoType));	
	m_iClip1 += primary;
	pOwner->RemoveAmmo( primary, m_iPrimaryAmmoType);
}
Change the last two lines to look like this:
if ( UsesClipsForAmmo1() )
{
	int primary = min( GetMaxClip1() - m_iClip1, pOwner->GetAmmoCount(m_iPrimaryAmmoType));	
	m_iClip1 = m_bMagazineStyleReloads ? GetMaxClip1() : primary;
	pOwner->RemoveAmmo( m_bMagazineStyleReloads ? GetMaxClip1() : primary, m_iPrimaryAmmoType);
}
All that has happened is that if the flag is turned on: you make the clip full again, and remove a whole clip out from the total ammo count. To display the amount of magazines you just have to divide the total amount of ammo by a full clip size and round up.
Note: this technique "hacks" a magazine style reload instead of creating a native implementation, thus the player can still receive individual bullets. So you will still receive a full clip even if the amount of your total ammo on reload is less than that... that's why always round up.
Covering Your Ass
Ever noticed, that the game auto-reloads your gun if you holster it? This is not good, especially the way this code is implemented; to remove this go to weapon_hl2mpbasehlmpcombatweapon.cpp or basehlcombatweapon_shared.cpp and find the ItemHolsterFrame procedure. Comment out the
FinishReload();
An Example
Now an example to turn on magazine style reloads for any weapon:
CWeaponPistol::CWeaponPistol( void )
{
	m_flSoonestPrimaryAttack = gpGlobals->curtime;
	m_flAccuracyPenalty = 0.0f;
	m_fMinRange1		= 24;
	m_fMaxRange1		= 1500;
	m_fMinRange2		= 24;
	m_fMaxRange2		= 200;
	m_bMagazineReloads	= true; // Magazine style reloads
	m_bFiresUnderwater	= true;
}