Skip to content

Commit

Permalink
rip bozo
Browse files Browse the repository at this point in the history
  • Loading branch information
xanunderscore committed Sep 30, 2024
1 parent e91d6d7 commit c9e8b41
Show file tree
Hide file tree
Showing 4 changed files with 26 additions and 149 deletions.
59 changes: 9 additions & 50 deletions BossMod/AI/AIBehaviour.cs
Original file line number Diff line number Diff line change
@@ -1,25 +1,22 @@
using BossMod.Autorotation;
using BossMod.Pathfinding;
using Dalamud.Interface.Utility;
using ImGuiNET;

namespace BossMod.AI;

public record struct Targeting(AIHints.Enemy Target, float PreferredRange = 3, Positional PreferredPosition = Positional.Any, bool PreferTanking = false);

// constantly follow master
sealed class AIBehaviour(AIController ctrl, RotationModuleManager autorot, Preset? aiPreset) : IDisposable
sealed class AIBehaviour(AIController ctrl, RotationModuleManager autorot) : IDisposable
{
public WorldState WorldState => autorot.Bossmods.WorldState;
public Preset? AIPreset = aiPreset;
public float ForceMovementIn { get; private set; } = float.MaxValue; // TODO: reconsider
private readonly AIConfig _config = Service.Config.Get<AIConfig>();
private readonly NavigationDecision.Context _naviCtx = new();
private NavigationDecision _naviDecision;
private bool _afkMode;
private bool _followMaster; // if true, our navigation target is master rather than primary target - this happens e.g. in outdoor or in dungeons during gathering trash
private WPos _masterPrevPos;
private WPos _masterMovementStart;
private DateTime _masterLastMoved;

public void Dispose()
Expand All @@ -39,26 +36,18 @@ public void Execute(Actor player, Actor master)
_afkMode = master != player && !master.InCombat && (WorldState.CurrentTime - _masterLastMoved).TotalSeconds > 10;
bool gazeImminent = autorot.Hints.ForbiddenDirections.Count > 0 && autorot.Hints.ForbiddenDirections[0].activation <= WorldState.FutureTime(0.5f);
bool pyreticImminent = autorot.Hints.ImminentSpecialMode.mode == AIHints.SpecialMode.Pyretic && autorot.Hints.ImminentSpecialMode.activation <= WorldState.FutureTime(1);
bool forbidActions = _config.ForbidActions || _afkMode || gazeImminent || pyreticImminent || autorot.Preset != null && autorot.Preset != AIPreset;
bool forbidTargeting = _config.ForbidActions || _afkMode || gazeImminent || pyreticImminent;

Targeting target = new();
if (!forbidActions)
if (!forbidTargeting)
{
target = SelectPrimaryTarget(player, master);
if (target.Target != null || TargetIsForbidden(player.TargetID))
autorot.Hints.ForcedTarget ??= target.Target?.Actor;
AdjustTargetPositional(player, ref target);
}

if (_config.OverridePositional)
{
target.PreferredPosition = _config.DesiredPositional;
target.PreferTanking = _config.DesiredPositional != Positional.Any;
}
if (_config.OverrideRange)
target.PreferredRange = _config.MaxDistanceToTarget;

_followMaster = master != player && (autorot.Bossmods.ActiveModule?.StateMachine.ActiveState == null || _config.FollowDuringActiveBossModule) && (!master.InCombat || _config.FollowDuringCombat || (_masterPrevPos - _masterMovementStart).LengthSq() > 100) && (player.InCombat || _config.FollowOutOfCombat);
_followMaster = master != player;

// note: if there are pending knockbacks, don't update navigation decision to avoid fucking up positioning
if (!WorldState.PendingEffects.PendingKnockbacks(player.InstanceID))
Expand All @@ -72,13 +61,7 @@ public void Execute(Actor player, Actor master)
bool moveWithMaster = masterIsMoving && _followMaster && master != player;
ForceMovementIn = moveWithMaster || gazeImminent || pyreticImminent ? 0 : _naviDecision.LeewaySeconds;

// note: that there is a 1-frame delay if target and/or strategy changes - we don't really care?..
if (!forbidActions)
{
autorot.Preset = AIPreset;
}

UpdateMovement(player, master, target, gazeImminent || pyreticImminent, !forbidActions ? autorot.Hints.ActionsToExecute : null);
UpdateMovement(player, master, target, gazeImminent || pyreticImminent, !forbidTargeting ? autorot.Hints.ActionsToExecute : null);
}

