2
0
mirror of https://github.com/alliedmodders/amxmodx.git synced 2025-01-18 01:38:04 +03:00

336 lines
7.6 KiB
C++
Raw Normal View History

2004-06-28 13:57:10 +00:00
/*
* TSX
* Copyright (c) 2004 Lukasz Wlasinski
*
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* In addition, as a special exception, the author gives permission to
* link the code of this program with the Half-Life Game Engine ("HL
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
* L.L.C ("Valve"). You must obey the GNU General Public License in all
* respects for all of the code used other than the HL Engine and MODs
* from Valve. If you modify this file, you may extend this exception
* to your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from your
* version.
*
*/
#include "amxxmodule.h"
#include "CRank.h"
#include "tsx.h"
// *****************************************************
// class Stats
// *****************************************************
2005-12-11 18:42:19 +00:00
Stats::Stats()
{
2004-06-28 13:57:10 +00:00
hits = shots = damage = hs = tks = kills = deaths = 0;
memset( bodyHits , 0 ,sizeof( bodyHits ) );
}
2005-12-11 18:42:19 +00:00
void Stats::commit(Stats* a)
{
2004-06-28 13:57:10 +00:00
hits += a->hits;
shots += a->shots;
damage += a->damage;
hs += a->hs;
tks += a->tks;
kills += a->kills;
deaths += a->deaths;
2005-12-11 18:42:19 +00:00
for(int i = 1; i < 8; ++i) bodyHits[i] += a->bodyHits[i];
2004-06-28 13:57:10 +00:00
}
// *****************************************************
// class RankSystem
// *****************************************************
2005-12-11 18:42:19 +00:00
RankSystem::RankStats::RankStats( const char* uu, const char* nn, RankSystem* pp )
{
2004-06-28 13:57:10 +00:00
name = 0;
namelen = 0;
unique = 0;
uniquelen = 0;
score = 0;
parent = pp;
id = ++parent->rankNum;
next = prev = 0;
setName( nn );
setUnique( uu );
}
2005-12-11 18:42:19 +00:00
RankSystem::RankStats::~RankStats()
{
delete name;
delete unique;
2004-06-28 13:57:10 +00:00
--parent->rankNum;
}
2005-12-11 18:42:19 +00:00
void RankSystem::RankStats::setName( const char* nn )
{
2004-06-28 13:57:10 +00:00
delete[] name;
namelen = strlen(nn) + 1;
name = new char[namelen];
2005-12-11 18:42:19 +00:00
if ( name ) strcpy( name , nn );
else namelen = 0;
2004-06-28 13:57:10 +00:00
}
2005-12-11 18:42:19 +00:00
void RankSystem::RankStats::setUnique( const char* nn )
{
delete unique;
2004-06-28 13:57:10 +00:00
uniquelen = strlen(nn) + 1;
unique = new char[uniquelen];
2005-12-11 18:42:19 +00:00
if ( unique ) strcpy( unique , nn );
else uniquelen = 0;
2004-06-28 13:57:10 +00:00
}
2005-12-11 18:42:19 +00:00
RankSystem::RankSystem()
{
2004-06-28 13:57:10 +00:00
head = 0;
tail = 0;
rankNum = 0;
calc.code = 0;
}
2005-12-11 18:42:19 +00:00
RankSystem::~RankSystem()
{
2004-06-28 13:57:10 +00:00
clear();
}
2005-12-11 18:42:19 +00:00
void RankSystem::put_before( RankStats* a, RankStats* ptr )
{
2004-06-28 13:57:10 +00:00
a->next = ptr;
2005-12-11 18:42:19 +00:00
if ( ptr )
{
2004-06-28 13:57:10 +00:00
a->prev = ptr->prev;
ptr->prev = a;
}
2005-12-11 18:42:19 +00:00
else
{
2004-06-28 13:57:10 +00:00
a->prev = head;
head = a;
}
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
if ( a->prev ) a->prev->next = a;
else tail = a;
}
2005-12-11 18:42:19 +00:00
void RankSystem::put_after( RankStats* a, RankStats* ptr )
{
2004-06-28 13:57:10 +00:00
a->prev = ptr;
2005-12-11 18:42:19 +00:00
if ( ptr )
{
2004-06-28 13:57:10 +00:00
a->next = ptr->next;
ptr->next = a;
}
2005-12-11 18:42:19 +00:00
else
{
2004-06-28 13:57:10 +00:00
a->next = tail;
tail = a;
}
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
if ( a->next ) a->next->prev = a;
else head = a;
}
2005-12-11 18:42:19 +00:00
void RankSystem::unlink( RankStats* ptr )
{
2004-06-28 13:57:10 +00:00
if (ptr->prev) ptr->prev->next = ptr->next;
else tail = ptr->next;
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
if (ptr->next) ptr->next->prev = ptr->prev;
else head = ptr->prev;
}
2005-12-11 18:42:19 +00:00
void RankSystem::clear()
{
while( tail )
{
2004-06-28 13:57:10 +00:00
head = tail->next;
delete tail;
tail = head;
}
}
bool RankSystem::loadCalc(const char* filename, char* error)
{
2004-09-08 07:48:23 +00:00
if ((MF_LoadAmxScript(&calc.amx,&calc.code,filename,error,0)!=AMX_ERR_NONE)||
2004-06-28 13:57:10 +00:00
(MF_AmxAllot(&calc.amx, 8 , &calc.amxAddr1, &calc.physAddr1)!=AMX_ERR_NONE)||
(MF_AmxAllot(&calc.amx, 8 , &calc.amxAddr2, &calc.physAddr2)!=AMX_ERR_NONE)||
2005-12-11 18:42:19 +00:00
(MF_AmxFindPublic(&calc.amx,"get_score",&calc.func)!=AMX_ERR_NONE))
{
2004-06-28 13:57:10 +00:00
MF_PrintSrvConsole("Couldn't load plugin (file \"%s\")",filename);
MF_UnloadAmxScript(&calc.amx, &calc.code);
return false;
}
return true;
}
void RankSystem::unloadCalc()
{
MF_UnloadAmxScript(&calc.amx , &calc.code);
}
RankSystem::RankStats* RankSystem::findEntryInRank(const char* unique, const char* name )
{
RankStats* a = head;
while ( a )
{
2005-12-11 18:42:19 +00:00
if ( strcmp( a->getUnique() ,unique ) == 0 ) return a;
2004-06-28 13:57:10 +00:00
a = a->prev;
}
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
a = new RankStats( unique ,name,this );
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
if ( a == 0 ) return 0;
put_after( a , 0 );
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
return a;
}
void RankSystem::updatePos( RankStats* rr , Stats* s )
{
rr->addStats( s );
2005-12-11 18:42:19 +00:00
if ( calc.code )
{
2004-06-28 13:57:10 +00:00
calc.physAddr1[0] = rr->kills;
calc.physAddr1[1] = rr->deaths;
calc.physAddr1[2] = rr->hs;
calc.physAddr1[3] = rr->tks;
calc.physAddr1[4] = rr->shots;
calc.physAddr1[5] = rr->hits;
calc.physAddr1[6] = rr->damage;
2005-12-11 18:42:19 +00:00
for(int i = 1; i < 8; ++i) calc.physAddr2[i] = rr->bodyHits[i];
2004-06-28 13:57:10 +00:00
cell result = 0;
int err;
2005-12-11 18:42:19 +00:00
2005-08-01 04:55:55 +00:00
MF_AmxPush(&calc.amx, calc.amxAddr2);
MF_AmxPush(&calc.amx, calc.amxAddr1);
2005-12-11 18:42:19 +00:00
2005-08-01 04:55:55 +00:00
if ((err = MF_AmxExec(&calc.amx,&result, calc.func)) != AMX_ERR_NONE)
MF_LogError(&calc.amx, err, "Fatal error calculating stats");
2005-12-11 18:42:19 +00:00
2004-06-28 13:57:10 +00:00
rr->score = result;
}
else rr->score = rr->kills - rr->deaths;
RankStats* aa = rr->next;
2005-12-11 18:42:19 +00:00
while ( aa && (aa->score <= rr->score) )
{ // try to nominate
2004-06-28 13:57:10 +00:00
rr->goUp();
aa->goDown();
aa = aa->next; // go to next rank
}
if ( aa != rr->next )
{
unlink( rr );
put_before( rr, aa );
}
else
{
aa = rr->prev;
2005-12-11 18:42:19 +00:00
while ( aa && (aa->score > rr->score) )
{ // go down
2004-06-28 13:57:10 +00:00
rr->goDown();
aa->goUp();
aa = aa->prev; // go to prev rank
}
2005-12-11 18:42:19 +00:00
if ( aa != rr->prev )
{
2004-06-28 13:57:10 +00:00
unlink( rr );
put_after( rr, aa );
}
}
}
void RankSystem::loadRank( const char* filename )
{
FILE *bfp = fopen( filename , "rb" );
if ( !bfp ) return;
short int i = 0;
fread(&i, 1 , sizeof(short int) , bfp);
if (i == RANK_VERSION)
{
Stats d;
char unique[64], name[64];
fread(&i , 1, sizeof(short int), bfp);
while( i )
{
fread(name , i,sizeof(char) , bfp);
fread(&i , 1, sizeof(short int), bfp);
fread(unique , i,sizeof(char) , bfp);
fread(&d.tks, 1,sizeof(int), bfp);
fread(&d.damage, 1,sizeof(int), bfp);
fread(&d.deaths, 1,sizeof(int), bfp);
fread(&d.kills, 1,sizeof(int), bfp);
fread(&d.shots, 1,sizeof(int), bfp);
fread(&d.hits, 1,sizeof(int), bfp);
fread(&d.hs, 1,sizeof(int), bfp);
fread(d.bodyHits, 1,sizeof(d.bodyHits), bfp);
fread(&i , 1, sizeof(short int), bfp);
RankSystem::RankStats* a = findEntryInRank( unique , name );
if ( a ) a->updatePosition( &d );
}
}
fclose(bfp);
}
void RankSystem::saveRank( const char* filename )
{
FILE *bfp = fopen(filename, "wb");
if ( !bfp ) return;
short int i = RANK_VERSION;
fwrite(&i, 1, sizeof(short int) , bfp);
RankSystem::iterator a = front();
while ( a )
{
if ( (*a).score != (1<<31) ) // score must be different than mincell
{
fwrite( &(*a).namelen , 1, sizeof(short int), bfp);
fwrite( (*a).name , (*a).namelen , sizeof(char) , bfp);
fwrite( &(*a).uniquelen , 1, sizeof(short int), bfp);
fwrite( (*a).unique , (*a).uniquelen , sizeof(char) , bfp);
fwrite( &(*a).kills, 1, sizeof(int), bfp);
fwrite( &(*a).deaths, 1, sizeof(int), bfp);
fwrite( &(*a).hs, 1, sizeof(int), bfp);
fwrite( &(*a).tks, 1, sizeof(int), bfp);
fwrite( &(*a).damage, 1, sizeof(int), bfp);
fwrite( &(*a).hits, 1, sizeof(int), bfp);
fwrite( &(*a).shots, 1, sizeof(int), bfp);
fwrite( (*a).bodyHits, 1, sizeof((*a).bodyHits), bfp);
}
--a;
}
i = 0;
fwrite( &i , 1, sizeof(short int), bfp); // null terminator
fclose(bfp);
2004-09-08 07:48:23 +00:00
}