]> git.rm.cloudns.org Git - xonotic/xonotic-data.pk3dir.git/commitdiff
Improve ka/tka/kh bot key/ball chasing logic 1429/head
authorotta8634 <k9wolf@pm.me>
Sun, 9 Mar 2025 08:36:40 +0000 (16:36 +0800)
committerotta8634 <k9wolf@pm.me>
Sun, 9 Mar 2025 08:49:31 +0000 (16:49 +0800)
Made them only chase a key/ball they can see.
- First check if a waypoint exists and is visible.
- If not, use checkpvs.
Made the tka bots "chase" their own ball, so they can protect their team's ball carrier.
- Uses a lower rating scale (half) than chasing a ball the other team has.

qcsrc/common/gamemodes/gamemode/keepaway/sv_keepaway.qc
qcsrc/common/gamemodes/gamemode/keyhunt/sv_keyhunt.qc
qcsrc/common/gamemodes/gamemode/tka/sv_tka.qc

index 37054373dd315415f5ee0484b3ce7700c1788b67..648d05c3f81270303d69e18f86f8ae2f6b1944f6 100644 (file)
@@ -302,6 +302,18 @@ void ka_Handler_CheckBall(entity this)
 // Bot player logic
 // ================
 
+bool ka_waypointsprite_visible_for_bot(entity this, entity e, bool dropped)
+{
+       entity ball_wp = e.waypointsprite_attachedforcarrier;
+       if (!ball_wp)
+       if (dropped || !ball_wp.waypointsprite_visible_for_player(ball_wp, this, WaypointSprite_getviewentity(this)))
+       { // has no waypoint, or waypoint not visible
+               if (!checkpvs(this.origin + this.view_ofs, e)) // ball cannot be seen
+                       return false;
+       }
+       return true;
+}
+
 void havocbot_goalrating_ball(entity this, float ratingscale, vector org)
 {
        entity ball = NULL, ball_carried = NULL;
@@ -309,15 +321,23 @@ void havocbot_goalrating_ball(entity this, float ratingscale, vector org)
        // stops at last ball, prefers ball without carrier
        IL_EACH(g_kaballs, it.owner != this,
        {
-               if(it.owner)
+               if (it.owner)
+               {
+                       if (!ka_waypointsprite_visible_for_bot(this, it.owner, false))
+                               continue;
                        ball_carried = it.owner;
+               }
                else
+               {
+                       if (!ka_waypointsprite_visible_for_bot(this, it, true))
+                               continue;
                        ball = it;
+               }
        });
 
-       if(ball)
+       if (ball)
                navigation_routerating(this, ball, ratingscale, 2000);
-       else if(ball_carried)
+       else if (ball_carried)
                navigation_routerating(this, ball_carried, ratingscale, 2000);
 }
 
index 04433c597777932aa20d42bb2d5839cdab17837a..85814dfecdc21a35b6756e352629c8c8dba635b2 100644 (file)
@@ -1025,30 +1025,37 @@ void(entity this) havocbot_role_kh_offense;
 void(entity this) havocbot_role_kh_freelancer;
 
 