// returns null if we're to be idle, otherwise target to attack
Expand Down Expand Up @@ -146,7 +129,7 @@ private NavigationDecision BuildNavigationDecision(Actor player, Actor master, r
if (_followMaster)
{
autorot.Hints.GoalZones.Clear();
autorot.Hints.GoalZones.Add(autorot.Hints.GoalSingleTarget(master.Position, _config.OverrideRange ? _config.MaxDistanceToSlot : 1));
autorot.Hints.GoalZones.Add(autorot.Hints.GoalSingleTarget(master.Position, 1));
return NavigationDecision.Build(_naviCtx, WorldState, autorot.Hints, player);
}

Expand All @@ -162,7 +145,7 @@ private void FocusMaster(Actor master)
if (masterChanged)
{
ctrl.SetFocusTarget(master);
_masterPrevPos = _masterMovementStart = master.Position;
_masterPrevPos = master.Position;
_masterLastMoved = WorldState.CurrentTime.AddSeconds(-1);
}
}
Expand All @@ -179,8 +162,6 @@ private bool TrackMasterMovement(Actor master)
}
else if ((WorldState.CurrentTime - _masterLastMoved).TotalSeconds > 0.5f)
{
// master has stopped, consider previous movement finished
_masterMovementStart = _masterPrevPos;
masterIsMoving = false;
}
// else: don't consider master to have stopped moving unless he's standing still for some small time
Expand Down Expand Up @@ -224,31 +205,9 @@ private void UpdateMovement(Actor player, Actor master, Targeting target, bool g

public void DrawDebug()
{
ImGui.Checkbox("Forbid actions", ref _config.ForbidActions);
ImGui.SameLine();
ImGui.Checkbox("Forbid movement", ref _config.ForbidMovement);
ImGui.Checkbox("Disable autotarget", ref _config.ForbidActions);
ImGui.SameLine();
ImGui.Checkbox("Show extra options", ref _config.ShowExtraUIOptions);
if (_config.ShowExtraUIOptions)
{
ImGui.Checkbox("Follow during combat", ref _config.FollowDuringCombat);
ImGui.SameLine();
ImGui.Checkbox("Follow during active boss module", ref _config.FollowDuringActiveBossModule);
ImGui.Spacing();
ImGui.Checkbox("Follow out of combat", ref _config.FollowOutOfCombat);
ImGui.SameLine();
ImGui.Checkbox("Follow target", ref _config.FollowTarget);
ImGui.SameLine();
ImGui.Checkbox("Override follow range", ref _config.OverrideRange);
ImGui.PushItemWidth(75 * ImGuiHelpers.GlobalScale);
ImGui.InputFloat("Follow slot range", ref _config.MaxDistanceToSlot);
ImGui.SameLine();
ImGui.InputFloat("Follow target range", ref _config.MaxDistanceToTarget);
ImGui.PopItemWidth();
}
var player = WorldState.Party.Player();
var dist = _naviDecision.Destination != null && player != null ? (_naviDecision.Destination.Value - player.Position).Length() : 0;
ImGui.TextUnformatted($"Max-cast={MathF.Min(ForceMovementIn, 1000):f3}, afk={_afkMode}, follow={_followMaster}, \n{_naviDecision.Destination} (d={dist:f3}), master standing for {Math.Clamp((WorldState.CurrentTime - _masterLastMoved).TotalSeconds, 0, 1000):f1}");
ImGui.Checkbox("Disable movement", ref _config.ForbidMovement);
}

private bool TargetIsForbidden(ulong actorId) => autorot.Hints.ForbiddenTargets.Any(e => e.Actor.InstanceID == actorId);
Expand Down
36 changes: 2 additions & 34 deletions BossMod/AI/AIConfig.cs
Original file line number Diff line number Diff line change
Expand Up @@ -12,57 +12,25 @@ public enum Slot { One, Two, Three, Four }
[PropertyDisplay("Show in-game UI")]
public bool DrawUI = true;

[PropertyDisplay("Show advanced options in the UI")]
public bool ShowExtraUIOptions = true;

[PropertyDisplay("Show AI status in the in-game UI's title bar")]
public bool ShowStatusOnTitlebar = true;

[PropertyDisplay("Show AI status in the server info bar")]
public bool ShowDTR = true;

// ai settings
[PropertyDisplay($"Override positional")]
public bool OverridePositional = false;

[PropertyDisplay("Desired positional")]
public Positional DesiredPositional = 0;

[PropertyDisplay($"Follow slot")]
public Slot FollowSlot = 0;

[PropertyDisplay($"Override follow range")]
public bool OverrideRange = false;

[PropertyDisplay($"Follow slot range")]
public float MaxDistanceToSlot = 1;

[PropertyDisplay($"Follow target")]
public bool FollowTarget = false;

[PropertyDisplay($"Follow target range")]
public float MaxDistanceToTarget = 2.6f;

[PropertyDisplay("Follow during active boss module")]
public bool FollowDuringActiveBossModule = false;

[PropertyDisplay("Follow during combat")]
public bool FollowDuringCombat = false;

[PropertyDisplay("Follow out of combat")]
public bool FollowOutOfCombat = false;

[PropertyDisplay("Forbid movement")]
[PropertyDisplay("Disable movement")]
public bool ForbidMovement = false;

[PropertyDisplay("Forbid actions")]
[PropertyDisplay("Disable auto-target")]
public bool ForbidActions = false;

[PropertyDisplay("Focus target master")]
public bool FocusTargetMaster = false;

[PropertyDisplay("Broadcast keypresses to other windows")]
public bool BroadcastToSlaves = false;

public string LastAIPreset = "";
}
68 changes: 5 additions & 63 deletions BossMod/AI/AIManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,8 @@
using Dalamud.Game.Text.SeStringHandling.Payloads;
using Dalamud.Interface;
using Dalamud.Interface.Utility.Raii;
using Dalamud.Utility;
using FFXIVClientStructs.FFXIV.Client.Game.Group;
using ImGuiNET;
using static Dalamud.Interface.Windowing.Window;

