]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/server/client.qh
Merge branch 'master' into terencehill/bot_waypoints
[xonotic/xonotic-data.pk3dir.git] / qcsrc / server / client.qh
1 #pragma once
2
3 void ClientState_attach(entity this);
4
5 IntrusiveList g_players;
6 STATIC_INIT(g_players) { g_players = IL_NEW(); }
7
8 CLASS(Client, Object)
9     /** Client name */
10     ATTRIB(Client, netname, string, this.netname);
11     ATTRIB(Client, colormap, int, this.colormap);
12     ATTRIB(Client, team, int, this.team);
13     ATTRIB(Client, clientcolors, int, this.clientcolors);
14     /** Client IP */
15     ATTRIB(Client, netaddress, string, this.netaddress);
16     ATTRIB(Client, playermodel, string, this.playermodel);
17     ATTRIB(Client, playerskin, int, this.playerskin);
18
19     /** fingerprint of CA key the player used to authenticate */
20     ATTRIB(Client, crypto_keyfp, string, this.crypto_keyfp);
21     /** fingerprint of CA key the server used to authenticate to the player */
22     ATTRIB(Client, crypto_mykeyfp, string, this.crypto_mykeyfp);
23     /** fingerprint of ID used by the player entity, or string_null if not identified */
24     ATTRIB(Client, crypto_idfp, string, this.crypto_idfp);
25     /** set if the player's ID has been signed */
26     ATTRIB(Client, crypto_idfp_signed, bool, this.crypto_idfp_signed);
27     /** the string "AES128" if encrypting, and string_null if plaintext */
28     ATTRIB(Client, crypto_encryptmethod, string, this.crypto_encryptmethod);
29     /** the string "HMAC-SHA256" if signing, and string_null if plaintext */
30     ATTRIB(Client, crypto_signmethod, string, this.crypto_signmethod);
31
32     // engine client fields
33     ATTRIB(Client, impulse, int, this.impulse);
34
35     ATTRIB(Client, button0, int, this.button0);
36     ATTRIB(Client, button2, int, this.button2);
37     ATTRIB(Client, button3, int, this.button3);
38     ATTRIB(Client, button4, int, this.button4);
39     ATTRIB(Client, button5, int, this.button5);
40     ATTRIB(Client, button6, int, this.button6);
41     ATTRIB(Client, button7, int, this.button7);
42     ATTRIB(Client, button8, int, this.button8);
43     ATTRIB(Client, button9, int, this.button9);
44     ATTRIB(Client, button10, int, this.button10);
45     ATTRIB(Client, button11, int, this.button11);
46     ATTRIB(Client, button12, int, this.button12);
47     ATTRIB(Client, button13, int, this.button13);
48     ATTRIB(Client, button14, int, this.button14);
49     ATTRIB(Client, button15, int, this.button15);
50     ATTRIB(Client, button16, int, this.button16);
51     ATTRIB(Client, buttonuse, int, this.buttonuse);
52     ATTRIB(Client, buttonchat, int, this.buttonchat);
53
54     ATTRIB(Client, cursor_active, int, this.cursor_active);
55     ATTRIB(Client, cursor_screen, vector, this.cursor_screen);
56     ATTRIB(Client, cursor_trace_start, vector, this.cursor_trace_start);
57     ATTRIB(Client, cursor_trace_endpos, vector, this.cursor_trace_endpos);
58     ATTRIB(Client, cursor_trace_ent, entity, this.cursor_trace_ent);
59
60     ATTRIB(Client, ping, float, this.ping);
61     ATTRIB(Client, ping_packetloss, float, this.ping_packetloss);
62     ATTRIB(Client, ping_movementloss, float, this.ping_movementloss);
63
64     ATTRIB(Client, v_angle, vector, this.v_angle);
65     ATTRIB(Client, movement, vector, this.movement);
66
67     // custom
68
69     ATTRIB(Client, playerid, int, this.playerid);
70
71     ATTRIB(Client, parm_idlesince, int, this.parm_idlesince);
72     ATTRIB(Client, muted, bool, this.muted);
73     ATTRIB(Client, killindicator_teamchange, int, this.killindicator_teamchange);
74     ATTRIB(Client, idlekick_lasttimeleft, float, this.idlekick_lasttimeleft);
75     ATTRIB(Client, pm_frametime, float, this.pm_frametime);
76     ATTRIB(Client, pressedkeys, int, this.pressedkeys);
77     ATTRIB(Client, movement_old, vector, this.movement_old);
78     ATTRIB(Client, buttons_old, int, this.buttons_old);
79     ATTRIB(Client, teamkill_complain, float, this.teamkill_complain);
80     ATTRIB(Client, teamkill_soundtime, float, this.teamkill_soundtime);
81     ATTRIB(Client, teamkill_soundsource, entity, this.teamkill_soundsource);
82     ATTRIB(Client, usekeypressed, bool, this.usekeypressed);
83     ATTRIB(Client, motd_actived_time, float, this.motd_actived_time);
84     ATTRIB(Client, jointime, float, this.jointime);
85     ATTRIB(Client, spectatortime, float, this.spectatortime);
86     ATTRIB(Client, version_nagtime, float, this.version_nagtime);
87     ATTRIB(Client, netname_previous, string, this.netname_previous);
88     ATTRIB(Client, allowed_timeouts, int, this.allowed_timeouts);
89     ATTRIB(Client, active_minigame, entity, this.active_minigame);
90     ATTRIB(Client, taunt_soundtime, float, this.taunt_soundtime);
91     ATTRIB(Client, killcount, int, this.killcount);
92     ATTRIB(Client, version_mismatch, bool, this.version_mismatch);
93     ATTRIB(Client, version, int, this.version);
94     ATTRIB(Client, spectatee_status, int, this.spectatee_status);
95     ATTRIB(Client, zoomstate, bool, this.zoomstate);
96     ATTRIB(Client, just_joined, bool, this.just_joined);
97     ATTRIB(Client, race_completed, bool, this.race_completed);
98
99     METHOD(Client, m_unwind, bool(Client this));
100
101     STATIC_METHOD(Client, Add, void(Client this, int _team));
102     STATIC_METHOD(Client, Remove, void(Client this));
103
104     INIT(Client) {
105         if (this.m_unwind(this)) return this;
106         make_impure(this);
107         this.classname = "player_joining";
108         static int playerid_last;
109         this.playerid = ++playerid_last;
110         ClientState_attach(this);
111     }
112     DESTRUCTOR(Client) {
113         Client_Remove(this);
114     }
115     CONSTRUCTOR(Client, string name) {
116         CONSTRUCT(Client);
117         this.netname = name;
118         this.netaddress = "local";
119         this.playermodel = cvar_defstring("sv_defaultplayermodel");
120     }
121 ENDCLASS(Client)
122
123 CLASS(Observer, Client)
124     INIT(Observer) {
125         this.classname = STR_OBSERVER;
126     }
127     DESTRUCTOR(Observer) { }
128 ENDCLASS(Observer)
129
130 CLASS(Spectator, Client)
131     INIT(Spectator) {
132         this.classname = STR_SPECTATOR;
133     }
134     DESTRUCTOR(Spectator) { }
135 ENDCLASS(Spectator)
136
137 CLASS(Player, Client)
138     
139     // custom
140
141     ATTRIB(Player, dual_weapons, vector, this.dual_weapons); // TODO: actually WepSet!
142     ATTRIB(Player, itemkeys, int, this.itemkeys);
143
144     INIT(Player) {
145         this.classname = STR_PLAYER;
146         IL_PUSH(g_players, this);
147     }
148     DESTRUCTOR(Player) {
149         IL_REMOVE(g_players, this);
150     }
151 ENDCLASS(Player)
152
153 METHOD(Client, m_unwind, bool(Client this))
154 {
155     TC(Client, this);
156     #define UNWIND(class) MACRO_BEGIN if (this.instanceOf##class) { METHOD_REFERENCE(class, dtorimpl)(this); } MACRO_END
157     switch (this.classname) {
158         case "Observer":
159             UNWIND(Spectator);
160             UNWIND(Player);
161             return true;
162         case "Spectator":
163             UNWIND(Observer);
164             UNWIND(Player);
165             return true;
166         case "Player":
167             UNWIND(Observer);
168             UNWIND(Spectator);
169             return true;
170     }
171     #undef UNWIND
172     return false;
173 }
174
175 float c1, c2, c3, c4;
176
177 void play_countdown(entity this, float finished, Sound samp);
178
179 float CalcRotRegen(float current, float regenstable, float regenfactor, float regenlinear, float regenframetime, float rotstable, float rotfactor, float rotlinear, float rotframetime, float limit);
180
181 bool Spectate(entity this, entity pl);
182
183 #define SPECTATE_COPY() [[accumulate]] void SpectateCopy(entity this, entity spectatee)
184 #define SPECTATE_COPYFIELD(fld) SPECTATE_COPY() { this.(fld) = spectatee.(fld); }