+bool kh_waypointsprite_visible_for_bot(entity this, entity e)
+{
+       entity key_wp = e.waypointsprite_attachedforcarrier;
+       if (!key_wp)
+       if (!key_wp.waypointsprite_visible_for_player(key_wp, this, WaypointSprite_getviewentity(this)))
+       { // has no waypoint, or waypoint not visible
+               if (!checkpvs(this.origin + this.view_ofs, e)) // key cannot be seen
+                       return false;
+       }
+       return true;
+}
+
 void havocbot_goalrating_kh(entity this, float ratingscale_team, float ratingscale_dropped, float ratingscale_enemy)
 {
-       entity head;
-       for (head = kh_worldkeylist; head; head = head.kh_worldkeynext)
+       for (entity head = kh_worldkeylist; head; head = head.kh_worldkeynext)
        {
-               if(head.owner == this)
-                       continue;
-               if(!kh_tracking_enabled)
+               if (head.owner)
                {
-                       // if it's carried by our team we know about it
-                       // otherwise we have to see it to know about it
-                       if(!head.owner || head.team != this.team)
-                       {
-                               traceline(this.origin + this.view_ofs, head.origin, MOVE_NOMONSTERS, this);
-                               if (trace_fraction < 1 && trace_ent != head)
-                                       continue; // skip what I can't see
-                       }
+                       if (head.owner == this || !kh_waypointsprite_visible_for_bot(this, head.owner))
+                               continue;
+                       if (SAME_TEAM(head, this))
+                               navigation_routerating(this, head.owner, ratingscale_team * 10000, 100000);
+                       else
+                               navigation_routerating(this, head.owner, ratingscale_enemy * 10000, 100000);
                }
-               if(!head.owner)
-                       navigation_routerating(this, head, ratingscale_dropped * 10000, 100000);
-               else if(head.team == this.team)
-                       navigation_routerating(this, head.owner, ratingscale_team * 10000, 100000);
                else
-                       navigation_routerating(this, head.owner, ratingscale_enemy * 10000, 100000);
+               {
+                       if (!kh_waypointsprite_visible_for_bot(this, head))
+                               continue;
+                       navigation_routerating(this, head, ratingscale_dropped * 10000, 100000);
+               }
        }
 
        havocbot_goalrating_items(this, 80000, this.origin, 10000);
index df807f430df122c0f62c81ea366fcec0b4f8eda7..f6835cee2bb8dea00feaeec9c368e3a145c62c2f 100644 (file)
@@ -280,23 +280,48 @@ void tka_Handler_CheckBall(entity this)
 // Bot player logic
 // ================
 
-void havocbot_goalrating_tkaball(entity this, float ratingscale, vector org)
+bool tka_waypointsprite_visible_for_bot(entity this, entity e, bool dropped)
+{
+       entity ball_wp = e.waypointsprite_attachedforcarrier;
+       if (!ball_wp)
+       if (dropped || !ball_wp.waypointsprite_visible_for_player(ball_wp, this, WaypointSprite_getviewentity(this)))
+       { // has no waypoint, or waypoint not visible
+               if (!checkpvs(this.origin + this.view_ofs, e)) // ball cannot be seen
+                       return false;
+       }
+       return true;
+}
+
+void havocbot_goalrating_tkaball(entity this, float ratingscale, float ratingscale_sameteam, vector org)
 {
        entity ball = NULL, ball_carried = NULL;
 
        // stops at last ball, prefers ball without carrier
-       IL_EACH(g_tkaballs, it.owner != this && DIFF_TEAM(ball.owner, this),
+       IL_EACH(g_tkaballs, it.owner != this,
        {
-               if(it.owner)
+               if (it.owner)
+               {
+                       if (!tka_waypointsprite_visible_for_bot(this, it.owner, false))
+                               continue;
                        ball_carried = it.owner;
+               }
                else
+               {
+                       if (!tka_waypointsprite_visible_for_bot(this, it, true))
+                               continue;
                        ball = it;
+               }
        });
 
        if (ball)
                navigation_routerating(this, ball, ratingscale, 2000);
-       else if(ball_carried)
-               navigation_routerating(this, ball_carried, ratingscale, 2000);
+       else if (ball_carried)
+       {
+               if (DIFF_TEAM(ball_carried, this))
+                       navigation_routerating(this, ball_carried, ratingscale, 2000);
+               else
+                       navigation_routerating(this, ball_carried, ratingscale_sameteam, 2000);
+       }
 }
 
 void havocbot_role_tka_carrier(entity this)
@@ -332,7 +357,7 @@ void havocbot_role_tka_collector(entity this)
                navigation_goalrating_start(this);
                havocbot_goalrating_items(this, 10000, this.origin, 10000);
                havocbot_goalrating_enemyplayers(this, 500, this.origin, 10000);
-               havocbot_goalrating_tkaball(this, 8000, this.origin);
+               havocbot_goalrating_tkaball(this, 8000, 4000, this.origin);
                navigation_goalrating_end(this);
 
                navigation_goalrating_timeout_set(this);