namespace BossMod.AI;

Expand All @@ -17,14 +15,11 @@ sealed class AIManager : IDisposable
private readonly AIController _controller;
private readonly AIConfig _config;
private int MasterSlot => (int)_config.FollowSlot; // non-zero means corresponding player is master
private Positional DesiredPositional => _config.DesiredPositional;
private Preset? _aiPreset;
private readonly UISimpleWindow _ui;
private WorldState WorldState => _autorot.Bossmods.WorldState;
private string _aiStatus = "";
private string _naviStatus = "";

public string GetAIPreset => _aiPreset?.Name ?? string.Empty;
public float ForceMovementIn => Behaviour?.ForceMovementIn ?? float.MaxValue;
public AIBehaviour? Behaviour { get; private set; }

Expand Down Expand Up @@ -57,9 +52,6 @@ public AIManager(RotationModuleManager autorot, ActionManagerEx amex, MovementOv
};
Service.ChatGui.ChatMessage += OnChatMessage;
Service.CommandManager.AddHandler("/vbmai", new Dalamud.Game.Command.CommandInfo(OnCommand) { HelpMessage = "Toggle AI mode" });

if (!_config.LastAIPreset.IsNullOrEmpty())
_aiPreset = autorot.Database.Presets.Presets.FirstOrDefault(p => p.Name == _config.LastAIPreset);
}

public void Dispose()
Expand All @@ -82,7 +74,6 @@ public void Update()

var player = WorldState.Party.Player();
var master = WorldState.Party[MasterSlot];
var target = WorldState.Actors.Find(WorldState.Party.Player()?.TargetID ?? 0);

if (Behaviour != null && player != null && master != null)
{
Expand All @@ -94,22 +85,13 @@ public void Update()
}

_controller.Update(player, _autorot.Hints, WorldState.CurrentTime);
_aiStatus = $"AI: {(Behaviour != null ? $"on, {(_config.FollowTarget && target != null ? $"target={target.Name}" : $"master={master?.Name}[{((int)_config.FollowSlot) + 1}]")}" : "off")}";
_naviStatus = $"Navi={_controller.NaviTargetPos}";
_aiStatus = $"AI: {(Behaviour != null ? $"on, {$"master={master?.Name}[{(int)_config.FollowSlot + 1}]"}" : "off")}";
var dist = _controller.NaviTargetPos != null && player != null ? (_controller.NaviTargetPos.Value - player.Position).Length() : 0;
_naviStatus = $"Navi={_controller.NaviTargetPos?.ToString() ?? "<none>"} (d={dist:f3}, max-cast={MathF.Min(Behaviour?.ForceMovementIn ?? float.MaxValue, 1000):f3})";
_ui.IsOpen = player != null && _config.DrawUI;
_ui.WindowName = _config.ShowStatusOnTitlebar ? $"{_aiStatus}, {_naviStatus}###AI" : $"AI###AI";
}

