| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- using System;
- using System.Numerics;
- using System.Collections.Concurrent;
- using System.Collections.Generic;
- using System.Globalization;
- using System.Linq;
- using System.Threading.Tasks;
- using Dalamud.Data;
- using Dalamud.Game;
- using Dalamud.Game.ClientState;
- using Dalamud.Game.Command;
- using Dalamud.Game.Gui;
- using Dalamud.IoC;
- using Dalamud.Logging;
- using Dalamud.Plugin;
- using Dalamud.Utility;
- using Lumina.Excel.GeneratedSheets;
- using HuntBuddy.Attributes;
- using HuntBuddy.Ipc;
- using HuntBuddy.Structs;
- using ImGuiNET;
- using ImGuiScene;
- using Lumina.Data.Files;
- namespace HuntBuddy
- {
- public class Plugin : IDalamudPlugin
- {
- public string Name => "Hunt Buddy";
- [PluginService]
- [RequiredVersion("1.0")]
- public static DalamudPluginInterface PluginInterface { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static CommandManager Commands { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static ChatGui Chat { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static DataManager DataManager { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static SigScanner SigScanner { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static GameGui GameGui { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static ClientState ClientState { get; set; } = null!;
- [PluginService]
- [RequiredVersion("1.0")]
- public static Framework Framework { get; set; } = null!;
- private readonly PluginCommandManager<Plugin> commandManager;
- private readonly Interface pluginInterface;
- private ObtainedBillEnum lastState;
- // Dictionary<string ExpansionName, Dictionary<KeyValuePair<uint MobTerritoryType, string MobTerritoryName>, List<MobHuntEntry MobsInZone>>>
- public readonly Dictionary<string, Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>> MobHuntEntries;
- public readonly ConcurrentBag<MobHuntEntry> CurrentAreaMobHuntEntries;
- public bool MobHuntEntriesReady = true;
- public readonly unsafe MobHuntStruct* MobHuntStruct;
- public readonly Configuration Configuration;
- public static TeleportConsumer? TeleportConsumer;
- public Plugin()
- {
- this.commandManager = new PluginCommandManager<Plugin>(this, Commands);
- this.pluginInterface = new Interface(this);
- this.MobHuntEntries = new Dictionary<string, Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>>();
- this.CurrentAreaMobHuntEntries = new ConcurrentBag<MobHuntEntry>();
- this.Configuration = (Configuration)(PluginInterface.GetPluginConfig() ?? new Configuration());
- this.Configuration.IconBackgroundColourU32 =
- ImGui.ColorConvertFloat4ToU32(this.Configuration.IconBackgroundColour);
- unsafe
- {
- this.MobHuntStruct =
- (MobHuntStruct*)SigScanner.GetStaticAddressFromSig(
- "48 8D 0D ?? ?? ?? ?? 8B D8 0F B6 52");
- }
- Plugin.TeleportConsumer = new TeleportConsumer();
- Plugin.ClientState.TerritoryChanged += this.ClientStateOnTerritoryChanged;
- Plugin.PluginInterface.UiBuilder.Draw += this.DrawInterface;
- Plugin.PluginInterface.UiBuilder.Draw += this.pluginInterface.DrawLocalHunts;
- Plugin.PluginInterface.UiBuilder.OpenConfigUi += this.OpenConfigUi;
- Plugin.Framework.Update += this.FrameworkOnUpdate;
- }
- private unsafe void FrameworkOnUpdate(Framework framework)
- {
- if (this.lastState == this.MobHuntStruct->ObtainedBillEnumFlags)
- {
- return;
- }
- this.lastState = this.MobHuntStruct->ObtainedBillEnumFlags;
- this.PluginCommand(string.Empty, "reload");
- }
- private void ClientStateOnTerritoryChanged(object? sender, ushort e)
- {
- this.CurrentAreaMobHuntEntries.Clear();
- foreach (var mobHuntEntry in this.MobHuntEntries.SelectMany(
- expansionEntry => expansionEntry.Value
- .Where(entry => entry.Key.Key == Plugin.ClientState.TerritoryType)
- .SelectMany(entry => entry.Value)))
- {
- this.CurrentAreaMobHuntEntries.Add(mobHuntEntry);
- }
- }
- private void OpenConfigUi()
- {
- this.pluginInterface.DrawInterface = !this.pluginInterface.DrawInterface;
- }
- private void DrawInterface()
- {
- this.pluginInterface.DrawInterface = this.pluginInterface.DrawInterface && this.pluginInterface.Draw();
- }
- private void Dispose(bool disposing)
- {
- if (!disposing)
- {
- return;
- }
- this.MobHuntEntriesReady = false;
- Plugin.ClientState.TerritoryChanged -= this.ClientStateOnTerritoryChanged;
- Plugin.Framework.Update -= this.FrameworkOnUpdate;
- Plugin.PluginInterface.UiBuilder.Draw -= this.DrawInterface;
- Plugin.PluginInterface.UiBuilder.Draw -= this.pluginInterface.DrawLocalHunts;
- Plugin.PluginInterface.UiBuilder.OpenConfigUi -= this.OpenConfigUi;
- this.commandManager.Dispose();
- }
- [Command("/phb")]
- [HelpMessage("Toggles UI\nArguments:\nreload - Reloads data\nlocal - Toggles the local hunt marks window\nnext - Flags the next hunt target to find")]
- public unsafe void PluginCommand(string command, string args)
- {
- switch (args.Trim().ToLower()) {
- case "reload":
- this.MobHuntEntriesReady = false;
- Task.Run(this.ReloadData);
- break;
- case "local":
- this.Configuration.ShowLocalHunts = !this.Configuration.ShowLocalHunts;
- this.Configuration.Save();
- break;
- case "next":
- if (this.MobHuntEntries.Count > 0)
- {
- var openType = Location.OpenType.None;
- var playerLocation = Plugin.ClientState.LocalPlayer!.Position;
- var map = Plugin.DataManager.GetExcelSheet<TerritoryType>()!.GetRow(Plugin.ClientState.TerritoryType)!.Map!.Value!;
- var playerVec2 = MapUtil.WorldToMap(new Vector2(playerLocation.X, playerLocation.Z), map);
- var chosen = this.CurrentAreaMobHuntEntries
- .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
- .OrderBy(entry => Vector2.Distance(Location.Database[entry.MobHuntId].Coordinate, playerVec2))
- .FirstOrDefault();
- if (chosen == null)
- {
- PluginLog.Information("No marks in current zone, looking in current expansion");
- openType = this.Configuration.IncludeAreaOnMap ? Location.OpenType.ShowOpen : Location.OpenType.MarkerOpen;
- var expansion = Plugin.DataManager.Excel.GetSheet<TerritoryType>()!.GetRow(Plugin.ClientState.TerritoryType)!.ExVersion.Value!.Name;
- PluginLog.Information($"Player is in a zone from {expansion}; known expansions are {string.Join(", ", this.MobHuntEntries.Keys)}");
- var candidates = this.MobHuntEntries.ContainsKey(expansion)
- ? this.MobHuntEntries[expansion]
- .Values
- .SelectMany(l => l)
- .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
- .ToList()
- : new List<MobHuntEntry>();
- if (candidates.Count == 0)
- {
- PluginLog.Information("Nothing in current expansion, looking globally");
- candidates =
- this.MobHuntEntries.Values
- .SelectMany(dict => dict.Values)
- .SelectMany(l => l)
- .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
- .ToList();
- }
- if (candidates.Count > 1)
- {
- PluginLog.Information($"Found {candidates.Count} marks");
- chosen = candidates[new Random().Next(candidates.Count)];
- }
- }
- if (chosen != null)
- {
- PluginLog.Information($"Selected next hunt target: {chosen.Name}");
- Location.CreateMapMarker(
- chosen.TerritoryType,
- chosen.MapId,
- chosen.MobHuntId,
- chosen.Name,
- openType);
- }
- }
- break;
- default:
- this.OpenConfigUi();
- break;
- }
- }
- public unsafe void ReloadData()
- {
- this.MobHuntEntries.Clear();
- var mobHuntList = new List<MobHuntEntry>();
- var mobHuntOrderSheet = Plugin.DataManager.Excel.GetSheet<MobHuntOrder>()!;
- foreach (var billNumber in Enum.GetValues<BillEnum>())
- {
- if (!this.MobHuntStruct->ObtainedBillEnumFlags.HasFlag((ObtainedBillEnum)(1 << (int)billNumber)))
- {
- continue;
- }
- var mobHuntOrderTypeRow =
- Plugin.DataManager.Excel.GetSheet<MobHuntOrderType>()!.GetRow((uint)billNumber)!;
- var rowId = mobHuntOrderTypeRow.OrderStart.Value!.RowId +
- (uint)(this.MobHuntStruct->BillOffset[mobHuntOrderTypeRow.RowId] - 1);
- if (rowId > mobHuntOrderSheet.RowCount)
- {
- continue;
- }
- var mobHuntOrderRows = mobHuntOrderSheet.Where(x => x.RowId == rowId);
- foreach (var mobHuntOrderRow in mobHuntOrderRows)
- {
- var mobHuntEntry =
- mobHuntList.FirstOrDefault(x => x.MobHuntId == mobHuntOrderRow.Target.Value!.Name.Row);
- if (mobHuntEntry == null)
- {
- mobHuntList.Add(
- new MobHuntEntry
- {
- Name = CultureInfo.InvariantCulture.TextInfo.ToTitleCase(
- mobHuntOrderRow.Target.Value!.Name.Value!.Singular),
- TerritoryName =
- mobHuntOrderRow.Target.Value!.TerritoryType.Value!.PlaceName.Value!.Name,
- ExpansionName = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Value!
- .ExVersion.Value!.Name,
- ExpansionId = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Value!
- .ExVersion.Row,
- MapId = mobHuntOrderRow.Target.Value!.TerritoryType.Row,
- TerritoryType = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Row,
- MobHuntId = mobHuntOrderRow.Target.Value!.Name.Row,
- IsEliteMark = billNumber is BillEnum.ArrElite or BillEnum.HwElite or BillEnum.SbElite
- or BillEnum.ShbElite or BillEnum.EwElite,
- CurrentKillsOffset = (5 * (uint)billNumber) + mobHuntOrderRow.SubRowId,
- NeededKills = mobHuntOrderRow.NeededKills,
- Icon = Plugin.LoadIcon(mobHuntOrderRow.Target.Value.Icon)
- });
- }
- else
- {
- if (mobHuntEntry.NeededKills < mobHuntOrderRow.NeededKills)
- {
- mobHuntEntry.NeededKills = mobHuntOrderRow.NeededKills;
- }
- }
- }
- }
- foreach (var entry in mobHuntList)
- {
- var key = entry.ExpansionName ?? "Unknown";
- var subKey = new KeyValuePair<uint, string>(entry.TerritoryType, entry.TerritoryName ?? "Unknown");
- if (!this.MobHuntEntries.ContainsKey(key))
- {
- this.MobHuntEntries[key] = new Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>();
- }
- if (!this.MobHuntEntries[key].ContainsKey(subKey))
- {
- this.MobHuntEntries[key][subKey] = new List<MobHuntEntry>();
- }
- this.MobHuntEntries[key][subKey].Add(entry);
- }
- this.ClientStateOnTerritoryChanged(null, 0);
- this.MobHuntEntriesReady = true;
- }
- private static TexFile? GetHdIcon(uint id)
- {
- var path = $"ui/icon/{id / 1000 * 1000:000000}/{id:000000}_hr1.tex";
- return Plugin.DataManager.GetFile<TexFile>(path);
- }
- private static TextureWrap LoadIcon(uint id)
- {
- var icon = Plugin.GetHdIcon(id) ?? Plugin.DataManager.GetIcon(id)!;
- var iconData = icon.GetRgbaImageData();
- return Plugin.PluginInterface.UiBuilder.LoadImageRaw(iconData, icon.Header.Width, icon.Header.Height, 4);
- }
- public void Dispose()
- {
- this.Dispose(true);
- GC.SuppressFinalize(this);
- }
- }
- }
|