Plugin.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. using System;
  2. using System.Numerics;
  3. using System.Collections.Concurrent;
  4. using System.Collections.Generic;
  5. using System.Globalization;
  6. using System.Linq;
  7. using System.Threading.Tasks;
  8. using Dalamud.Data;
  9. using Dalamud.Game;
  10. using Dalamud.Game.ClientState;
  11. using Dalamud.Game.Command;
  12. using Dalamud.Game.Gui;
  13. using Dalamud.IoC;
  14. using Dalamud.Logging;
  15. using Dalamud.Plugin;
  16. using Dalamud.Utility;
  17. using Lumina.Excel.GeneratedSheets;
  18. using HuntBuddy.Attributes;
  19. using HuntBuddy.Ipc;
  20. using HuntBuddy.Structs;
  21. using ImGuiNET;
  22. using ImGuiScene;
  23. using Lumina.Data.Files;
  24. namespace HuntBuddy
  25. {
  26. public class Plugin : IDalamudPlugin
  27. {
  28. public string Name => "Hunt Buddy";
  29. [PluginService]
  30. [RequiredVersion("1.0")]
  31. public static DalamudPluginInterface PluginInterface { get; set; } = null!;
  32. [PluginService]
  33. [RequiredVersion("1.0")]
  34. public static CommandManager Commands { get; set; } = null!;
  35. [PluginService]
  36. [RequiredVersion("1.0")]
  37. public static ChatGui Chat { get; set; } = null!;
  38. [PluginService]
  39. [RequiredVersion("1.0")]
  40. public static DataManager DataManager { get; set; } = null!;
  41. [PluginService]
  42. [RequiredVersion("1.0")]
  43. public static SigScanner SigScanner { get; set; } = null!;
  44. [PluginService]
  45. [RequiredVersion("1.0")]
  46. public static GameGui GameGui { get; set; } = null!;
  47. [PluginService]
  48. [RequiredVersion("1.0")]
  49. public static ClientState ClientState { get; set; } = null!;
  50. [PluginService]
  51. [RequiredVersion("1.0")]
  52. public static Framework Framework { get; set; } = null!;
  53. private readonly PluginCommandManager<Plugin> commandManager;
  54. private readonly Interface pluginInterface;
  55. private ObtainedBillEnum lastState;
  56. // Dictionary<string ExpansionName, Dictionary<KeyValuePair<uint MobTerritoryType, string MobTerritoryName>, List<MobHuntEntry MobsInZone>>>
  57. public readonly Dictionary<string, Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>> MobHuntEntries;
  58. public readonly ConcurrentBag<MobHuntEntry> CurrentAreaMobHuntEntries;
  59. public bool MobHuntEntriesReady = true;
  60. public readonly unsafe MobHuntStruct* MobHuntStruct;
  61. public readonly Configuration Configuration;
  62. public static TeleportConsumer? TeleportConsumer;
  63. public Plugin()
  64. {
  65. this.commandManager = new PluginCommandManager<Plugin>(this, Commands);
  66. this.pluginInterface = new Interface(this);
  67. this.MobHuntEntries = new Dictionary<string, Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>>();
  68. this.CurrentAreaMobHuntEntries = new ConcurrentBag<MobHuntEntry>();
  69. this.Configuration = (Configuration)(PluginInterface.GetPluginConfig() ?? new Configuration());
  70. this.Configuration.IconBackgroundColourU32 =
  71. ImGui.ColorConvertFloat4ToU32(this.Configuration.IconBackgroundColour);
  72. unsafe
  73. {
  74. this.MobHuntStruct =
  75. (MobHuntStruct*)SigScanner.GetStaticAddressFromSig(
  76. "48 8D 0D ?? ?? ?? ?? 8B D8 0F B6 52");
  77. }
  78. Plugin.TeleportConsumer = new TeleportConsumer();
  79. Plugin.ClientState.TerritoryChanged += this.ClientStateOnTerritoryChanged;
  80. Plugin.PluginInterface.UiBuilder.Draw += this.DrawInterface;
  81. Plugin.PluginInterface.UiBuilder.Draw += this.pluginInterface.DrawLocalHunts;
  82. Plugin.PluginInterface.UiBuilder.OpenConfigUi += this.OpenConfigUi;
  83. Plugin.Framework.Update += this.FrameworkOnUpdate;
  84. }
  85. private unsafe void FrameworkOnUpdate(Framework framework)
  86. {
  87. if (this.lastState == this.MobHuntStruct->ObtainedBillEnumFlags)
  88. {
  89. return;
  90. }
  91. this.lastState = this.MobHuntStruct->ObtainedBillEnumFlags;
  92. this.PluginCommand(string.Empty, "reload");
  93. }
  94. private void ClientStateOnTerritoryChanged(object? sender, ushort e)
  95. {
  96. this.CurrentAreaMobHuntEntries.Clear();
  97. foreach (var mobHuntEntry in this.MobHuntEntries.SelectMany(
  98. expansionEntry => expansionEntry.Value
  99. .Where(entry => entry.Key.Key == Plugin.ClientState.TerritoryType)
  100. .SelectMany(entry => entry.Value)))
  101. {
  102. this.CurrentAreaMobHuntEntries.Add(mobHuntEntry);
  103. }
  104. }
  105. private void OpenConfigUi()
  106. {
  107. this.pluginInterface.DrawInterface = !this.pluginInterface.DrawInterface;
  108. }
  109. private void DrawInterface()
  110. {
  111. this.pluginInterface.DrawInterface = this.pluginInterface.DrawInterface && this.pluginInterface.Draw();
  112. }
  113. private void Dispose(bool disposing)
  114. {
  115. if (!disposing)
  116. {
  117. return;
  118. }
  119. this.MobHuntEntriesReady = false;
  120. Plugin.ClientState.TerritoryChanged -= this.ClientStateOnTerritoryChanged;
  121. Plugin.Framework.Update -= this.FrameworkOnUpdate;
  122. Plugin.PluginInterface.UiBuilder.Draw -= this.DrawInterface;
  123. Plugin.PluginInterface.UiBuilder.Draw -= this.pluginInterface.DrawLocalHunts;
  124. Plugin.PluginInterface.UiBuilder.OpenConfigUi -= this.OpenConfigUi;
  125. this.commandManager.Dispose();
  126. }
  127. [Command("/phb")]
  128. [HelpMessage("Toggles UI\nArguments:\nreload - Reloads data\nlocal - Toggles the local hunt marks window\nnext - Flags the next hunt target to find")]
  129. public unsafe void PluginCommand(string command, string args)
  130. {
  131. switch (args.Trim().ToLower()) {
  132. case "reload":
  133. this.MobHuntEntriesReady = false;
  134. Task.Run(this.ReloadData);
  135. break;
  136. case "local":
  137. this.Configuration.ShowLocalHunts = !this.Configuration.ShowLocalHunts;
  138. this.Configuration.Save();
  139. break;
  140. case "next":
  141. if (this.MobHuntEntries.Count > 0)
  142. {
  143. var openType = Location.OpenType.None;
  144. var playerLocation = Plugin.ClientState.LocalPlayer!.Position;
  145. var map = Plugin.DataManager.GetExcelSheet<TerritoryType>()!.GetRow(Plugin.ClientState.TerritoryType)!.Map!.Value!;
  146. var playerVec2 = MapUtil.WorldToMap(new Vector2(playerLocation.X, playerLocation.Z), map);
  147. var chosen = this.CurrentAreaMobHuntEntries
  148. .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
  149. .OrderBy(entry => Vector2.Distance(Location.Database[entry.MobHuntId].Coordinate, playerVec2))
  150. .FirstOrDefault();
  151. if (chosen == null)
  152. {
  153. PluginLog.Information("No marks in current zone, looking in current expansion");
  154. openType = this.Configuration.IncludeAreaOnMap ? Location.OpenType.ShowOpen : Location.OpenType.MarkerOpen;
  155. var expansion = Plugin.DataManager.Excel.GetSheet<TerritoryType>()!.GetRow(Plugin.ClientState.TerritoryType)!.ExVersion.Value!.Name;
  156. PluginLog.Information($"Player is in a zone from {expansion}; known expansions are {string.Join(", ", this.MobHuntEntries.Keys)}");
  157. var candidates = this.MobHuntEntries.ContainsKey(expansion)
  158. ? this.MobHuntEntries[expansion]
  159. .Values
  160. .SelectMany(l => l)
  161. .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
  162. .ToList()
  163. : new List<MobHuntEntry>();
  164. if (candidates.Count == 0)
  165. {
  166. PluginLog.Information("Nothing in current expansion, looking globally");
  167. candidates =
  168. this.MobHuntEntries.Values
  169. .SelectMany(dict => dict.Values)
  170. .SelectMany(l => l)
  171. .Where(entry => this.MobHuntStruct->CurrentKills[entry.CurrentKillsOffset] < entry.NeededKills)
  172. .ToList();
  173. }
  174. if (candidates.Count > 1)
  175. {
  176. PluginLog.Information($"Found {candidates.Count} marks");
  177. chosen = candidates[new Random().Next(candidates.Count)];
  178. }
  179. }
  180. if (chosen != null)
  181. {
  182. PluginLog.Information($"Selected next hunt target: {chosen.Name}");
  183. Location.CreateMapMarker(
  184. chosen.TerritoryType,
  185. chosen.MapId,
  186. chosen.MobHuntId,
  187. chosen.Name,
  188. openType);
  189. }
  190. }
  191. break;
  192. default:
  193. this.OpenConfigUi();
  194. break;
  195. }
  196. }
  197. public unsafe void ReloadData()
  198. {
  199. this.MobHuntEntries.Clear();
  200. var mobHuntList = new List<MobHuntEntry>();
  201. var mobHuntOrderSheet = Plugin.DataManager.Excel.GetSheet<MobHuntOrder>()!;
  202. foreach (var billNumber in Enum.GetValues<BillEnum>())
  203. {
  204. if (!this.MobHuntStruct->ObtainedBillEnumFlags.HasFlag((ObtainedBillEnum)(1 << (int)billNumber)))
  205. {
  206. continue;
  207. }
  208. var mobHuntOrderTypeRow =
  209. Plugin.DataManager.Excel.GetSheet<MobHuntOrderType>()!.GetRow((uint)billNumber)!;
  210. var rowId = mobHuntOrderTypeRow.OrderStart.Value!.RowId +
  211. (uint)(this.MobHuntStruct->BillOffset[mobHuntOrderTypeRow.RowId] - 1);
  212. if (rowId > mobHuntOrderSheet.RowCount)
  213. {
  214. continue;
  215. }
  216. var mobHuntOrderRows = mobHuntOrderSheet.Where(x => x.RowId == rowId);
  217. foreach (var mobHuntOrderRow in mobHuntOrderRows)
  218. {
  219. var mobHuntEntry =
  220. mobHuntList.FirstOrDefault(x => x.MobHuntId == mobHuntOrderRow.Target.Value!.Name.Row);
  221. if (mobHuntEntry == null)
  222. {
  223. mobHuntList.Add(
  224. new MobHuntEntry
  225. {
  226. Name = CultureInfo.InvariantCulture.TextInfo.ToTitleCase(
  227. mobHuntOrderRow.Target.Value!.Name.Value!.Singular),
  228. TerritoryName =
  229. mobHuntOrderRow.Target.Value!.TerritoryType.Value!.PlaceName.Value!.Name,
  230. ExpansionName = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Value!
  231. .ExVersion.Value!.Name,
  232. ExpansionId = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Value!
  233. .ExVersion.Row,
  234. MapId = mobHuntOrderRow.Target.Value!.TerritoryType.Row,
  235. TerritoryType = mobHuntOrderRow.Target.Value!.TerritoryType.Value.TerritoryType.Row,
  236. MobHuntId = mobHuntOrderRow.Target.Value!.Name.Row,
  237. IsEliteMark = billNumber is BillEnum.ArrElite or BillEnum.HwElite or BillEnum.SbElite
  238. or BillEnum.ShbElite or BillEnum.EwElite,
  239. CurrentKillsOffset = (5 * (uint)billNumber) + mobHuntOrderRow.SubRowId,
  240. NeededKills = mobHuntOrderRow.NeededKills,
  241. Icon = Plugin.LoadIcon(mobHuntOrderRow.Target.Value.Icon)
  242. });
  243. }
  244. else
  245. {
  246. if (mobHuntEntry.NeededKills < mobHuntOrderRow.NeededKills)
  247. {
  248. mobHuntEntry.NeededKills = mobHuntOrderRow.NeededKills;
  249. }
  250. }
  251. }
  252. }
  253. foreach (var entry in mobHuntList)
  254. {
  255. var key = entry.ExpansionName ?? "Unknown";
  256. var subKey = new KeyValuePair<uint, string>(entry.TerritoryType, entry.TerritoryName ?? "Unknown");
  257. if (!this.MobHuntEntries.ContainsKey(key))
  258. {
  259. this.MobHuntEntries[key] = new Dictionary<KeyValuePair<uint, string>, List<MobHuntEntry>>();
  260. }
  261. if (!this.MobHuntEntries[key].ContainsKey(subKey))
  262. {
  263. this.MobHuntEntries[key][subKey] = new List<MobHuntEntry>();
  264. }
  265. this.MobHuntEntries[key][subKey].Add(entry);
  266. }
  267. this.ClientStateOnTerritoryChanged(null, 0);
  268. this.MobHuntEntriesReady = true;
  269. }
  270. private static TexFile? GetHdIcon(uint id)
  271. {
  272. var path = $"ui/icon/{id / 1000 * 1000:000000}/{id:000000}_hr1.tex";
  273. return Plugin.DataManager.GetFile<TexFile>(path);
  274. }
  275. private static TextureWrap LoadIcon(uint id)
  276. {
  277. var icon = Plugin.GetHdIcon(id) ?? Plugin.DataManager.GetIcon(id)!;
  278. var iconData = icon.GetRgbaImageData();
  279. return Plugin.PluginInterface.UiBuilder.LoadImageRaw(iconData, icon.Header.Width, icon.Header.Height, 4);
  280. }
  281. public void Dispose()
  282. {
  283. this.Dispose(true);
  284. GC.SuppressFinalize(this);
  285. }
  286. }
  287. }