ddnet/src/game/server/gs_game.cpp

272 lines
5.1 KiB
C++
Raw Normal View History

2007-11-25 19:42:40 +00:00
/* copyright (c) 2007 magnus auvinen, see licence.txt for more info */
2007-12-15 10:24:49 +00:00
#include <engine/e_config.h>
#include "gs_common.h"
#include <string.h>
gameobject::gameobject()
: entity(OBJTYPE_GAME)
{
// select gametype
2007-12-11 23:10:07 +00:00
if(strcmp(config.sv_gametype, "ctf") == 0)
{
gametype = GAMETYPE_CTF;
dbg_msg("game", "-- Capture The Flag --");
}
2007-12-11 23:10:07 +00:00
else if(strcmp(config.sv_gametype, "tdm") == 0)
{
gametype = GAMETYPE_TDM;
dbg_msg("game", "-- Team Death Match --");
}
else
{
gametype = GAMETYPE_DM;
dbg_msg("game", "-- Death Match --");
}
//
2007-12-11 23:10:07 +00:00
do_warmup(config.sv_warmup);
game_over_tick = -1;
sudden_death = 0;
round_start_tick = server_tick();
round_count = 0;
is_teamplay = false;
2007-12-09 09:47:05 +00:00
teamscore[0] = 0;
teamscore[1] = 0;
}
void gameobject::endround()
{
2007-10-06 17:36:24 +00:00
if(warmup) // game can't end when we are running warmup
return;
world->paused = true;
game_over_tick = server_tick();
sudden_death = 0;
}
void gameobject::resetgame()
{
world->reset_requested = true;
}
2008-01-12 12:27:55 +00:00
static bool is_separator(char c) { return c == ';' || c == ' ' || c == ',' || c == '\t'; }
void gameobject::startround()
{
resetgame();
round_start_tick = server_tick();
sudden_death = 0;
game_over_tick = -1;
world->paused = false;
teamscore[0] = 0;
teamscore[1] = 0;
round_count++;
}
void gameobject::cyclemap()
{
if(!strlen(config.sv_maprotation))
return;
// handle maprotation
char buf[512];
const char *s = strstr(config.sv_maprotation, config.sv_map);
if(s == 0)
s = config.sv_maprotation; // restart rotation
else
{
s += strlen(config.sv_map); // skip this map
while(is_separator(s[0]))
s++;
if(s[0] == 0)
s = config.sv_maprotation; // restart rotation
}
int i = 0;
for(; i < 512; i++)
{
buf[i] = s[i];
if(is_separator(s[i]) || s[i] == 0)
{
buf[i] = 0;
break;
}
}
i = 0; // skip spaces
while(is_separator(buf[i]))
i++;
dbg_msg("game", "rotating map to %s", &buf[i]);
strcpy(config.sv_map, &buf[i]);
}
void gameobject::post_reset()
{
for(int i = 0; i < MAX_CLIENTS; i++)
{
if(players[i].client_id != -1)
players[i].respawn();
}
}
void gameobject::on_player_info_change(class player *p)
{
2007-11-26 22:26:33 +00:00
const int team_colors[2] = {65387, 10223467};
if(is_teamplay)
{
if(p->team >= 0 || p->team <= 1)
{
p->use_custom_color = 1;
p->color_body = team_colors[p->team];
p->color_feet = team_colors[p->team];
}
}
}
int gameobject::on_player_death(class player *victim, class player *killer, int weapon)
{
// do scoreing
if(!killer)
return 0;
if(killer == victim)
2007-12-11 21:19:52 +00:00
victim->score--; // suicide
else
2007-12-11 21:19:52 +00:00
{
if(is_teamplay && victim->team == killer->team)
killer->score--; // teamkill
else
killer->score++; // normal kill
}
return 0;
}
2007-10-06 17:36:24 +00:00
void gameobject::do_warmup(int seconds)
{
warmup = seconds*SERVER_TICK_SPEED;
}
2007-12-10 20:53:37 +00:00
bool gameobject::is_friendly_fire(int cid1, int cid2)
{
if(cid1 == cid2)
return false;
if(is_teamplay)
{
if(players[cid1].team == players[cid2].team)
return true;
}
return false;
}
void gameobject::tick()
{
2007-10-06 17:36:24 +00:00
// do warmup
if(warmup)
{
warmup--;
if(!warmup)
2007-12-10 21:05:57 +00:00
startround();
2007-10-06 17:36:24 +00:00
}
if(game_over_tick != -1)
{
// game over.. wait for restart
if(server_tick() > game_over_tick+server_tickspeed()*10)
{
cyclemap();
startround();
}
}
// update browse info
int prog = -1;
2007-12-11 23:10:07 +00:00
if(config.sv_timelimit > 0)
prog = max(prog, (server_tick()-round_start_tick) * 100 / (config.sv_timelimit*server_tickspeed()*60));
2007-12-11 23:10:07 +00:00
if(config.sv_scorelimit)
{
if(is_teamplay)
{
2007-12-11 23:10:07 +00:00
prog = max(prog, (teamscore[0]*100)/config.sv_scorelimit);
prog = max(prog, (teamscore[1]*100)/config.sv_scorelimit);
}
else
{
for(int i = 0; i < MAX_CLIENTS; i++)
{
if(players[i].client_id != -1)
2007-12-11 23:10:07 +00:00
prog = max(prog, (players[i].score*100)/config.sv_scorelimit);
}
}
}
if(warmup)
prog = -1;
server_setbrowseinfo(gametype, prog);
}
void gameobject::snap(int snapping_client)
{
obj_game *game = (obj_game *)snap_new_item(OBJTYPE_GAME, 0, sizeof(obj_game));
game->paused = world->paused;
game->game_over = game_over_tick==-1?0:1;
game->sudden_death = sudden_death;
2007-12-11 23:10:07 +00:00
game->score_limit = config.sv_scorelimit;
game->time_limit = config.sv_timelimit;
game->round_start_tick = round_start_tick;
game->gametype = gametype;
2007-10-06 17:36:24 +00:00
game->warmup = warmup;
game->teamscore[0] = teamscore[0];
game->teamscore[1] = teamscore[1];
}
int gameobject::getteam(int notthisid)
{
int numplayers[2] = {0,0};
for(int i = 0; i < MAX_CLIENTS; i++)
{
if(players[i].client_id != -1 && players[i].client_id != notthisid)
{
if(players[i].team == 0 || players[i].team == 1)
numplayers[players[i].team]++;
}
}
return numplayers[0] > numplayers[1] ? 1 : 0;
}
2007-11-26 20:47:49 +00:00
void gameobject::do_team_wincheck()
{
if(game_over_tick == -1 && !warmup)
{
// check score win condition
2007-12-11 23:10:07 +00:00
if((config.sv_scorelimit > 0 && (teamscore[0] >= config.sv_scorelimit || teamscore[1] >= config.sv_scorelimit)) ||
(config.sv_timelimit > 0 && (server_tick()-round_start_tick) >= config.sv_timelimit*server_tickspeed()*60))
2007-11-26 20:47:49 +00:00
{
if(teamscore[0] != teamscore[1])
endround();
else
sudden_death = 1;
}
}
}
int gameobject::clampteam(int team)
{
if(team < 0) // spectator
return -1;
if(is_teamplay)
return team&1;
return 0;
}
gameobject *gameobj = 0;