OctaCore/src/game/game.cc

722 lines
20 KiB
C++

#include "game.hh"
namespace game
{
bool intermission = false;
int maptime = 0, maprealtime = 0, maplimit = -1;
int lasthit = 0, lastspawnattempt = 0;
gameent *player1 = NULL; // our client
vector<gameent *> players; // other clients
int following = -1;
VARFP(specmode, 0, 0, 2,
{
if(!specmode) stopfollowing();
else if(following < 0) nextfollow();
});
gameent *followingplayer(gameent *fallback)
{
if(player1->state!=CS_SPECTATOR || following<0) return fallback;
gameent *target = getclient(following);
if(target && target->state!=CS_SPECTATOR) return target;
return fallback;
}
ICOMMAND(getfollow, "", (),
{
gameent *f = followingplayer();
intret(f ? f->clientnum : -1);
});
void stopfollowing()
{
if(following<0) return;
following = -1;
}
void follow(char *arg)
{
}
COMMAND(follow, "s");
void nextfollow(int dir)
{
if(player1->state!=CS_SPECTATOR) return;
int cur = following >= 0 ? following : (dir < 0 ? clients.length() - 1 : 0);
loopv(clients)
{
cur = (cur + dir + clients.length()) % clients.length();
if(clients[cur] && clients[cur]->state!=CS_SPECTATOR)
{
following = cur;
return;
}
}
stopfollowing();
}
ICOMMAND(nextfollow, "i", (int *dir), nextfollow(*dir < 0 ? -1 : 1));
void checkfollow()
{
if(player1->state != CS_SPECTATOR)
{
if(following >= 0) stopfollowing();
}
else
{
if(following >= 0)
{
gameent *d = clients.inrange(following) ? clients[following] : NULL;
if(!d || d->state == CS_SPECTATOR) stopfollowing();
}
if(following < 0 && specmode) nextfollow();
}
}
const char *getclientmap() { return clientmap; }
void resetgamestate()
{
clearprojectiles();
clearbouncers();
}
gameent *spawnstate(gameent *d) // reset player state not persistent accross spawns
{
d->respawn();
d->spawnstate(0);
return d;
}
void respawnself()
{
if(ispaused()) return;
spawnplayer(player1);
lasthit = 0;
if(cmode) cmode->respawned(player1);
}
gameent *pointatplayer()
{
loopv(players) if(players[i] != player1 && intersect(players[i], player1->o, worldpos)) return players[i];
return NULL;
}
gameent *hudplayer()
{
if((thirdperson && allowthirdperson()) || specmode > 1) return player1;
return followingplayer(player1);
}
void setupcamera()
{
gameent *target = followingplayer();
if(target)
{
player1->yaw = target->yaw;
player1->pitch = target->state==CS_DEAD ? 0 : target->pitch;
player1->o = target->o;
player1->resetinterp();
}
}
bool allowthirdperson()
{
return !multiplayer(false) || player1->state==CS_SPECTATOR || player1->state==CS_EDITING || m_edit;
}
bool detachcamera()
{
gameent *d = followingplayer();
if(d) return specmode > 1 || d->state == CS_DEAD;
return player1->state == CS_DEAD;
}
bool collidecamera()
{
switch(player1->state)
{
case CS_EDITING: return false;
case CS_SPECTATOR: return followingplayer()!=NULL;
}
return true;
}
VARP(smoothmove, 0, 75, 100);
VARP(smoothdist, 0, 32, 64);
void predictplayer(gameent *d, bool move)
{
d->o = d->newpos;
d->yaw = d->newyaw;
d->pitch = d->newpitch;
d->roll = d->newroll;
if(move)
{
moveplayer(d, 1, false);
d->newpos = d->o;
}
float k = 1.0f - float(lastmillis - d->smoothmillis)/smoothmove;
if(k>0)
{
d->o.add(vec(d->deltapos).mul(k));
d->yaw += d->deltayaw*k;
if(d->yaw<0) d->yaw += 360;
else if(d->yaw>=360) d->yaw -= 360;
d->pitch += d->deltapitch*k;
d->roll += d->deltaroll*k;
}
}
void otherplayers(int curtime)
{
loopv(players)
{
gameent *d = players[i];
if(d == player1) continue;
if(d->state==CS_DEAD && d->ragdoll) moveragdoll(d);
else if(!intermission)
{
if(lastmillis - d->lastaction >= d->gunwait) d->gunwait = 0;
}
const int lagtime = totalmillis-d->lastupdate;
if(!lagtime || intermission) continue;
else if(lagtime>1000 && d->state==CS_ALIVE)
{
d->state = CS_LAGGED;
continue;
}
if(d->state==CS_ALIVE || d->state==CS_EDITING)
{
crouchplayer(d, 10, false);
if(smoothmove && d->smoothmillis>0) predictplayer(d, true);
else moveplayer(d, 1, false);
}
else if(d->state==CS_DEAD && !d->ragdoll && lastmillis-d->lastpain<2000) moveplayer(d, 1, true);
}
}
void updateworld() // main game update loop
{
if(!maptime) { maptime = lastmillis; maprealtime = totalmillis; return; }
if(!curtime) { gets2c(); if(player1->clientnum>=0) c2sinfo(); return; }
physicsframe();
updateweapons(curtime);
otherplayers(curtime);
moveragdolls();
gets2c();
if(connected)
{
if(player1->state == CS_DEAD)
{
if(player1->ragdoll) moveragdoll(player1);
else if(lastmillis-player1->lastpain<2000)
{
player1->move = player1->strafe = 0;
moveplayer(player1, 10, true);
}
}
else if(!intermission)
{
if(player1->ragdoll) cleanragdoll(player1);
crouchplayer(player1, 10, true);
moveplayer(player1, 10, true);
swayhudgun(curtime);
entities::checkitems(player1);
if(cmode) cmode->checkitems(player1);
}
}
if(player1->clientnum>=0) c2sinfo(); // do this last, to reduce the effective frame lag
}
void spawnplayer(gameent *d) // place at random spawn
{
if(cmode) cmode->pickspawn(d);
else findplayerspawn(d, -1, m_teammode ? d->team : 0);
spawnstate(d);
if(d==player1)
{
if(editmode) d->state = CS_EDITING;
else if(d->state != CS_SPECTATOR) d->state = CS_ALIVE;
}
else d->state = CS_ALIVE;
checkfollow();
}
VARP(spawnwait, 0, 0, 1000);
void respawn()
{
if(player1->state==CS_DEAD)
{
player1->attacking = ACT_IDLE;
int wait = cmode ? cmode->respawnwait(player1) : 0;
if(wait>0)
{
lastspawnattempt = lastmillis;
//conoutf(CON_GAMEINFO, "\f2you must wait %d second%s before respawn!", wait, wait!=1 ? "s" : "");
return;
}
if(lastmillis < player1->lastpain + spawnwait) return;
respawnself();
}
}
COMMAND(respawn, "");
// inputs
VARP(attackspawn, 0, 1, 1);
void doaction(int act)
{
if(!connected || intermission) return;
if((player1->attacking = act) && attackspawn) respawn();
}
ICOMMAND(shoot, "D", (int *down), doaction(*down ? ACT_SHOOT : ACT_IDLE));
ICOMMAND(melee, "D", (int *down), doaction(*down ? ACT_MELEE : ACT_IDLE));
VARP(jumpspawn, 0, 1, 1);
bool canjump()
{
if(!connected || intermission) return false;
if(jumpspawn) respawn();
return player1->state!=CS_DEAD;
}
bool cancrouch()
{
if(!connected || intermission) return false;
return player1->state!=CS_DEAD;
}
bool allowmove(physent *d)
{
if(d->type!=ENT_PLAYER) return true;
return !((gameent *)d)->lasttaunt || lastmillis-((gameent *)d)->lasttaunt>=1000;
}
void taunt()
{
if(player1->state!=CS_ALIVE || player1->physstate<PHYS_SLOPE) return;
if(lastmillis-player1->lasttaunt<1000) return;
player1->lasttaunt = lastmillis;
addmsg(N_TAUNT, "rc", player1);
}
COMMAND(taunt, "");
VARP(hitsound, 0, 0, 1);
void damaged(int damage, gameent *d, gameent *actor, bool local)
{
if((d->state!=CS_ALIVE && d->state != CS_LAGGED && d->state != CS_SPAWNING) || intermission) return;
if(local) damage = d->dodamage(damage);
else if(actor==player1) return;
gameent *h = hudplayer();
if(h!=player1 && actor==h && d!=actor)
{
if(hitsound && lasthit != lastmillis) playsound(S_HIT);
lasthit = lastmillis;
}
if(d==h)
{
damageblend(damage);
damagecompass(damage, actor->o);
}
damageeffect(damage, d, d!=h);
if(d->health<=0) { if(local) killed(d, actor); }
else if(d==h) playsound(S_PAIN2);
else playsound(S_PAIN1, &d->o);
}
void deathstate(gameent *d, bool restore)
{
d->state = CS_DEAD;
d->lastpain = lastmillis;
if(!restore)
{
gibeffect(max(-d->health, 0), d->vel, d);
d->deaths++;
}
if(d==player1)
{
disablezoom();
d->attacking = ACT_IDLE;
//d->pitch = 0;
d->roll = 0;
playsound(S_DIE2);
}
else
{
d->move = d->strafe = 0;
d->resetinterp();
d->smoothmillis = 0;
playsound(S_DIE1, &d->o);
}
}
VARP(teamcolorfrags, 0, 1, 1);
void killed(gameent *d, gameent *actor)
{
if(d->state==CS_EDITING)
{
d->editstate = CS_DEAD;
d->deaths++;
if(d!=player1) d->resetinterp();
return;
}
else if((d->state!=CS_ALIVE && d->state != CS_LAGGED && d->state != CS_SPAWNING) || intermission) return;
gameent *h = followingplayer(player1);
int contype = d==h || actor==h ? CON_FRAG_SELF : CON_FRAG_OTHER;
const char *dname = "", *aname = "";
if(m_teammode && teamcolorfrags)
{
dname = teamcolorname(d, "you");
aname = teamcolorname(actor, "you");
}
else
{
dname = colorname(d, NULL, "you");
aname = colorname(actor, NULL, "you");
}
if(d==actor)
conoutf(contype, "\f2%s suicided%s", dname, d==player1 ? "!" : "");
else if(isteam(d->team, actor->team))
{
contype |= CON_TEAMKILL;
if(actor==player1) conoutf(contype, "\f6%s fragged a teammate (%s)", aname, dname);
else if(d==player1) conoutf(contype, "\f6%s got fragged by a teammate (%s)", dname, aname);
else conoutf(contype, "\f2%s fragged a teammate (%s)", aname, dname);
}
else
{
if(d==player1) conoutf(contype, "\f2%s got fragged by %s", dname, aname);
else conoutf(contype, "\f2%s fragged %s", aname, dname);
}
deathstate(d);
}
void timeupdate(int secs)
{
if(secs > 0)
{
maplimit = lastmillis + secs*1000;
}
else
{
intermission = true;
player1->attacking = ACT_IDLE;
if(cmode) cmode->gameover();
conoutf(CON_GAMEINFO, "\f2intermission:");
conoutf(CON_GAMEINFO, "\f2game has ended!");
conoutf(CON_GAMEINFO, "\f2player frags: %d, deaths: %d", player1->frags, player1->deaths);
int accuracy = (player1->totaldamage*100)/max(player1->totalshots, 1);
conoutf(CON_GAMEINFO, "\f2player total damage dealt: %d, damage wasted: %d, accuracy(%%): %d", player1->totaldamage, player1->totalshots-player1->totaldamage, accuracy);
disablezoom();
execident("intermission");
}
}
ICOMMAND(getfrags, "", (), intret(player1->frags));
ICOMMAND(getflags, "", (), intret(player1->flags));
ICOMMAND(getdeaths, "", (), intret(player1->deaths));
ICOMMAND(getaccuracy, "", (), intret((player1->totaldamage*100)/max(player1->totalshots, 1)));
ICOMMAND(gettotaldamage, "", (), intret(player1->totaldamage));
ICOMMAND(gettotalshots, "", (), intret(player1->totalshots));
vector<gameent *> clients;
gameent *newclient(int cn) // ensure valid entity
{
if(cn < 0 || cn > max(0xFF, MAXCLIENTS))
{
neterr("clientnum", false);
return NULL;
}
if(cn == player1->clientnum) return player1;
while(cn >= clients.length()) clients.add(NULL);
if(!clients[cn])
{
gameent *d = new gameent;
d->clientnum = cn;
clients[cn] = d;
players.add(d);
}
return clients[cn];
}
gameent *getclient(int cn) // ensure valid entity
{
if(cn == player1->clientnum) return player1;
return clients.inrange(cn) ? clients[cn] : NULL;
}
void clientdisconnected(int cn, bool notify)
{
if(!clients.inrange(cn)) return;
gameent *d = clients[cn];
if(d)
{
if(notify && d->name[0]) conoutf("\f4leave:\f7 %s", colorname(d));
removeweapons(d);
removetrackedparticles(d);
removetrackeddynlights(d);
if(cmode) cmode->removeplayer(d);
players.removeobj(d);
DELETEP(clients[cn]);
cleardynentcache();
}
if(following == cn)
{
if(specmode) nextfollow();
else stopfollowing();
}
}
void clearclients(bool notify)
{
loopv(clients) if(clients[i]) clientdisconnected(i, notify);
}
void initclient()
{
player1 = spawnstate(new gameent);
filtertext(player1->name, "unnamed", false, false, MAXNAMELEN);
players.add(player1);
}
VARP(showmodeinfo, 0, 1, 1);
void startgame()
{
clearprojectiles();
clearbouncers();
clearragdolls();
// reset perma-state
loopv(players) players[i]->startgame();
setclientmode();
intermission = false;
maptime = maprealtime = 0;
maplimit = -1;
if(cmode)
{
cmode->preload();
cmode->setup();
}
syncplayer();
disablezoom();
lasthit = 0;
execident("mapstart");
}
void startmap(const char *name) // called just after a map load
{
spawnplayer(player1);
entities::resetspawns();
copystring(clientmap, name ? name : "");
sendmapinfo();
}
const char *getmapinfo()
{
return NULL;
}
const char *getscreenshotinfo()
{
return NULL;
}
void physicstrigger(physent *d, bool local, int floorlevel, int waterlevel, int material)
{
if (waterlevel>0) { if(material!=MAT_LAVA) playsound(S_SPLASHOUT, d==player1 ? NULL : &d->o); }
else if(waterlevel<0) playsound(material==MAT_LAVA ? S_BURN : S_SPLASHIN, d==player1 ? NULL : &d->o);
if (floorlevel>0) { if(d==player1 || d->type!=ENT_PLAYER) msgsound(S_JUMP, d); }
else if(floorlevel<0) { if(d==player1 || d->type!=ENT_PLAYER) msgsound(S_LAND, d); }
}
void dynentcollide(physent *d, physent *o, const vec &dir)
{
}
void msgsound(int n, physent *d)
{
if(!d || d==player1)
{
addmsg(N_SOUND, "ci", d, n);
playsound(n);
}
else
{
playsound(n, &d->o);
}
}
int numdynents() { return players.length(); }
dynent *iterdynents(int i)
{
if(i<players.length()) return players[i];
return NULL;
}
bool duplicatename(gameent *d, const char *name = NULL, const char *alt = NULL)
{
if(!name) name = d->name;
if(alt && d != player1 && !strcmp(name, alt)) return true;
loopv(players) if(d!=players[i] && !strcmp(name, players[i]->name)) return true;
return false;
}
const char *colorname(gameent *d, const char *name, const char * alt, const char *color)
{
if(!name) name = alt && d == player1 ? alt : d->name;
bool dup = !name[0] || duplicatename(d, name, alt);
if(dup || color[0])
{
if(dup) return tempformatstring("\fs%s%s \f5(%d)\fr", color, name, d->clientnum);
return tempformatstring("\fs%s%s\fr", color, name);
}
return name;
}
VARP(teamcolortext, 0, 1, 1);
const char *teamcolorname(gameent *d, const char *alt)
{
if(!teamcolortext || !m_teammode || !validteam(d->team) || d->state == CS_SPECTATOR) return colorname(d, NULL, alt);
return colorname(d, NULL, alt, teamtextcode[d->team]);
}
const char *teamcolor(const char *prefix, const char *suffix, int team, const char *alt)
{
if(!teamcolortext || !m_teammode || !validteam(team)) return alt;
return tempformatstring("\fs%s%s%s%s\fr", teamtextcode[team], prefix, teamnames[team], suffix);
}
VARP(teamsounds, 0, 1, 1);
void teamsound(bool sameteam, int n, const vec *loc)
{
playsound(n, loc, NULL, teamsounds ? (m_teammode && sameteam ? SND_USE_ALT : SND_NO_ALT) : 0);
}
void teamsound(gameent *d, int n, const vec *loc)
{
teamsound(isteam(d->team, player1->team), n, loc);
}
bool needminimap() { return false; }
void drawicon(int icon, float x, float y, float sz)
{
settexture("media/interface/hud/items.png");
float tsz = 0.25f, tx = tsz*(icon%4), ty = tsz*(icon/4);
gle::defvertex(2);
gle::deftexcoord0();
gle::begin(GL_TRIANGLE_STRIP);
gle::attribf(x, y); gle::attribf(tx, ty);
gle::attribf(x+sz, y); gle::attribf(tx+tsz, ty);
gle::attribf(x, y+sz); gle::attribf(tx, ty+tsz);
gle::attribf(x+sz, y+sz); gle::attribf(tx+tsz, ty+tsz);
gle::end();
}
float abovegameplayhud(int w, int h)
{
switch(hudplayer()->state)
{
case CS_EDITING:
case CS_SPECTATOR:
return 1;
default:
return 1650.0f/1800.0f;
}
}
void drawhudicons(gameent *d)
{
}
void gameplayhud(int w, int h)
{
pushhudscale(h/1800.0f);
gameent *d = hudplayer();
if(d->state!=CS_EDITING)
{
if(d->state!=CS_SPECTATOR) drawhudicons(d);
if(cmode) cmode->drawhud(d, w, h);
}
pophudmatrix();
}
float clipconsole(float w, float h)
{
if(cmode) return cmode->clipconsole(w, h);
return 0;
}
VARP(teamcrosshair, 0, 1, 1);
VARP(hitcrosshair, 0, 425, 1000);
const char *defaultcrosshair(int index)
{
switch(index)
{
case 2: return "media/interface/crosshair/default_hit.png";
case 1: return "media/interface/crosshair/teammate.png";
default: return "media/interface/crosshair/default.png";
}
}
int selectcrosshair(vec &)
{
return 0;
}
// any data written into this vector will get saved with the map data. Must take care to do own versioning, and endianess if applicable. Will not get called when loading maps from other games, so provide defaults.
void writegamedata(vector<char> &extras) {}
void readgamedata(vector<char> &extras) {}
const char *gameconfig() { return "config/game.cfg"; }
const char *savedconfig() { return "config/saved.cfg"; }
const char *restoreconfig() { return "config/restore.cfg"; }
const char *defaultconfig() { return "config/default.cfg"; }
const char *autoexec() { return "config/autoexec.cfg"; }
const char *savedservers() { return "config/servers.cfg"; }
void loadconfigs()
{
execfile("config/auth.cfg", false);
}
bool clientoption(const char *arg) { return false; }
}