1 #include "scripting.qh"
7 .float bot_cmdqueuebuf_allocated;
8 .float bot_cmdqueuebuf;
9 .float bot_cmdqueuebuf_start;
10 .float bot_cmdqueuebuf_end;
12 void bot_clearqueue(entity bot)
14 if(!bot.bot_cmdqueuebuf_allocated)
16 buf_del(bot.bot_cmdqueuebuf);
17 bot.bot_cmdqueuebuf_allocated = false;
18 LOG_TRACE("bot ", bot.netname, " queue cleared\n");
21 void bot_queuecommand(entity bot, string cmdstring)
23 if(!bot.bot_cmdqueuebuf_allocated)
25 bot.bot_cmdqueuebuf = buf_create();
26 bot.bot_cmdqueuebuf_allocated = true;
27 bot.bot_cmdqueuebuf_start = 0;
28 bot.bot_cmdqueuebuf_end = 0;
31 bufstr_set(bot.bot_cmdqueuebuf, bot.bot_cmdqueuebuf_end, cmdstring);
33 // if the command was a "sound" command, precache the sound NOW
34 // this prevents lagging!
40 sp = strstrofs(cmdstring, " ", 0);
43 parm = substring(cmdstring, sp + 1, -1);
44 cmdstr = substring(cmdstring, 0, sp);
50 sp = strstrofs(parm, " ", 0);
53 parm = substring(parm, sp + 1, -1);
60 bot.bot_cmdqueuebuf_end += 1;
63 void bot_dequeuecommand(entity bot, float idx)
65 if(!bot.bot_cmdqueuebuf_allocated)
66 error("dequeuecommand but no queue allocated");
67 if(idx < bot.bot_cmdqueuebuf_start)
68 error("dequeueing a command in the past");
69 if(idx >= bot.bot_cmdqueuebuf_end)
70 error("dequeueing a command in the future");
71 bufstr_set(bot.bot_cmdqueuebuf, idx, "");
72 if(idx == bot.bot_cmdqueuebuf_start)
73 bot.bot_cmdqueuebuf_start += 1;
74 if(bot.bot_cmdqueuebuf_start >= bot.bot_cmdqueuebuf_end)
78 string bot_readcommand(entity bot, float idx)
80 if(!bot.bot_cmdqueuebuf_allocated)
81 error("readcommand but no queue allocated");
82 if(idx < bot.bot_cmdqueuebuf_start)
83 error("reading a command in the past");
84 if(idx >= bot.bot_cmdqueuebuf_end)
85 error("reading a command in the future");
86 return bufstr_get(bot.bot_cmdqueuebuf, idx);
89 float bot_havecommand(entity bot, float idx)
91 if(!bot.bot_cmdqueuebuf_allocated)
93 if(idx < bot.bot_cmdqueuebuf_start)
95 if(idx >= bot.bot_cmdqueuebuf_end)
100 const int MAX_BOT_PLACES = 4;
101 .float bot_places_count;
102 .entity bot_places[MAX_BOT_PLACES];
103 .string bot_placenames[MAX_BOT_PLACES];
104 entity bot_getplace(string placename)
107 if(substring(placename, 0, 1) == "@")
110 placename = substring(placename, 1, -1);
112 for(i = 0; i < self.bot_places_count; ++i)
113 if(self.(bot_placenames[i]) == placename)
114 return self.(bot_places[i]);
115 // now: i == self.bot_places_count
116 s = s2 = cvar_string(placename);
117 p = strstrofs(s2, " ", 0);
120 s = substring(s2, 0, p);
121 //print("places: ", placename, " -> ", cvar_string(placename), "\n");
122 cvar_set(placename, strcat(substring(s2, p+1, -1), " ", s));
123 //print("places: ", placename, " := ", cvar_string(placename), "\n");
125 e = find(world, targetname, s);
127 LOG_INFO("invalid place ", s, "\n");
128 if(i < MAX_BOT_PLACES)
130 self.(bot_placenames[i]) = strzone(placename);
131 self.(bot_places[i]) = e;
132 self.bot_places_count += 1;
138 e = find(world, targetname, placename);
140 LOG_INFO("invalid place ", placename, "\n");
146 // Initialize global commands list
147 // NOTE: New commands should be initialized here
148 void bot_commands_init()
150 bot_cmd_string[BOT_CMD_NULL] = "";
151 bot_cmd_parm_type[BOT_CMD_NULL] = BOT_CMD_PARAMETER_NONE;
153 bot_cmd_string[BOT_CMD_PAUSE] = "pause";
154 bot_cmd_parm_type[BOT_CMD_PAUSE] = BOT_CMD_PARAMETER_NONE;
156 bot_cmd_string[BOT_CMD_CONTINUE] = "continue";
157 bot_cmd_parm_type[BOT_CMD_CONTINUE] = BOT_CMD_PARAMETER_NONE;
159 bot_cmd_string[BOT_CMD_WAIT] = "wait";
160 bot_cmd_parm_type[BOT_CMD_WAIT] = BOT_CMD_PARAMETER_FLOAT;
162 bot_cmd_string[BOT_CMD_TURN] = "turn";
163 bot_cmd_parm_type[BOT_CMD_TURN] = BOT_CMD_PARAMETER_FLOAT;
165 bot_cmd_string[BOT_CMD_MOVETO] = "moveto";
166 bot_cmd_parm_type[BOT_CMD_MOVETO] = BOT_CMD_PARAMETER_VECTOR;
168 bot_cmd_string[BOT_CMD_MOVETOTARGET] = "movetotarget";
169 bot_cmd_parm_type[BOT_CMD_MOVETOTARGET] = BOT_CMD_PARAMETER_STRING;
171 bot_cmd_string[BOT_CMD_RESETGOAL] = "resetgoal";
172 bot_cmd_parm_type[BOT_CMD_RESETGOAL] = BOT_CMD_PARAMETER_NONE;
174 bot_cmd_string[BOT_CMD_CC] = "cc";
175 bot_cmd_parm_type[BOT_CMD_CC] = BOT_CMD_PARAMETER_STRING;
177 bot_cmd_string[BOT_CMD_IF] = "if";
178 bot_cmd_parm_type[BOT_CMD_IF] = BOT_CMD_PARAMETER_STRING;
180 bot_cmd_string[BOT_CMD_ELSE] = "else";
181 bot_cmd_parm_type[BOT_CMD_ELSE] = BOT_CMD_PARAMETER_NONE;
183 bot_cmd_string[BOT_CMD_FI] = "fi";
184 bot_cmd_parm_type[BOT_CMD_FI] = BOT_CMD_PARAMETER_NONE;
186 bot_cmd_string[BOT_CMD_RESETAIM] = "resetaim";
187 bot_cmd_parm_type[BOT_CMD_RESETAIM] = BOT_CMD_PARAMETER_NONE;
189 bot_cmd_string[BOT_CMD_AIM] = "aim";
190 bot_cmd_parm_type[BOT_CMD_AIM] = BOT_CMD_PARAMETER_STRING;
192 bot_cmd_string[BOT_CMD_AIMTARGET] = "aimtarget";
193 bot_cmd_parm_type[BOT_CMD_AIMTARGET] = BOT_CMD_PARAMETER_STRING;
195 bot_cmd_string[BOT_CMD_PRESSKEY] = "presskey";
196 bot_cmd_parm_type[BOT_CMD_PRESSKEY] = BOT_CMD_PARAMETER_STRING;
198 bot_cmd_string[BOT_CMD_RELEASEKEY] = "releasekey";
199 bot_cmd_parm_type[BOT_CMD_RELEASEKEY] = BOT_CMD_PARAMETER_STRING;
201 bot_cmd_string[BOT_CMD_SELECTWEAPON] = "selectweapon";
202 bot_cmd_parm_type[BOT_CMD_SELECTWEAPON] = BOT_CMD_PARAMETER_FLOAT;
204 bot_cmd_string[BOT_CMD_IMPULSE] = "impulse";
205 bot_cmd_parm_type[BOT_CMD_IMPULSE] = BOT_CMD_PARAMETER_FLOAT;
207 bot_cmd_string[BOT_CMD_WAIT_UNTIL] = "wait_until";
208 bot_cmd_parm_type[BOT_CMD_WAIT_UNTIL] = BOT_CMD_PARAMETER_FLOAT;
210 bot_cmd_string[BOT_CMD_BARRIER] = "barrier";
211 bot_cmd_parm_type[BOT_CMD_BARRIER] = BOT_CMD_PARAMETER_NONE;
213 bot_cmd_string[BOT_CMD_CONSOLE] = "console";
214 bot_cmd_parm_type[BOT_CMD_CONSOLE] = BOT_CMD_PARAMETER_STRING;
216 bot_cmd_string[BOT_CMD_SOUND] = "sound";
217 bot_cmd_parm_type[BOT_CMD_SOUND] = BOT_CMD_PARAMETER_STRING;
219 bot_cmd_string[BOT_CMD_DEBUG_ASSERT_CANFIRE] = "debug_assert_canfire";
220 bot_cmd_parm_type[BOT_CMD_DEBUG_ASSERT_CANFIRE] = BOT_CMD_PARAMETER_FLOAT;
222 bot_cmds_initialized = true;
225 // Returns first bot with matching name
226 entity find_bot_by_name(string name)
230 bot = findchainflags(flags, FL_CLIENT);
233 if(IS_BOT_CLIENT(bot))
234 if(bot.netname==name)
243 // Returns a bot by number on list
244 entity find_bot_by_number(float number)
252 bot = findchainflags(flags, FL_CLIENT);
255 if(IS_BOT_CLIENT(bot))
266 float bot_decodecommand(string cmdstring)
272 sp = strstrofs(cmdstring, " ", 0);
279 parm = substring(cmdstring, sp + 1, -1);
280 cmdstring = substring(cmdstring, 0, sp);
283 if(!bot_cmds_initialized)
287 for(i=1;i<BOT_CMD_COUNTER;++i)
289 if(bot_cmd_string[i]!=cmdstring)
292 cmd_parm_type = bot_cmd_parm_type[i];
294 if(cmd_parm_type!=BOT_CMD_PARAMETER_NONE&&parm=="")
296 LOG_INFO("ERROR: A parameter is required for this command\n");
300 // Load command into queue
301 bot_cmd.bot_cmd_type = i;
304 switch(cmd_parm_type)
306 case BOT_CMD_PARAMETER_FLOAT:
307 bot_cmd.bot_cmd_parm_float = stof(parm);
309 case BOT_CMD_PARAMETER_STRING:
310 if(bot_cmd.bot_cmd_parm_string)
311 strunzone(bot_cmd.bot_cmd_parm_string);
312 bot_cmd.bot_cmd_parm_string = strzone(parm);
314 case BOT_CMD_PARAMETER_VECTOR:
315 bot_cmd.bot_cmd_parm_vector = stov(parm);
322 LOG_INFO("ERROR: No such command '", cmdstring, "'\n");
326 void bot_cmdhelp(string scmd)
331 if(!bot_cmds_initialized)
334 for(i=1;i<BOT_CMD_COUNTER;++i)
336 if(bot_cmd_string[i]!=scmd)
339 ntype = bot_cmd_parm_type[i];
343 case BOT_CMD_PARAMETER_FLOAT:
344 stype = "float number";
346 case BOT_CMD_PARAMETER_STRING:
349 case BOT_CMD_PARAMETER_VECTOR:
357 LOG_INFO(strcat("Command: ",bot_cmd_string[i],"\nParameter: <",stype,"> \n"));
359 LOG_INFO("Description: ");
363 LOG_INFO("Stops the bot completely. Any command other than 'continue' will be ignored.");
365 case BOT_CMD_CONTINUE:
366 LOG_INFO("Disable paused status");
369 LOG_INFO("Pause command parsing and bot ai for N seconds. Pressed key will remain pressed");
371 case BOT_CMD_WAIT_UNTIL:
372 LOG_INFO("Pause command parsing and bot ai until time is N from the last barrier. Pressed key will remain pressed");
374 case BOT_CMD_BARRIER:
375 LOG_INFO("Waits till all bots that have a command queue reach this command. Pressed key will remain pressed");
378 LOG_INFO("Look to the right or left N degrees. For turning to the left use positive numbers.");
381 LOG_INFO("Walk to an specific coordinate on the map. Usage: moveto \"x y z\"");
383 case BOT_CMD_MOVETOTARGET:
384 LOG_INFO("Walk to the specific target on the map");
386 case BOT_CMD_RESETGOAL:
387 LOG_INFO("Resets the goal stack");
390 LOG_INFO("Execute client command. Examples: cc \"say something\"; cc god; cc \"name newnickname\"; cc kill;");
393 LOG_INFO("Perform simple conditional execution.\n");
394 LOG_INFO("Syntax: \n");
395 LOG_INFO(" sv_cmd .. if \"condition\"\n");
396 LOG_INFO(" sv_cmd .. <instruction if true>\n");
397 LOG_INFO(" sv_cmd .. <instruction if true>\n");
398 LOG_INFO(" sv_cmd .. else\n");
399 LOG_INFO(" sv_cmd .. <instruction if false>\n");
400 LOG_INFO(" sv_cmd .. <instruction if false>\n");
401 LOG_INFO(" sv_cmd .. fi\n");
402 LOG_INFO("Conditions: a=b, a>b, a<b, a\t\t(spaces not allowed)\n");
403 LOG_INFO(" Values in conditions can be numbers, cvars in the form cvar.cvar_string or special fields\n");
404 LOG_INFO("Fields: health, speed, flagcarrier\n");
405 LOG_INFO("Examples: if health>50; if health>cvar.g_balance_laser_primary_damage; if flagcarrier;");
407 case BOT_CMD_RESETAIM:
408 LOG_INFO("Points the aim to the coordinates x,y 0,0");
411 LOG_INFO("Move the aim x/y (horizontal/vertical) degrees relatives to the bot\n");
412 LOG_INFO("There is a 3rd optional parameter telling in how many seconds the aim has to reach the new position\n");
413 LOG_INFO("Examples: aim \"90 0\" // Turn 90 degrees inmediately (positive numbers move to the left/up)\n");
414 LOG_INFO(" aim \"0 90 2\" // Will gradually look to the sky in the next two seconds");
416 case BOT_CMD_AIMTARGET:
417 LOG_INFO("Points the aim to given target");
419 case BOT_CMD_PRESSKEY:
420 LOG_INFO("Press one of the following keys: forward, backward, left, right, jump, crouch, attack1, attack2, use\n");
421 LOG_INFO("Multiple keys can be pressed at time (with many presskey calls) and it will remain pressed until the command \"releasekey\" is called");
422 LOG_INFO("Note: The script will not return the control to the bot ai until all keys are released");
424 case BOT_CMD_RELEASEKEY:
425 LOG_INFO("Release previoulsy used keys. Use the parameter \"all\" to release all keys");
428 LOG_INFO("play sound file at bot location");
430 case BOT_CMD_DEBUG_ASSERT_CANFIRE:
431 LOG_INFO("verify the state of the weapon entity");
434 LOG_INFO("This command has no description yet.");
441 void bot_list_commands()
446 if(!bot_cmds_initialized)
449 LOG_INFO("List of all available commands:\n");
450 LOG_INFO(" Command - Parameter Type\n");
452 for(i=1;i<BOT_CMD_COUNTER;++i)
454 switch(bot_cmd_parm_type[i])
456 case BOT_CMD_PARAMETER_FLOAT:
457 ptype = "float number";
459 case BOT_CMD_PARAMETER_STRING:
462 case BOT_CMD_PARAMETER_VECTOR:
469 LOG_INFO(strcat(" ",bot_cmd_string[i]," - <",ptype,"> \n"));
474 .int bot_exec_status;
476 void SV_ParseClientCommand(string s);
479 SV_ParseClientCommand(bot_cmd.bot_cmd_parm_string);
480 return CMD_STATUS_FINISHED;
483 float bot_cmd_impulse()
485 self.impulse = bot_cmd.bot_cmd_parm_float;
486 return CMD_STATUS_FINISHED;
489 float bot_cmd_continue()
491 self.bot_exec_status &= ~BOT_EXEC_STATUS_PAUSED;
492 return CMD_STATUS_FINISHED;
495 .float bot_cmd_wait_time;
498 if(self.bot_exec_status & BOT_EXEC_STATUS_WAITING)
500 if(time>=self.bot_cmd_wait_time)
502 self.bot_exec_status &= ~BOT_EXEC_STATUS_WAITING;
503 return CMD_STATUS_FINISHED;
506 return CMD_STATUS_EXECUTING;
509 self.bot_cmd_wait_time = time + bot_cmd.bot_cmd_parm_float;
510 self.bot_exec_status |= BOT_EXEC_STATUS_WAITING;
511 return CMD_STATUS_EXECUTING;
514 float bot_cmd_wait_until()
516 if(time < bot_cmd.bot_cmd_parm_float + bot_barriertime)
518 self.bot_exec_status |= BOT_EXEC_STATUS_WAITING;
519 return CMD_STATUS_EXECUTING;
521 self.bot_exec_status &= ~BOT_EXEC_STATUS_WAITING;
522 return CMD_STATUS_FINISHED;
525 float bot_cmd_barrier()
527 // 0 = no barrier, 1 = waiting, 2 = waiting finished
529 if(self.bot_barrier == 0) // initialization
531 self.bot_barrier = 1;
533 //self.colormod = '4 4 0';
536 if(self.bot_barrier == 1) // find other bots
538 FOREACH_CLIENT(it.isbot, LAMBDA(
539 if(it.bot_cmdqueuebuf_allocated)
540 if(it.bot_barrier != 1)
541 return CMD_STATUS_EXECUTING; // not all are at the barrier yet
544 // all bots hit the barrier!
546 // acknowledge barrier
547 FOREACH_CLIENT(it.isbot, LAMBDA(it.bot_barrier = 2));
549 bot_barriertime = time;
552 // if we get here, the barrier is finished
554 self.bot_barrier = 0;
555 //self.colormod = '0 0 0';
557 return CMD_STATUS_FINISHED;
562 self.v_angle_y = self.v_angle.y + bot_cmd.bot_cmd_parm_float;
563 self.v_angle_y = self.v_angle.y - floor(self.v_angle.y / 360) * 360;
564 return CMD_STATUS_FINISHED;
567 float bot_cmd_select_weapon()
569 float id = bot_cmd.bot_cmd_parm_float;
571 if(id < WEP_FIRST || id > WEP_LAST)
572 return CMD_STATUS_ERROR;
574 if(client_hasweapon(self, Weapons_from(id), true, false))
575 PS(self).m_switchweapon = Weapons_from(id);
577 return CMD_STATUS_ERROR;
579 return CMD_STATUS_FINISHED;
582 .int bot_cmd_condition_status;
584 const int CMD_CONDITION_NONE = 0;
585 const int CMD_CONDITION_true = 1;
586 const int CMD_CONDITION_false = 2;
587 const int CMD_CONDITION_true_BLOCK = 4;
588 const int CMD_CONDITION_false_BLOCK = 8;
590 float bot_cmd_eval(string expr)
592 // Search for numbers
593 if(strstrofs("0123456789", substring(expr, 0, 1), 0) >= 0)
599 if(substring(expr, 0, 5)=="cvar.")
601 return cvar(substring(expr, 5, strlen(expr)));
610 return vlen(self.velocity);
612 return ((self.flagcarried!=world));
615 LOG_INFO(strcat("ERROR: Unable to convert the expression '",expr,"' into a numeric value\n"));
621 string expr, val_a, val_b;
624 if(self.bot_cmd_condition_status != CMD_CONDITION_NONE)
626 // Only one "if" block is allowed at time
627 LOG_INFO("ERROR: Only one conditional block can be processed at time");
628 bot_clearqueue(self);
629 return CMD_STATUS_ERROR;
632 self.bot_cmd_condition_status |= CMD_CONDITION_true_BLOCK;
634 // search for operators
635 expr = bot_cmd.bot_cmd_parm_string;
637 cmpofs = strstrofs(expr,"=",0);
641 val_a = substring(expr,0,cmpofs);
642 val_b = substring(expr,cmpofs+1,strlen(expr));
644 if(bot_cmd_eval(val_a)==bot_cmd_eval(val_b))
645 self.bot_cmd_condition_status |= CMD_CONDITION_true;
647 self.bot_cmd_condition_status |= CMD_CONDITION_false;
649 return CMD_STATUS_FINISHED;
652 cmpofs = strstrofs(expr,">",0);
656 val_a = substring(expr,0,cmpofs);
657 val_b = substring(expr,cmpofs+1,strlen(expr));
659 if(bot_cmd_eval(val_a)>bot_cmd_eval(val_b))
660 self.bot_cmd_condition_status |= CMD_CONDITION_true;
662 self.bot_cmd_condition_status |= CMD_CONDITION_false;
664 return CMD_STATUS_FINISHED;
667 cmpofs = strstrofs(expr,"<",0);
671 val_a = substring(expr,0,cmpofs);
672 val_b = substring(expr,cmpofs+1,strlen(expr));
674 if(bot_cmd_eval(val_a)<bot_cmd_eval(val_b))
675 self.bot_cmd_condition_status |= CMD_CONDITION_true;
677 self.bot_cmd_condition_status |= CMD_CONDITION_false;
679 return CMD_STATUS_FINISHED;
682 if(bot_cmd_eval(expr))
683 self.bot_cmd_condition_status |= CMD_CONDITION_true;
685 self.bot_cmd_condition_status |= CMD_CONDITION_false;
687 return CMD_STATUS_FINISHED;
692 self.bot_cmd_condition_status &= ~CMD_CONDITION_true_BLOCK;
693 self.bot_cmd_condition_status |= CMD_CONDITION_false_BLOCK;
694 return CMD_STATUS_FINISHED;
699 self.bot_cmd_condition_status = CMD_CONDITION_NONE;
700 return CMD_STATUS_FINISHED;
703 float bot_cmd_resetaim()
705 self.v_angle = '0 0 0';
706 return CMD_STATUS_FINISHED;
709 .float bot_cmd_aim_begintime;
710 .float bot_cmd_aim_endtime;
711 .vector bot_cmd_aim_begin;
712 .vector bot_cmd_aim_end;
717 if(self.bot_cmd_aim_endtime)
721 progress = min(1 - (self.bot_cmd_aim_endtime - time) / (self.bot_cmd_aim_endtime - self.bot_cmd_aim_begintime),1);
722 self.v_angle = self.bot_cmd_aim_begin + ((self.bot_cmd_aim_end - self.bot_cmd_aim_begin) * progress);
724 if(time>=self.bot_cmd_aim_endtime)
726 self.bot_cmd_aim_endtime = 0;
727 return CMD_STATUS_FINISHED;
730 return CMD_STATUS_EXECUTING;
733 // New aiming direction
737 parms = bot_cmd.bot_cmd_parm_string;
739 tokens = tokenizebyseparator(parms, " ");
741 if(tokens<2||tokens>3)
742 return CMD_STATUS_ERROR;
744 step = (tokens == 3) ? stof(argv(2)) : 0;
748 self.v_angle_x -= stof(argv(1));
749 self.v_angle_y += stof(argv(0));
750 return CMD_STATUS_FINISHED;
753 self.bot_cmd_aim_begin = self.v_angle;
755 self.bot_cmd_aim_end_x = self.v_angle.x - stof(argv(1));
756 self.bot_cmd_aim_end_y = self.v_angle.y + stof(argv(0));
757 self.bot_cmd_aim_end_z = 0;
759 self.bot_cmd_aim_begintime = time;
760 self.bot_cmd_aim_endtime = time + step;
762 return CMD_STATUS_EXECUTING;
765 float bot_cmd_aimtarget()
767 if(self.bot_cmd_aim_endtime)
769 return bot_cmd_aim();
777 parms = bot_cmd.bot_cmd_parm_string;
779 tokens = tokenizebyseparator(parms, " ");
781 e = bot_getplace(argv(0));
783 return CMD_STATUS_ERROR;
785 v = e.origin + (e.mins + e.maxs) * 0.5;
789 self.v_angle = vectoangles(v - (self.origin + self.view_ofs));
790 self.v_angle_x = -self.v_angle.x;
791 return CMD_STATUS_FINISHED;
794 if(tokens<1||tokens>2)
795 return CMD_STATUS_ERROR;
797 step = stof(argv(1));
799 self.bot_cmd_aim_begin = self.v_angle;
800 self.bot_cmd_aim_end = vectoangles(v - (self.origin + self.view_ofs));
801 self.bot_cmd_aim_end_x = -self.bot_cmd_aim_end.x;
803 self.bot_cmd_aim_begintime = time;
804 self.bot_cmd_aim_endtime = time + step;
806 return CMD_STATUS_EXECUTING;
811 const int BOT_CMD_KEY_NONE = 0;
812 const int BOT_CMD_KEY_FORWARD = BIT(0);
813 const int BOT_CMD_KEY_BACKWARD = BIT(1);
814 const int BOT_CMD_KEY_RIGHT = BIT(2);
815 const int BOT_CMD_KEY_LEFT = BIT(3);
816 const int BOT_CMD_KEY_JUMP = BIT(4);
817 const int BOT_CMD_KEY_ATTACK1 = BIT(5);
818 const int BOT_CMD_KEY_ATTACK2 = BIT(6);
819 const int BOT_CMD_KEY_USE = BIT(7);
820 const int BOT_CMD_KEY_HOOK = BIT(8);
821 const int BOT_CMD_KEY_CROUCH = BIT(9);
822 const int BOT_CMD_KEY_CHAT = BIT(10);
824 float bot_presskeys()
826 self.movement = '0 0 0';
827 self.BUTTON_JUMP = false;
828 self.BUTTON_CROUCH = false;
829 self.BUTTON_ATCK = false;
830 self.BUTTON_ATCK2 = false;
831 self.BUTTON_USE = false;
832 self.BUTTON_HOOK = false;
833 self.BUTTON_CHAT = false;
835 if(self.bot_cmd_keys == BOT_CMD_KEY_NONE)
838 if(self.bot_cmd_keys & BOT_CMD_KEY_FORWARD)
839 self.movement_x = autocvar_sv_maxspeed;
840 else if(self.bot_cmd_keys & BOT_CMD_KEY_BACKWARD)
841 self.movement_x = -autocvar_sv_maxspeed;
843 if(self.bot_cmd_keys & BOT_CMD_KEY_RIGHT)
844 self.movement_y = autocvar_sv_maxspeed;
845 else if(self.bot_cmd_keys & BOT_CMD_KEY_LEFT)
846 self.movement_y = -autocvar_sv_maxspeed;
848 if(self.bot_cmd_keys & BOT_CMD_KEY_JUMP)
849 self.BUTTON_JUMP = true;
851 if(self.bot_cmd_keys & BOT_CMD_KEY_CROUCH)
852 self.BUTTON_CROUCH = true;
854 if(self.bot_cmd_keys & BOT_CMD_KEY_ATTACK1)
855 self.BUTTON_ATCK = true;
857 if(self.bot_cmd_keys & BOT_CMD_KEY_ATTACK2)
858 self.BUTTON_ATCK2 = true;
860 if(self.bot_cmd_keys & BOT_CMD_KEY_USE)
861 self.BUTTON_USE = true;
863 if(self.bot_cmd_keys & BOT_CMD_KEY_HOOK)
864 self.BUTTON_HOOK = true;
866 if(self.bot_cmd_keys & BOT_CMD_KEY_CHAT)
867 self.BUTTON_CHAT = true;
873 float bot_cmd_keypress_handler(string key, float enabled)
879 self.bot_cmd_keys = power2of(20) - 1; // >:)
881 self.bot_cmd_keys = BOT_CMD_KEY_NONE;
885 self.bot_cmd_keys |= BOT_CMD_KEY_FORWARD;
886 self.bot_cmd_keys &= ~BOT_CMD_KEY_BACKWARD;
889 self.bot_cmd_keys &= ~BOT_CMD_KEY_FORWARD;
894 self.bot_cmd_keys |= BOT_CMD_KEY_BACKWARD;
895 self.bot_cmd_keys &= ~BOT_CMD_KEY_FORWARD;
898 self.bot_cmd_keys &= ~BOT_CMD_KEY_BACKWARD;
903 self.bot_cmd_keys |= BOT_CMD_KEY_LEFT;
904 self.bot_cmd_keys &= ~BOT_CMD_KEY_RIGHT;
907 self.bot_cmd_keys &= ~BOT_CMD_KEY_LEFT;
912 self.bot_cmd_keys |= BOT_CMD_KEY_RIGHT;
913 self.bot_cmd_keys &= ~BOT_CMD_KEY_LEFT;
916 self.bot_cmd_keys &= ~BOT_CMD_KEY_RIGHT;
920 self.bot_cmd_keys |= BOT_CMD_KEY_JUMP;
922 self.bot_cmd_keys &= ~BOT_CMD_KEY_JUMP;
926 self.bot_cmd_keys |= BOT_CMD_KEY_CROUCH;
928 self.bot_cmd_keys &= ~BOT_CMD_KEY_CROUCH;
932 self.bot_cmd_keys |= BOT_CMD_KEY_ATTACK1;
934 self.bot_cmd_keys &= ~BOT_CMD_KEY_ATTACK1;
938 self.bot_cmd_keys |= BOT_CMD_KEY_ATTACK2;
940 self.bot_cmd_keys &= ~BOT_CMD_KEY_ATTACK2;
944 self.bot_cmd_keys |= BOT_CMD_KEY_USE;
946 self.bot_cmd_keys &= ~BOT_CMD_KEY_USE;
950 self.bot_cmd_keys |= BOT_CMD_KEY_HOOK;
952 self.bot_cmd_keys &= ~BOT_CMD_KEY_HOOK;
956 self.bot_cmd_keys |= BOT_CMD_KEY_CHAT;
958 self.bot_cmd_keys &= ~BOT_CMD_KEY_CHAT;
964 return CMD_STATUS_FINISHED;
967 float bot_cmd_presskey()
971 key = bot_cmd.bot_cmd_parm_string;
973 bot_cmd_keypress_handler(key,true);
975 return CMD_STATUS_FINISHED;
978 float bot_cmd_releasekey()
982 key = bot_cmd.bot_cmd_parm_string;
984 return bot_cmd_keypress_handler(key,false);
987 float bot_cmd_pause()
992 self.BUTTON_ATCK = 0;
993 self.BUTTON_JUMP = 0;
994 self.BUTTON_HOOK = 0;
995 self.BUTTON_CHAT = 0;
996 self.BUTTON_ATCK2 = 0;
997 self.BUTTON_CROUCH = 0;
999 self.movement = '0 0 0';
1000 self.bot_cmd_keys = BOT_CMD_KEY_NONE;
1002 self.bot_exec_status |= BOT_EXEC_STATUS_PAUSED;
1003 return CMD_STATUS_FINISHED;
1006 float bot_cmd_moveto()
1008 return self.cmd_moveto(bot_cmd.bot_cmd_parm_vector);
1011 float bot_cmd_movetotarget()
1014 e = bot_getplace(bot_cmd.bot_cmd_parm_string);
1016 return CMD_STATUS_ERROR;
1017 return self.cmd_moveto(e.origin + (e.mins + e.maxs) * 0.5);
1020 float bot_cmd_resetgoal()
1022 return self.cmd_resetgoal();
1026 float bot_cmd_sound()
1029 f = bot_cmd.bot_cmd_parm_string;
1031 float n = tokenizebyseparator(f, " ");
1034 float chan = CH_WEAPON_B;
1035 float vol = VOL_BASE;
1036 float atten = ATTEN_MIN;
1039 sample = argv(n - 1);
1041 chan = stof(argv(0));
1043 vol = stof(argv(1));
1045 atten = stof(argv(2));
1048 _sound(self, chan, sample, vol, atten);
1050 return CMD_STATUS_FINISHED;
1054 float bot_cmd_debug_assert_canfire()
1056 float f = bot_cmd.bot_cmd_parm_float;
1059 .entity weaponentity = weaponentities[slot];
1060 if(self.(weaponentity).state != WS_READY)
1064 self.colormod = '0 8 8';
1065 LOG_INFO("Bot ", self.netname, " using ", self.weaponname, " wants to fire, inhibited by weaponentity state\n");
1068 else if(ATTACK_FINISHED(self, slot) > time)
1072 self.colormod = '8 0 8';
1073 LOG_INFO("Bot ", self.netname, " using ", self.weaponname, " wants to fire, inhibited by ATTACK_FINISHED (", ftos(ATTACK_FINISHED(self, slot) - time), " seconds left)\n");
1076 else if(self.tuba_note)
1080 self.colormod = '8 0 0';
1081 LOG_INFO("Bot ", self.netname, " using ", self.weaponname, " wants to fire, bot still has an active tuba note\n");
1088 self.colormod = '8 8 0';
1089 LOG_INFO("Bot ", self.netname, " using ", self.weaponname, " thinks it has fired, but apparently did not; ATTACK_FINISHED says ", ftos(ATTACK_FINISHED(self, slot) - time), " seconds left\n");
1093 return CMD_STATUS_FINISHED;
1098 void bot_command_executed(float rm)
1105 bot_dequeuecommand(self, self.bot_cmd_execution_index);
1107 self.bot_cmd_execution_index++;
1110 void bot_setcurrentcommand()
1114 if(!self.bot_cmd_current)
1116 self.bot_cmd_current = new(bot_cmd);
1117 self.bot_cmd_current.is_bot_cmd = true;
1120 bot_cmd = self.bot_cmd_current;
1121 if(bot_cmd.bot_cmd_index != self.bot_cmd_execution_index || self.bot_cmd_execution_index == 0)
1123 if(bot_havecommand(self, self.bot_cmd_execution_index))
1126 cmdstring = bot_readcommand(self, self.bot_cmd_execution_index);
1127 if(bot_decodecommand(cmdstring))
1129 bot_cmd.owner = self;
1130 bot_cmd.bot_cmd_index = self.bot_cmd_execution_index;
1134 // Invalid command, remove from queue
1136 bot_dequeuecommand(self, self.bot_cmd_execution_index);
1137 self.bot_cmd_execution_index++;
1145 void bot_resetqueues()
1147 FOREACH_CLIENT(it.isbot, LAMBDA(
1148 it.bot_cmd_execution_index = 0;
1150 // also, cancel all barriers
1152 for(int i = 0; i < it.bot_places_count; ++i)
1154 strunzone(it.(bot_placenames[i]));
1155 it.(bot_placenames[i]) = string_null;
1157 it.bot_places_count = 0;
1160 bot_barriertime = time;
1163 // Here we map commands to functions and deal with complex interactions between commands and execution states
1164 // NOTE: Of course you need to include your commands here too :)
1165 float bot_execute_commands_once()
1167 float status, ispressingkey;
1170 bot_setcurrentcommand();
1172 // if we have no bot command, better return
1173 // old logic kept pressing previously pressed keys, but that has problems
1174 // (namely, it means you cannot make a bot "normal" ever again)
1175 // to keep a bot walking for a while, use the "wait" bot command
1176 if(bot_cmd == world)
1179 // Ignore all commands except continue when the bot is paused
1180 if(self.bot_exec_status & BOT_EXEC_STATUS_PAUSED)
1181 if(bot_cmd.bot_cmd_type!=BOT_CMD_CONTINUE)
1183 if(bot_cmd.bot_cmd_type!=BOT_CMD_NULL)
1185 bot_command_executed(true);
1186 LOG_INFO( "WARNING: Commands are ignored while the bot is paused. Use the command 'continue' instead.\n");
1191 // Keep pressing keys raised by the "presskey" command
1192 ispressingkey = boolean(bot_presskeys());
1194 // Handle conditions
1195 if (!(bot_cmd.bot_cmd_type==BOT_CMD_FI||bot_cmd.bot_cmd_type==BOT_CMD_ELSE))
1196 if(self.bot_cmd_condition_status & CMD_CONDITION_true && self.bot_cmd_condition_status & CMD_CONDITION_false_BLOCK)
1198 bot_command_executed(true);
1201 else if(self.bot_cmd_condition_status & CMD_CONDITION_false && self.bot_cmd_condition_status & CMD_CONDITION_true_BLOCK)
1203 bot_command_executed(true);
1207 // Map commands to functions
1208 switch(bot_cmd.bot_cmd_type)
1211 return ispressingkey;
1214 status = bot_cmd_pause();
1216 case BOT_CMD_CONTINUE:
1217 status = bot_cmd_continue();
1220 status = bot_cmd_wait();
1222 case BOT_CMD_WAIT_UNTIL:
1223 status = bot_cmd_wait_until();
1226 status = bot_cmd_turn();
1228 case BOT_CMD_MOVETO:
1229 status = bot_cmd_moveto();
1231 case BOT_CMD_MOVETOTARGET:
1232 status = bot_cmd_movetotarget();
1234 case BOT_CMD_RESETGOAL:
1235 status = bot_cmd_resetgoal();
1238 status = bot_cmd_cc();
1241 status = bot_cmd_if();
1244 status = bot_cmd_else();
1247 status = bot_cmd_fi();
1249 case BOT_CMD_RESETAIM:
1250 status = bot_cmd_resetaim();
1253 status = bot_cmd_aim();
1255 case BOT_CMD_AIMTARGET:
1256 status = bot_cmd_aimtarget();
1258 case BOT_CMD_PRESSKEY:
1259 status = bot_cmd_presskey();
1261 case BOT_CMD_RELEASEKEY:
1262 status = bot_cmd_releasekey();
1264 case BOT_CMD_SELECTWEAPON:
1265 status = bot_cmd_select_weapon();
1267 case BOT_CMD_IMPULSE:
1268 status = bot_cmd_impulse();
1270 case BOT_CMD_BARRIER:
1271 status = bot_cmd_barrier();
1273 case BOT_CMD_CONSOLE:
1274 localcmd(strcat(bot_cmd.bot_cmd_parm_string, "\n"));
1275 status = CMD_STATUS_FINISHED;
1278 status = bot_cmd_sound();
1280 case BOT_CMD_DEBUG_ASSERT_CANFIRE:
1281 status = bot_cmd_debug_assert_canfire();
1284 LOG_INFO(strcat("ERROR: Invalid command on queue with id '",ftos(bot_cmd.bot_cmd_type),"'\n"));
1288 if (status==CMD_STATUS_ERROR)
1289 LOG_INFO(strcat("ERROR: The command '",bot_cmd_string[bot_cmd.bot_cmd_type],"' returned an error status\n"));
1291 // Move execution pointer
1292 if(status==CMD_STATUS_EXECUTING)
1298 if(autocvar_g_debug_bot_commands)
1302 switch(bot_cmd_parm_type[bot_cmd.bot_cmd_type])
1304 case BOT_CMD_PARAMETER_FLOAT:
1305 parms = ftos(bot_cmd.bot_cmd_parm_float);
1307 case BOT_CMD_PARAMETER_STRING:
1308 parms = bot_cmd.bot_cmd_parm_string;
1310 case BOT_CMD_PARAMETER_VECTOR:
1311 parms = vtos(bot_cmd.bot_cmd_parm_vector);
1317 clientcommand(self,strcat("say ^7", bot_cmd_string[bot_cmd.bot_cmd_type]," ",parms,"\n"));
1320 bot_command_executed(true);
1323 if(status == CMD_STATUS_FINISHED)
1326 return CMD_STATUS_ERROR;
1329 // This function should be (the only) called directly from the bot ai loop
1330 float bot_execute_commands()
1335 f = bot_execute_commands_once();