Team Fortress 2 Source Code as on 22/4/2020
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

256 lines
8.0 KiB

  1. //========= Copyright Valve Corporation, All rights reserved. ============//
  2. //
  3. // Purpose:
  4. //
  5. // $NoKeywords: $
  6. //
  7. //=============================================================================//
  8. #include "cbase.h"
  9. #include "eventlist.h"
  10. #include "stringregistry.h"
  11. // memdbgon must be the last include file in a .cpp file!!!
  12. #include "tier0/memdbgon.h"
  13. // NOTE: If CStringRegistry allowed storing arbitrary data, we could just use that.
  14. // in this case we have the "isPrivate" member and the replacement rules
  15. // (eventIndex can be reused by private activities), so a custom table is necessary
  16. struct eventlist_t
  17. {
  18. int eventIndex;
  19. int iType;
  20. unsigned short stringKey;
  21. short isPrivate;
  22. };
  23. CUtlVector<eventlist_t> g_EventList;
  24. // This stores the actual event names. Also, the string ID in the registry is simply an index
  25. // into the g_EventList array.
  26. CStringRegistry g_EventStrings;
  27. // this is just here to accelerate adds
  28. static int g_HighestEvent = 0;
  29. int g_nEventListVersion = 1;
  30. void EventList_Init( void )
  31. {
  32. g_HighestEvent = 0;
  33. }
  34. void EventList_Free( void )
  35. {
  36. g_EventStrings.ClearStrings();
  37. g_EventList.Purge();
  38. // So studiohdrs can reindex event indices
  39. ++g_nEventListVersion;
  40. }
  41. // add a new event to the database
  42. eventlist_t *EventList_AddEventEntry( const char *pName, int iEventIndex, bool isPrivate, int iType )
  43. {
  44. MEM_ALLOC_CREDIT();
  45. int index = g_EventList.AddToTail();
  46. eventlist_t *pList = &g_EventList[index];
  47. pList->eventIndex = iEventIndex;
  48. pList->stringKey = g_EventStrings.AddString( pName, index );
  49. pList->isPrivate = isPrivate;
  50. pList->iType = iType;
  51. // UNDONE: This implies that ALL shared activities are added before ANY custom activities
  52. // UNDONE: Segment these instead? It's a 32-bit int, how many activities do we need?
  53. if ( iEventIndex > g_HighestEvent )
  54. {
  55. g_HighestEvent = iEventIndex;
  56. }
  57. return pList;
  58. }
  59. // get the database entry from a string
  60. static eventlist_t *ListFromString( const char *pString )
  61. {
  62. // just use the string registry to do this search/map
  63. int stringID = g_EventStrings.GetStringID( pString );
  64. if ( stringID < 0 )
  65. return NULL;
  66. return &g_EventList[stringID];
  67. }
  68. // Get the database entry for an index
  69. static eventlist_t *ListFromEvent( int eventIndex )
  70. {
  71. // ugly linear search
  72. for ( int i = 0; i < g_EventList.Size(); i++ )
  73. {
  74. if ( g_EventList[i].eventIndex == eventIndex )
  75. {
  76. return &g_EventList[i];
  77. }
  78. }
  79. return NULL;
  80. }
  81. int EventList_GetEventType( int eventIndex )
  82. {
  83. eventlist_t *pEvent = ListFromEvent( eventIndex );
  84. if ( pEvent )
  85. {
  86. return pEvent->iType;
  87. }
  88. return -1;
  89. }
  90. bool EventList_RegisterSharedEvent( const char *pszEventName, int iEventIndex, int iType )
  91. {
  92. // UNDONE: Do we want to do these checks when not in developer mode? or maybe DEBUG only?
  93. // They really only matter when you change the list of code controlled activities. IDs
  94. // for content controlled activities never collide because they are generated.
  95. // first, check to make sure the slot we're asking for is free. It must be for
  96. // a shared event.
  97. eventlist_t *pList = ListFromString( pszEventName );
  98. if ( !pList )
  99. {
  100. pList = ListFromEvent( iEventIndex );
  101. }
  102. //Already in list.
  103. if ( pList )
  104. {
  105. return false;
  106. }
  107. // ----------------------------------------------------------------
  108. EventList_AddEventEntry( pszEventName, iEventIndex, false, iType );
  109. return true;
  110. }
  111. Animevent EventList_RegisterPrivateEvent( const char *pszEventName )
  112. {
  113. eventlist_t *pList = ListFromString( pszEventName );
  114. if ( pList )
  115. {
  116. // this activity is already in the list. If the activity we collided with is also private,
  117. // then the collision is OK. Otherwise, it's a bug.
  118. if ( pList->isPrivate )
  119. {
  120. return (Animevent)pList->eventIndex;
  121. }
  122. else
  123. {
  124. // this private activity collides with a shared activity. That is not allowed.
  125. Warning( "***\nShared<->Private Event collision!\n***\n" );
  126. Assert(0);
  127. return AE_INVALID;
  128. }
  129. }
  130. pList = EventList_AddEventEntry( pszEventName, g_HighestEvent+1, true, AE_TYPE_SERVER );
  131. return (Animevent)pList->eventIndex;
  132. }
  133. // Get the index for a given Event name
  134. // Done at load time for all models
  135. int EventList_IndexForName( const char *pszEventName )
  136. {
  137. // this is a fast O(lgn) search (actually does 2 O(lgn) searches)
  138. eventlist_t *pList = ListFromString( pszEventName );
  139. if ( pList )
  140. {
  141. return pList->eventIndex;
  142. }
  143. return -1;
  144. }
  145. // Get the name for a given index
  146. // This should only be used in debug code, it does a linear search
  147. // But at least it only compares integers
  148. const char *EventList_NameForIndex( int eventIndex )
  149. {
  150. eventlist_t *pList = ListFromEvent( eventIndex );
  151. if ( pList )
  152. {
  153. return g_EventStrings.GetStringForKey( pList->stringKey );
  154. }
  155. return NULL;
  156. }
  157. void EventList_RegisterSharedEvents( void )
  158. {
  159. REGISTER_SHARED_ANIMEVENT( AE_EMPTY, AE_TYPE_SERVER );
  160. REGISTER_SHARED_ANIMEVENT( AE_NPC_LEFTFOOT, AE_TYPE_SERVER );
  161. REGISTER_SHARED_ANIMEVENT( AE_NPC_RIGHTFOOT, AE_TYPE_SERVER );
  162. REGISTER_SHARED_ANIMEVENT( AE_NPC_BODYDROP_LIGHT, AE_TYPE_SERVER );
  163. REGISTER_SHARED_ANIMEVENT( AE_NPC_BODYDROP_HEAVY, AE_TYPE_SERVER );
  164. REGISTER_SHARED_ANIMEVENT( AE_NPC_SWISHSOUND, AE_TYPE_SERVER );
  165. REGISTER_SHARED_ANIMEVENT( AE_NPC_180TURN, AE_TYPE_SERVER );
  166. REGISTER_SHARED_ANIMEVENT( AE_NPC_ITEM_PICKUP, AE_TYPE_SERVER );
  167. REGISTER_SHARED_ANIMEVENT( AE_NPC_WEAPON_DROP, AE_TYPE_SERVER );
  168. REGISTER_SHARED_ANIMEVENT( AE_NPC_WEAPON_SET_SEQUENCE_NAME, AE_TYPE_SERVER );
  169. REGISTER_SHARED_ANIMEVENT( AE_NPC_WEAPON_SET_SEQUENCE_NUMBER, AE_TYPE_SERVER );
  170. REGISTER_SHARED_ANIMEVENT( AE_NPC_WEAPON_SET_ACTIVITY, AE_TYPE_SERVER );
  171. REGISTER_SHARED_ANIMEVENT( AE_NPC_HOLSTER, AE_TYPE_SERVER );
  172. REGISTER_SHARED_ANIMEVENT( AE_NPC_DRAW, AE_TYPE_SERVER );
  173. REGISTER_SHARED_ANIMEVENT( AE_NPC_WEAPON_FIRE, AE_TYPE_SERVER | AE_TYPE_WEAPON );
  174. REGISTER_SHARED_ANIMEVENT( AE_CL_PLAYSOUND, AE_TYPE_CLIENT );
  175. REGISTER_SHARED_ANIMEVENT( AE_SV_PLAYSOUND, AE_TYPE_SERVER );
  176. REGISTER_SHARED_ANIMEVENT( AE_CL_STOPSOUND, AE_TYPE_CLIENT );
  177. REGISTER_SHARED_ANIMEVENT( AE_START_SCRIPTED_EFFECT, AE_TYPE_SERVER );
  178. REGISTER_SHARED_ANIMEVENT( AE_STOP_SCRIPTED_EFFECT, AE_TYPE_SERVER );
  179. REGISTER_SHARED_ANIMEVENT( AE_CLIENT_EFFECT_ATTACH, AE_TYPE_CLIENT );
  180. REGISTER_SHARED_ANIMEVENT( AE_MUZZLEFLASH, AE_TYPE_CLIENT );
  181. REGISTER_SHARED_ANIMEVENT( AE_NPC_MUZZLEFLASH, AE_TYPE_CLIENT );
  182. REGISTER_SHARED_ANIMEVENT( AE_THUMPER_THUMP, AE_TYPE_SERVER );
  183. REGISTER_SHARED_ANIMEVENT( AE_AMMOCRATE_PICKUP_AMMO, AE_TYPE_SERVER );
  184. REGISTER_SHARED_ANIMEVENT( AE_NPC_RAGDOLL, AE_TYPE_SERVER );
  185. REGISTER_SHARED_ANIMEVENT( AE_NPC_ADDGESTURE, AE_TYPE_SERVER );
  186. REGISTER_SHARED_ANIMEVENT( AE_NPC_RESTARTGESTURE, AE_TYPE_SERVER );
  187. REGISTER_SHARED_ANIMEVENT( AE_NPC_ATTACK_BROADCAST, AE_TYPE_SERVER );
  188. REGISTER_SHARED_ANIMEVENT( AE_NPC_HURT_INTERACTION_PARTNER, AE_TYPE_SERVER );
  189. REGISTER_SHARED_ANIMEVENT( AE_NPC_SET_INTERACTION_CANTDIE, AE_TYPE_SERVER );
  190. REGISTER_SHARED_ANIMEVENT( AE_SV_DUSTTRAIL, AE_TYPE_SERVER );
  191. REGISTER_SHARED_ANIMEVENT( AE_CL_CREATE_PARTICLE_EFFECT, AE_TYPE_CLIENT );
  192. REGISTER_SHARED_ANIMEVENT( AE_RAGDOLL, AE_TYPE_SERVER );
  193. REGISTER_SHARED_ANIMEVENT( AE_CL_ENABLE_BODYGROUP, AE_TYPE_CLIENT );
  194. REGISTER_SHARED_ANIMEVENT( AE_CL_DISABLE_BODYGROUP, AE_TYPE_CLIENT );
  195. REGISTER_SHARED_ANIMEVENT( AE_CL_BODYGROUP_SET_VALUE, AE_TYPE_CLIENT );
  196. REGISTER_SHARED_ANIMEVENT( AE_CL_BODYGROUP_SET_VALUE_CMODEL_WPN, AE_TYPE_CLIENT );
  197. REGISTER_SHARED_ANIMEVENT( AE_WPN_PRIMARYATTACK, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  198. REGISTER_SHARED_ANIMEVENT( AE_WPN_INCREMENTAMMO, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  199. REGISTER_SHARED_ANIMEVENT( AE_WPN_HIDE, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  200. REGISTER_SHARED_ANIMEVENT( AE_WPN_UNHIDE, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  201. REGISTER_SHARED_ANIMEVENT( AE_WPN_PLAYWPNSOUND, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  202. REGISTER_SHARED_ANIMEVENT( AE_RD_ROBOT_POP_PANELS_OFF, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  203. REGISTER_SHARED_ANIMEVENT( AE_TAUNT_ENABLE_MOVE, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  204. REGISTER_SHARED_ANIMEVENT( AE_TAUNT_DISABLE_MOVE, AE_TYPE_CLIENT | AE_TYPE_SERVER );
  205. REGISTER_SHARED_ANIMEVENT( AE_CL_REMOVE_PARTICLE_EFFECT, AE_TYPE_CLIENT );
  206. }