public void SetAIPreset(Preset? p)
{
_aiPreset = p;
if (Behaviour != null)
Behaviour.AIPreset = p;
if (p != null)
_config.LastAIPreset = p.Name;
_config.Modified.Fire();
}

private void DrawOverlay()
{
if (!_config.ShowStatusOnTitlebar)
Expand All @@ -119,69 +101,29 @@ private void DrawOverlay()
}
Behaviour?.DrawDebug();

using (var leaderCombo = ImRaii.Combo("Follow", Behaviour == null ? "<idle>" : (_config.FollowTarget ? "<target>" : WorldState.Party[MasterSlot]?.Name ?? "<unknown>")))
using (var leaderCombo = ImRaii.Combo("Follow", Behaviour == null ? "<idle>" : WorldState.Party[MasterSlot]?.Name ?? "<unknown>"))
{
if (leaderCombo)
{
if (ImGui.Selectable("<idle>", Behaviour == null))
{
Enabled = false;
}
if (ImGui.Selectable("<target>", _config.FollowTarget))
{
_config.FollowSlot = 0;
_config.FollowTarget = true;
_config.Modified.Fire();
Enabled = true;
}
foreach (var (i, p) in WorldState.Party.WithSlot(true))
{
if (ImGui.Selectable(p.Name, MasterSlot == i))
{
_config.FollowSlot = (AIConfig.Slot)i;
_config.FollowTarget = false;
_config.Modified.Fire();
Enabled = true;
}
}
}
}

using (var positionalCombo = ImRaii.Combo("Positional", $"{DesiredPositional}"))
{
if (positionalCombo)
{
for (var i = 0; i < 4; i++)
{
if (ImGui.Selectable($"{(Positional)i}", DesiredPositional == (Positional)i))
{
_config.DesiredPositional = (Positional)i;
_config.Modified.Fire();
}
}
}
}

using (var presetCombo = ImRaii.Combo("AI preset", _aiPreset?.Name ?? ""))
{
if (presetCombo)
{
foreach (var p in _autorot.Database.Presets.Presets)
{
if (ImGui.Selectable(p.Name, p == _aiPreset))
{
SetAIPreset(p);
}
}
}
}
}

public void SwitchToIdle()
{
if (_autorot.Preset == Behaviour?.AIPreset)
_autorot.Preset = null;

Behaviour?.Dispose();
Behaviour = null;

Expand All @@ -195,7 +137,7 @@ public void SwitchToFollow(int masterSlot)
SwitchToIdle();
_config.FollowSlot = (AIConfig.Slot)masterSlot;
_config.Modified.Fire();
Behaviour = new AIBehaviour(_controller, _autorot, _aiPreset);
Behaviour = new AIBehaviour(_controller, _autorot);
}

private unsafe int FindPartyMemberSlotFromSender(SeString sender)
Expand Down
12 changes: 10 additions & 2 deletions BossMod/Framework/IPCProvider.cs
Original file line number Diff line number Diff line change
Expand Up @@ -86,8 +86,16 @@ public IPCProvider(RotationModuleManager autorotation, ActionManagerEx amex, Mov
return false;
});

Register("AI.SetPreset", (string name) => ai.SetAIPreset(autorotation.Database.Presets.Presets.FirstOrDefault(x => x.Name == name)));
Register("AI.GetPreset", () => ai.GetAIPreset);
Register("AI.SetPreset", (string name) =>
{
Service.Log($"calling deprecated method AI.SetPreset");
autorotation.Preset = autorotation.Database.Presets.Presets.FirstOrDefault(x => x.Name == name);
});
Register("AI.GetPreset", () =>
{
Service.Log($"calling deprecated method AI.GetPreset");
return autorotation.Preset?.Name ?? string.Empty;
});
}

public void Dispose() => _disposeActions?.Invoke();
Expand Down

0 comments on commit c9e8b41

Please sign in to comment.