Xonotic
getreplies.qc
Go to the documentation of this file.
1 #include "getreplies.qh"
2 
3 #include <common/command/_mod.qh>
4 #include <common/constants.qh>
7 #include <common/net_linked.qh>
9 #include <common/playerstats.qh>
10 #include <common/stats.qh>
11 #include <common/util.qh>
12 #include <common/weapons/_all.qh>
13 #include <common/wepent.qh>
15 #include <server/intermission.qh>
16 #include <server/main.qh>
17 #include <server/mapvoting.qh>
18 #include <server/mutators/_mod.qh>
19 #include <server/race.qh>
21 #include <server/world.qh>
22 
23 // =========================================================
24 // Reply messages for common commands, re-worked by Samual
25 // Last updated: December 30th, 2011
26 // =========================================================
27 
28 // These strings are set usually during init in world.qc,
29 // or also by some game modes or other functions manually,
30 // and their purpose is to output information to clients
31 // without using any extra processing time.
32 
33 // See common.qc for their proper commands
34 
35 string getrecords(int page)
36 {
37  string s = "";
38 
39  MUTATOR_CALLHOOK(GetRecords, page, s);
40  s = M_ARGV(1, string);
41 
43  return s;
44 }
45 
46 string getrankings()
47 {
48  float t, i;
49  string n, s, p, map;
50 
51  map = GetMapname();
52 
53  s = "";
54  for (i = 1; i <= RANKINGS_CNT; ++i)
55  {
56  t = race_readTime(map, i);
57 
58  if (t == 0) continue;
59 
60  n = race_readName(map, i);
61  p = count_ordinal(i);
62  s = strcat(s, strpad(8, p), " ", strpad(-8, TIME_ENCODED_TOSTRING(t)), " ", n, "\n");
63  }
64 
66 
67  if (s == "") return strcat("No records are available for the map: ", map, "\n");
68  else return strcat("Records for ", map, ":\n", s);
69 }
70 
71 string getladder()
72 {
73  int i, j, k, uidcnt = 0, thiscnt;
74  string s, temp_s, rr, myuid, thisuid;
75 
76  rr = (g_cts) ? CTS_RECORD : RACE_RECORD;
77 
78  for (k = 0; k < MapInfo_count; ++k)
79  {
80  if (MapInfo_Get_ByID(k))
81  {
82  for (i = 0; i <= LADDER_CNT; ++i) // i = 0 because it is the speed award
83  {
84  if (i == 0) // speed award
85  {
86  if (stof(db_get(ServerProgsDB, strcat(MapInfo_Map_bspname, rr, "speed/speed"))) == 0) continue;
87 
88  myuid = db_get(ServerProgsDB, strcat(MapInfo_Map_bspname, rr, "speed/crypto_idfp"));
89  }
90  else // normal record, if it exists (else break)
91  {
92  if (race_readTime(MapInfo_Map_bspname, i) == 0) continue;
93 
95  }
96 
97  // string s contains:
98  // arg 0 = # of speed recs
99  // arg 1 = # of 1st place recs
100  // arg 2 = # of 2nd place recs
101  // ... etc
102  // LADDER_CNT+1 = total points
103 
104  temp_s = db_get(TemporaryDB, strcat("ladder", myuid));
105 
106  if (temp_s == "")
107  {
108  db_put(TemporaryDB, strcat("uid", ftos(uidcnt)), myuid);
109  ++uidcnt;
110 
111  for (j = 0; j <= LADDER_CNT + 1; ++j)
112  {
113  if (j != LADDER_CNT + 1) temp_s = strcat(temp_s, "0 ");
114  else temp_s = strcat(temp_s, "0");
115  }
116  }
117 
118  tokenize_console(temp_s);
119  s = "";
120 
121  if (i == 0) // speed award
122  {
123  for (j = 0; j <= LADDER_CNT; ++j) // loop over each arg in the string
124  {
125  if (j == 0) // speed award
126  s = strcat(s, ftos(stof(argv(j)) + 1)); // add 1 to speed rec count and write
127  else s = strcat(s, " ", argv(j)); // just copy over everything else
128  }
129  }
130  else // record
131  {
132  for (j = 0; j <= LADDER_CNT; ++j) // loop over each arg in the string
133  {
134  if (j == 0) s = strcat(s, argv(j)); // speed award, dont prefix with " "
135  else if (j == i) // wanted rec!
136  s = strcat(s, " ", ftos(stof(argv(j)) + 1)); // update argv(j)
137  else s = strcat(s, " ", argv(j)); // just copy over everything else
138  }
139  }
140 
141  // total points are (by default) calculated like this:
142  // speedrec = floor(100 / 10) = 10 points
143  // 1st place = floor(100 / 1) = 100 points
144  // 2nd place = floor(100 / 2) = 50 points
145  // 3rd place = floor(100 / 3) = 33 points
146  // 4th place = floor(100 / 4) = 25 points
147  // 5th place = floor(100 / 5) = 20 points
148  // ... etc
149 
150  if (i == 0) s = strcat(s, " ", ftos(stof(argv(LADDER_CNT + 1)) + LADDER_FIRSTPOINT / 10)); // speed award, add LADDER_FIRSTPOINT / 10 points
151  else s = strcat(s, " ", ftos(stof(argv(LADDER_CNT + 1)) + floor(LADDER_FIRSTPOINT / i))); // record, add LADDER_FIRSTPOINT / i points
152 
153  db_put(TemporaryDB, strcat("ladder", myuid), s);
154  }
155  }
156  }
157 
158  for (i = 0; i <= uidcnt; ++i) // for each known uid
159  {
160  thisuid = db_get(TemporaryDB, strcat("uid", ftos(i)));
161  temp_s = db_get(TemporaryDB, strcat("ladder", thisuid));
162  tokenize_console(temp_s);
163  thiscnt = stof(argv(LADDER_CNT + 1));
164 
165  if (thiscnt > top_scores[LADDER_SIZE - 1])
166  {
167  for (j = 0; j < LADDER_SIZE; ++j) // for each place in ladder
168  {
169  if (thiscnt > top_scores[j])
170  {
171  for (k = LADDER_SIZE - 1; k >= j; --k)
172  {
173  top_uids[k] = top_uids[k - 1];
174  top_scores[k] = top_scores[k - 1];
175  }
176 
177  top_uids[j] = thisuid;
178  top_scores[j] = thiscnt;
179  break;
180  }
181  }
182  }
183  }
184 
185  s = "^3-----------------------\n\n";
186 
187  s = strcat(s, "Pos ^3|");
188  s = strcat(s, " ^7Total ^3|");
189 
190  for (i = 1; i <= LADDER_CNT; ++i)
191  s = strcat(s, " ^7", count_ordinal(i), " ^3|");
192  s = strcat(s, " ^7Speed awards ^3| ^7Name");
193  s = strcat(s, "\n^3----+--------");
194 
195  for (i = 1; i <= min(9, LADDER_CNT); ++i)
196  s = strcat(s, "+-----");
197 #if LADDER_CNT > 9
198  for (i = 1; i <= LADDER_CNT - 9; ++i)
199  s = strcat(s, "+------");
200 #endif
201 
202  s = strcat(s, "+--------------+--------------------\n");
203 
204  for (i = 0; i < LADDER_SIZE; ++i)
205  {
206  temp_s = db_get(TemporaryDB, strcat("ladder", top_uids[i]));
207  tokenize_console(temp_s);
208 
209  if (argv(LADDER_CNT + 1) == "") // total is 0, skip
210  continue;
211 
212  s = strcat(s, strpad(4, count_ordinal(i + 1)), "^3| ^7"); // pos
213  s = strcat(s, strpad(7, argv(LADDER_CNT + 1)), "^3| ^7"); // total
214 
215  for (j = 1; j <= min(9, LADDER_CNT); ++j)
216  s = strcat(s, strpad(4, argv(j)), "^3| ^7"); // 1st, 2nd, 3rd etc cnt
217 
218 #if LADDER_CNT > 9
219  for (j = 10; j <= LADDER_CNT; ++j)
220  s = strcat(s, strpad(4, argv(j)), " ^3| ^7"); // 1st, 2nd, 3rd etc cnt
221 #endif
222 
223  s = strcat(s, strpad(13, argv(0)), "^3| ^7"); // speed award cnt
224  s = strcat(s, uid2name(top_uids[i]), "\n"); // name
225  }
226 
228 
229  if (s == "") return "No ladder on this server!\n";
230  else return strcat("Top ", ftos(LADDER_SIZE), " ladder rankings:\n", s);
231 }
232 
233 string getmaplist()
234 {
235  string maplist = "", col;
236  int i, argc;
237 
239  for (i = 0; i < argc; ++i)
240  {
241  if (MapInfo_CheckMap(argv(i)))
242  {
243  if (i % 2) col = "^2"; else col = "^3";
244  maplist = sprintf("%s%s%s ", maplist, col, argv(i));
245  }
246  }
247 
249  return sprintf("^7Maps in list: %s\n", maplist);
250 }
251 
252 const int LSMAPS_MAX = 250;
253 string getlsmaps()
254 {
255  string lsmaps = "", col;
256  bool newmaps = false;
257  int added = 0;
258 
259  for (int i = 0; i < MapInfo_count; ++i)
260  {
262  {
263  ++added;
264 
265  if(added > LSMAPS_MAX)
266  continue; // we still get the added count, but skip the actual processing
267 
268  // todo: Check by play count of maps for other game types?
270  {
271  newmaps = true;
272  if (i % 2) col = "^4*"; else col = "^5*";
273  }
274  else
275  {
276  if (i % 2) col = "^2"; else col = "^3";
277  }
278 
279  lsmaps = sprintf("%s%s%s ", lsmaps, col, MapInfo_Map_bspname);
280  }
281  }
282 
283  if(added > LSMAPS_MAX)
284  lsmaps = sprintf("%s^7(%d not listed)", lsmaps, added - LSMAPS_MAX);
285 
287  return sprintf("^7Maps available (%d)%s: %s\n", added, (newmaps ? " (New maps have asterisks marked in blue)" : ""), lsmaps);
288 }
289 
291 {
292  string monsterlist = "";
293 
294  FOREACH(Monsters, it != MON_Null && !(it.spawnflags & MON_FLAG_HIDDEN),
295  {
296  string col = ((i % 2) ? "^2" : "^3");
297  monsterlist = sprintf("%s%s%s ", monsterlist, col, it.netname);
298  });
299 
300  return sprintf("^7Monsters available: %s\n", monsterlist);
301 }
302 
303 /*
304 =============
305 GetCvars
306 =============
307 Superseded by REPLICATE
308 Called with:
309  0: sends the request
310  >0: receives a cvar from name=argv(f) value=argv(f+1)
311 */
312 void GetCvars_handleString(entity this, entity store, string thisname, float f, .string field, string name)
313 {
314  if (f < 0)
315  {
316  strfree(store.(field));
317  }
318  else if (f > 0)
319  {
320  if (thisname == name)
321  {
322  strcpy(store.(field), argv(f + 1));
323  }
324  }
325  else
326  stuffcmd(this, strcat("cl_cmd sendcvar ", name, "\n"));
327 }
328 void GetCvars_handleString_Fixup(entity this, entity store, string thisname, float f, .string field, string name, string(entity, string) func)
329 {
330  GetCvars_handleString(this, store, thisname, f, field, name);
331  if (f >= 0) // also initialize to the fitting value for "" when sending cvars out
332  if (thisname == name)
333  {
334  string s = func(this, strcat1(store.(field)));
335  if (s != store.(field))
336  {
337  strcpy(store.(field), s);
338  }
339  }
340 }
341 void GetCvars_handleFloat(entity this, entity store, string thisname, float f, .float field, string name)
342 {
343  if (f < 0)
344  {
345  }
346  else if (f > 0)
347  {
348  if (thisname == name)
349  store.(field) = stof(argv(f + 1));
350  }
351  else
352  stuffcmd(this, strcat("cl_cmd sendcvar ", name, "\n"));
353 }
354 void GetCvars_handleFloatOnce(entity this, entity store, string thisname, float f, .float field, string name)
355 {
356  if (f < 0)
357  {
358  }
359  else if (f > 0)
360  {
361  if (thisname == name)
362  {
363  if (!store.(field))
364  {
365  store.(field) = stof(argv(f + 1));
366  if (!store.(field))
367  store.(field) = -1;
368  }
369  }
370  }
371  else
372  {
373  if (!store.(field))
374  stuffcmd(this, strcat("cl_cmd sendcvar ", name, "\n"));
375  }
376 }
377 
381 void GetCvars(entity this, entity store, int f)
382 {
383  string s = string_null;
384 
385  if (f == 0)
386  LOG_INFO("Warning: requesting cvar values is deprecated. Client should send them automatically using REPLICATE.\n");
387 
388  if (f > 0)
389  s = strcat1(argv(f));
390 
391  get_cvars_f = f;
392  get_cvars_s = s;
394 
395  Notification_GetCvars(this, store);
396 
397  ReplicateVars(this, store, s, f);
398  if (f > 0)
399  ReplicateVars_ApplyChange(this, store, s, f);
400 }
ERASEABLE void db_put(int db, string key, string value)
Definition: map.qh:101
const int MON_FLAG_HIDDEN
Definition: monster.qh:16
string string_null
Definition: nil.qh:9
string getrankings()
Definition: getreplies.qc:46
string uid2name(string myuid)
Definition: race.qc:36
string GetMapname()
Definition: intermission.qc:18
float ServerProgsDB
Definition: world.qh:131
entity() spawn
string race_readUID(string map, float pos)
Definition: race.qc:83
const int LSMAPS_MAX
Definition: getreplies.qc:252
#define RACE_RECORD
Definition: util.qh:58
#define LADDER_CNT
Definition: getreplies.qh:10
bool MapInfo_Get_ByID(int i)
Definition: mapinfo.qc:256
int MapInfo_ForbiddenFlags()
Definition: mapinfo.qc:1324
#define TIME_ENCODED_TOSTRING(n)
Definition: util.qh:57
int MapInfo_Map_flags
Definition: mapinfo.qh:15
#define CTS_RECORD
Definition: util.qh:59
#define strcpy(this, s)
Definition: string.qh:49
ERASEABLE string count_ordinal(int interval)
Definition: counting.qh:66
float TemporaryDB
Definition: world.qh:132
float MapInfo_count
Definition: mapinfo.qh:144
#define autocvar_g_maplist
Definition: mapvoting.qh:3
#define g_cts
Definition: cts.qh:36
#define g_race
Definition: race.qh:46
spree_cen s1 spree_cen s1 spree_cen s1 spree_cen s1 spree_cen s1 spree_cen s1 spree_cen s1 f1 s1 strcat(_("Level %s: "), "^BG%s\3\, _("^BGPress ^F2%s^BG to enter the game"))
void GetCvars(entity this, entity store, int f)
Definition: getreplies.qc:381
string get_cvars_s
Definition: events.qh:336
float race_readTime(string map, float pos)
Definition: race.qc:78
string getmaplist()
Definition: getreplies.qc:233
string getladder()
Definition: getreplies.qc:71
#define LOG_INFO(...)
Definition: log.qh:70
float get_cvars_f
Definition: events.qh:335
string getrecords(int page)
Definition: getreplies.qc:35
ERASEABLE string db_get(int db, string key)
Definition: map.qh:91
string getmonsterlist()
Definition: getreplies.qc:290
void GetCvars_handleString_Fixup(entity this, entity store, string thisname, float f,.string field, string name, string(entity, string) func)
Definition: getreplies.qc:328
float MapInfo_CheckMap(string s)
Definition: mapinfo.qc:1170
#define M_ARGV(x, type)
Definition: events.qh:17
void MapInfo_ClearTemps()
Definition: mapinfo.qc:1299
#define stuffcmd(cl,...)
Definition: progsdefs.qh:23
#define tokenize_console
Definition: dpextensions.qh:24
string race_readName(string map, float pos)
Definition: race.qc:138
void GetCvars_handleFloat(entity this, entity store, string thisname, float f,.float field, string name)
Definition: getreplies.qc:341
void GetCvars_handleFloatOnce(entity this, entity store, string thisname, float f,.float field, string name)
Definition: getreplies.qc:354
string MapInfo_Map_bspname
Definition: mapinfo.qh:6
const int LADDER_FIRSTPOINT
Definition: getreplies.qh:9
void GetCvars_handleString(entity this, entity store, string thisname, float f,.string field, string name)
Definition: getreplies.qc:312
#define MUTATOR_CALLHOOK(id,...)
Definition: base.qh:140
#define strfree(this)
Definition: string.qh:56
const int LADDER_SIZE
Definition: getreplies.qh:11
string top_uids[LADDER_SIZE]
Definition: getreplies.qh:13
float top_scores[LADDER_SIZE]
Definition: getreplies.qh:14
string getlsmaps()
Definition: getreplies.qc:253
#define FOREACH(list, cond, body)
Definition: iter.qh:19