amxmodx/dlls/ns/ns/NMisc.cpp
2006-08-18 19:08:51 +00:00

497 lines
12 KiB
C++
Executable File

#include "ns.h"
// These are natives directly from NS2AMX
static cell AMX_NATIVE_CALL ns_has_weapon(AMX *amx,cell *params)
{
CHECK_ENTITY(params[1]);
edict_t *pEntity = NULL;
pEntity = INDEXENT2(params[1]);
if (params[3] == -1)
{
if ((pEntity->v.weapons & (1<<params[2])) > 0)
{
return 1;
}
}
else
{
if ((pEntity->v.weapons & (1<<params[2])) > 0)
{
if (params[3] == 0)
{
pEntity->v.weapons &= ~(1<<params[2]);
return 1;
}
return 0;
}
else
{
if (params[3] == 1)
{
pEntity->v.weapons |= (1<<params[2]);
return 1;
}
}
return 0;
}
return 0;
}
static cell AMX_NATIVE_CALL ns_get_spawnpoints(AMX *amx, cell *params)
{
vec3_t vRet;
if (params[2] == 0)
{
return (int)ns_spawnpoints.getnum(params[1]);
}
else
{
vRet=ns_spawnpoints.getpoint(params[1],params[2]);
cell *vCell = MF_GetAmxAddr(amx,params[3]);
vCell[0] = FLOAT_TO_CELL(vRet.x);
vCell[1] = FLOAT_TO_CELL(vRet.y);
vCell[2] = FLOAT_TO_CELL(vRet.z);
}
return 1;
}
#define MASK_ELECTRICITY 8192
static cell AMX_NATIVE_CALL ns_get_build(AMX *amx, cell *params)
{
int iLength;
char *buildtype = MF_GetAmxString(amx,params[1],0,&iLength);
int iBuiltOnly = params[2];
int iNumber = params[3];
edict_t* pBuild = NULL;
int iCount=0;
while ((pBuild = UTIL_FindEntityByString(pBuild,"classname",buildtype)) != NULL)
{
if (iBuiltOnly > 0)
{
if (FStrEq("team_advarmory",buildtype) || FStrEq("team_advturretfactory",buildtype))
{
iCount++;
}
else
{
if (pBuild->v.fuser1 >= 1000 || pBuild->v.iuser4 & MASK_ELECTRICITY)
{
iCount++;
}
}
}
else
{
iCount++;
}
if (iNumber > 0 && iCount == iNumber)
return ENTINDEX(pBuild);
}
return iCount++;
}
static cell AMX_NATIVE_CALL ns_get_speedchange(AMX *amx, cell *params)
{
// Params: get_speedchange(index)
int index=params[1];
if (!(index>0 && index<=gpGlobals->maxClients))
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
return player->speedchange;
}
static cell AMX_NATIVE_CALL ns_set_speedchange(AMX *amx, cell *params)
{
// Params: set_speedchange(index,speedchange=0)
if (!(params[1]>0&&params[1]<=32))
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
player->speedchange=params[2];
return 1;
}
static cell AMX_NATIVE_CALL ns_get_maxspeed(AMX *amx, cell *params)
{
// Params: get_maxspeed(index) (returns the max speed of the player BEFORE speed change is factored in.)
if (!(params[1]>0 && params[1]<=32))
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
return player->maxspeed;
}
static cell AMX_NATIVE_CALL ns_set_player_model(AMX *amx, cell *params)
{
// Params: set_player_model(id,szModel[])
if (!(params[1] > 0 && params[1] <= gpGlobals->maxClients))
{
MF_LogError(amx, AMX_ERR_NATIVE, "Can't set player model for a non-player entity");
return 0;
}
int len;
char *temp = MF_GetAmxString(amx,params[2],0,&len);
CPlayer *player = GET_PLAYER_I(params[1]);
if (!FStrEq(temp,""))
{
PRECACHE_MODEL(temp);
snprintf(player->model,127,"%s",temp);
player->custommodel=true;
}
else
player->custommodel=false;
return 1;
}
static cell AMX_NATIVE_CALL ns_set_player_skin(AMX *amx, cell *params)
{
// Params: set_player_skin(id,skin=-1)
if (!(params[1] > 0 && params[1] <= gpGlobals->maxClients))
{
MF_LogError(amx, AMX_ERR_NATIVE, "Can't set player skin for a non-player entity");
return 0;
}
CPlayer *player = GET_PLAYER_I(params[1]);
if (params[2] < 0)
{
// Reset skin.
player->skin=0;
player->customskin=false;
}
else
{
player->customskin=true;
player->skin=params[2];
}
return 1;
}
static cell AMX_NATIVE_CALL ns_set_player_body(AMX *amx, cell *params)
{
// Params: set_player_body(id,body=-1)
if (!(params[1] > 0 && params[1] <= gpGlobals->maxClients))
{
MF_LogError(amx, AMX_ERR_NATIVE, "Can't set player body for a non-player entity");
return 0;
}
CPlayer *player = GET_PLAYER_I(params[1]);
if (params[2] < 0)
{
// Reset body.
player->body=0;
player->custombody=false;
}
else
{
player->custombody=true;
player->body=params[2];
}
return 1;
}
static cell AMX_NATIVE_CALL ns_get_class(AMX *amx, cell *params)
{
if (params[1] < 1 || params[1] > gpGlobals->maxClients)
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
return player->GetClass();
}
static cell AMX_NATIVE_CALL ns_get_jpfuel(AMX *amx, cell *params)
{
if (params[1] < 1 || params[1] > gpGlobals->maxClients)
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
return FLOAT_TO_CELL(player->pev->fuser3 / 10.0);
}
static cell AMX_NATIVE_CALL ns_set_jpfuel(AMX *amx, cell *params)
{
if (params[1] < 1 || params[1] > gpGlobals->maxClients)
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
REAL fuel = CELL_TO_FLOAT(params[2]);
if (fuel > 100.0)
fuel = 100.0;
if (fuel < 0.0)
fuel = 0.0;
player->pev->fuser3 = fuel * 10.0;
return 1;
}
static cell AMX_NATIVE_CALL ns_is_combat(AMX *amx, cell *params)
{
return iscombat;
}
static cell AMX_NATIVE_CALL ns_get_mask(AMX *amx, cell *params)
{
int id = params[1];
if (id < 1 || id > gpGlobals->maxEntities)
return -1;
edict_t *pEntity = INDEXENT2(params[1]);
if (pEntity->v.iuser4 & params[2])
return 1;
return 0;
}
static cell AMX_NATIVE_CALL ns_set_mask(AMX *amx, cell *params)
{
int id = params[1];
if (id < 1 || id > gpGlobals->maxEntities)
return -1;
edict_t *pEntity = INDEXENT2(params[1]);
if (params[3] > 0)
{
if (pEntity->v.iuser4 & params[2])
return 0;
pEntity->v.iuser4 |= params[2];
return 1;
}
if (pEntity->v.iuser4 & params[2])
{
pEntity->v.iuser4 &= ~params[2];
return 1;
}
return 0;
}
static cell AMX_NATIVE_CALL ns_popup(AMX *amx, cell *params)
{
if (!gmsgHudText2)
gmsgHudText2 = GET_USER_MSG_ID(PLID, "HudText2", NULL);
if (params[1])
{
if (params[1] > gpGlobals->maxClients)
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
MESSAGE_BEGIN(MSG_ONE,gmsgHudText2,NULL,player->edict);
}
else
MESSAGE_BEGIN(MSG_ALL,gmsgHudText2);
int len;
char *blah = MF_GetAmxString(amx,params[2],0,&len);
char msg[190];
strncpy(msg,blah,188);
WRITE_STRING(msg);
WRITE_BYTE(params[3]);
MESSAGE_END();
return 1;
}
static cell AMX_NATIVE_CALL ns_set_fov(AMX *amx, cell *params)
{
if (params[1] < 1 || params[1] > gpGlobals->maxClients)
return 0;
CPlayer *player = GET_PLAYER_I(params[1]);
REAL fov = CELL_TO_FLOAT(params[2]);
LOG_CONSOLE(PLID,"Got fov. fov=%f",fov);
int gmsgSetFov = GET_USER_MSG_ID(PLID, "SetFOV", NULL);
if (fov == 0.0)
{
player->foved=false;
player->fov=0.0;
MESSAGE_BEGIN(MSG_ONE,gmsgSetFov,NULL,player->edict);
WRITE_BYTE(0);
MESSAGE_END();
return 1;
}
if (fov > 0)
{
player->foved=true;
player->fov=fov;
MESSAGE_BEGIN(MSG_ONE,gmsgSetFov,NULL,player->edict);
WRITE_BYTE((int)fov);
MESSAGE_END();
return 1;
}
return 0;
}
static cell AMX_NATIVE_CALL ns_giveitem(AMX *amx, cell *params)
{
int index=params[1];
int len;
char *classname = MF_GetAmxString(amx,params[2],0,&len);
if (index<1 || index>gpGlobals->maxClients)
return 0;
edict_t *player=INDEXENT2(index);
if (player->v.deadflag > 0)
return 0;
edict_t *object=CREATE_NAMED_ENTITY(ALLOC_STRING(classname)); //create
if (!object)
{
MF_LogError(amx, AMX_ERR_NATIVE, "Error creating entity \"%s\"", classname);
return 0;
}
SET_ORIGIN(object,player->v.origin); // move to player
gpGamedllFuncs->dllapi_table->pfnSpawn(object); // emulate spawn
object->v.flags |= FL_ONGROUND; // make it think it's touched the ground
gpGamedllFuncs->dllapi_table->pfnThink(object); //
gpGamedllFuncs->dllapi_table->pfnTouch(object,player); // give it to the player
return 1;
}
static cell AMX_NATIVE_CALL ns_user_kill(AMX *amx, cell *params)
{
int index = params[1];
if (index<1||index>gpGlobals->maxClients)
return 0;
edict_t *e=INDEXENT2(index);
if (e->v.iuser3 == 2 /* Commander class*/)
return 0;
if (MF_IsPlayerIngame(index) && MF_IsPlayerAlive(index))
{
float bef = e->v.frags;
edict_t *pEntity = CREATE_NAMED_ENTITY(MAKE_STRING("trigger_hurt"));
if (pEntity)
{
KeyValueData kvd;
kvd.szClassName="trigger_hurt";
kvd.szKeyName="classname";
kvd.szValue="trigger_hurt";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="dmg";
kvd.szValue="20000.0";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="damagetype";
kvd.szValue="1";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="origin";
kvd.szValue="8192 8192 8192";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
MDLL_Spawn(pEntity);
pEntity->v.classname=MAKE_STRING("slay");
MDLL_Touch(pEntity,e);
REMOVE_ENTITY(pEntity);
}
if (params[2]) e->v.frags = bef;
return 1;
}
return 0;
}
#define ANGLEVECTORS (*g_engfuncs.pfnAngleVectors)
static cell AMX_NATIVE_CALL ns_user_slap(AMX *amx, cell *params) /* 2 param */
{
int index = params[1];
if (index<1||index>gpGlobals->maxClients)
return 0;
int power = abs((int)params[2]);
edict_t *e=INDEXENT2(index);
if (e->v.iuser3 == 2 /* Commander class*/)
return 0;
if (MF_IsPlayerIngame(index) && MF_IsPlayerAlive(index)) {
if (e->v.health <= power) {
float bef = e->v.frags;
/*MDLL_ClientKill(pPlayer->pEdict);*/
edict_t *pEntity = CREATE_NAMED_ENTITY(MAKE_STRING("trigger_hurt"));
if (pEntity)
{
KeyValueData kvd;
kvd.szClassName="trigger_hurt";
kvd.szKeyName="classname";
kvd.szValue="trigger_hurt";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="dmg";
kvd.szValue="20000.0";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="damagetype";
kvd.szValue="1";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
kvd.szClassName="trigger_hurt";
kvd.szKeyName="origin";
kvd.szValue="8192 8192 8192";
kvd.fHandled=0;
MDLL_KeyValue(pEntity,&kvd);
MDLL_Spawn(pEntity);
pEntity->v.classname=MAKE_STRING("slap");
MDLL_Touch(pEntity,e);
REMOVE_ENTITY(pEntity);
}
e->v.frags = bef;
}
else {
edict_t *pEdict = e;
int numparam = *params/sizeof(cell);
if (numparam<3 || params[3]) {
pEdict->v.velocity.x += RANDOM_LONG(-600,600);
pEdict->v.velocity.y += RANDOM_LONG(-180,180);
pEdict->v.velocity.z += RANDOM_LONG(100,200);
}
else {
vec3_t v_forward, v_right;
vec3_t vang = pEdict->v.angles;
float fang[3];
fang[0] = vang.x;
fang[1] = vang.y;
fang[2] = vang.z;
ANGLEVECTORS( fang, v_forward, v_right, NULL );
pEdict->v.velocity = pEdict->v.velocity + v_forward * 220 + Vector(0,0,200);
}
pEdict->v.punchangle.x = RANDOM_LONG(-10,10);
pEdict->v.punchangle.y = RANDOM_LONG(-10,10);
pEdict->v.health -= power;
int armor = (int)pEdict->v.armorvalue;
armor -= power;
if (armor < 0) armor = 0;
pEdict->v.armorvalue = armor;
pEdict->v.dmg_inflictor = pEdict;
static const char *bit_sound[3] = {
"weapons/cbar_hitbod1.wav",
"weapons/cbar_hitbod2.wav",
"weapons/cbar_hitbod3.wav" };
EMIT_SOUND_DYN2(pEdict, CHAN_VOICE, bit_sound[RANDOM_LONG(0,2)], 1.0, ATTN_NORM, 0, PITCH_NORM);
}
return 1;
}
return 0;
}
AMX_NATIVE_INFO ns_misc_natives[] = {
///////////////////
{ "user_kill", ns_user_kill },
{ "user_slap", ns_user_slap },
{ "ns_get_build", ns_get_build },
{ "ns_set_player_model", ns_set_player_model },
{ "ns_set_player_skin", ns_set_player_skin },
{ "ns_set_player_body", ns_set_player_body },
{ "ns_has_weapon", ns_has_weapon },
{ "ns_get_spawn", ns_get_spawnpoints },
{ "ns_get_speedchange", ns_get_speedchange },
{ "ns_set_speedchange", ns_set_speedchange },
{ "ns_get_maxspeed", ns_get_maxspeed },
{ "ns_get_class", ns_get_class },
{ "ns_get_jpfuel", ns_get_jpfuel },
{ "ns_set_jpfuel", ns_set_jpfuel },
{ "ns_get_energy", ns_get_jpfuel }, // They do the same thing...
{ "ns_set_energy", ns_set_jpfuel }, //
{ "ns_is_combat", ns_is_combat },
{ "ns_get_mask", ns_get_mask },
{ "ns_set_mask", ns_set_mask },
{ "ns_popup", ns_popup },
{ "ns_set_fov", ns_set_fov },
{ "ns_give_item", ns_giveitem },
///////////////////
{ NULL, NULL }
};