|
|
|
@ -3,13 +3,11 @@ using System.Collections.Generic;
|
|
|
|
|
using System.Collections.Immutable;
|
|
|
|
|
using System.Collections.ObjectModel;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using Dalamud.Game.Addon.Events;
|
|
|
|
|
using Dalamud.Game.Addon.Lifecycle;
|
|
|
|
|
using Dalamud.Game.Addon.Lifecycle.AddonArgTypes;
|
|
|
|
|
using Dalamud.Game.Command;
|
|
|
|
|
using Dalamud.Game.Text.SeStringHandling;
|
|
|
|
|
using Dalamud.Interface.Colors;
|
|
|
|
|
using Dalamud.Interface.Style;
|
|
|
|
|
using Dalamud.Interface.Utility.Raii;
|
|
|
|
|
using Dalamud.Interface.Windowing;
|
|
|
|
|
using Dalamud.Plugin;
|
|
|
|
@ -28,6 +26,13 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
private const uint ItemMgp = 29;
|
|
|
|
|
private const uint ItemTrophyCrystals = 36656;
|
|
|
|
|
|
|
|
|
|
private static readonly (uint ItemId, string Name)[] AlliedSocietyCurrencies =
|
|
|
|
|
[
|
|
|
|
|
(21074, "Vanu Whitebone"),
|
|
|
|
|
(21079, "Black Copper Gil"),
|
|
|
|
|
(21081, "Kojin Sango"),
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
private static readonly ImmutableHashSet<uint> MgpMakaiSets = new HashSet<uint>
|
|
|
|
|
{
|
|
|
|
|
// makai gear
|
|
|
|
@ -84,6 +89,8 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
private readonly IAddonLifecycle _addonLifecycle;
|
|
|
|
|
private readonly Configuration _configuration;
|
|
|
|
|
private readonly ReadOnlyCollection<GlamourSet> _glamourSets;
|
|
|
|
|
|
|
|
|
|
private readonly Dictionary<uint, int> _ownedCurrencies = [];
|
|
|
|
|
private Configuration.CharacterData? _characterData;
|
|
|
|
|
|
|
|
|
|
public GlamourSetter(IDalamudPluginInterface pluginInterface, IDataManager dataManager, IClientState clientState,
|
|
|
|
@ -98,28 +105,34 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
_addonLifecycle = addonLifecycle;
|
|
|
|
|
_configuration = configuration;
|
|
|
|
|
|
|
|
|
|
SizeConstraints = new WindowSizeConstraints
|
|
|
|
|
{
|
|
|
|
|
MinimumSize = new(300, 400)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var armoireItems = dataManager.GetExcelSheet<Cabinet>()
|
|
|
|
|
.Where(x => x.RowId > 0)
|
|
|
|
|
.Select(x => x.Item.RowId)
|
|
|
|
|
.ToHashSet();
|
|
|
|
|
var specialShopItems = dataManager.GetExcelSheet<SpecialShop>()
|
|
|
|
|
.Where(x => x.RowId > 0 && !string.IsNullOrEmpty(x.Name.ToString()))
|
|
|
|
|
.SelectMany(x => x.Item.SelectMany(y =>
|
|
|
|
|
y.ReceiveItems.Select(z => new SpecialShopItem
|
|
|
|
|
{
|
|
|
|
|
ItemId = z.Item.RowId,
|
|
|
|
|
CostItemId = y.ItemCosts[0].ItemCost.Value.RowId,
|
|
|
|
|
CostType = y.ItemCosts[0].ItemCost.Value.ItemUICategory.RowId,
|
|
|
|
|
CostName = y.ItemCosts[0].ItemCost.Value.Name.ToString(),
|
|
|
|
|
CostQuantity = y.ItemCosts[0].CurrencyCost,
|
|
|
|
|
})
|
|
|
|
|
.Where(z => z.ItemId > 0 && (z.CostItemId < 100 || z.CostType == 100))))
|
|
|
|
|
.GroupBy(x => x.ItemId)
|
|
|
|
|
.ToDictionary(x => x.Key, x => x.FirstOrDefault());
|
|
|
|
|
var specialShopItems = BuildSpecialShopItems(dataManager);
|
|
|
|
|
_glamourSets = BuildGlamourSets(dataManager, armoireItems, specialShopItems);
|
|
|
|
|
|
|
|
|
|
_commandManager.AddHandler("/glamoursets", new CommandInfo(ProcessCommand)
|
|
|
|
|
{
|
|
|
|
|
HelpMessage = "Shows the glamour set tracker"
|
|
|
|
|
});
|
|
|
|
|
_addonLifecycle.RegisterListener(AddonEvent.PostRefresh, "MiragePrismPrismBox", UpdateFromGlamourDresser);
|
|
|
|
|
_clientState.Logout += Reset;
|
|
|
|
|
|
|
|
|
|
if (_clientState.IsLoggedIn)
|
|
|
|
|
Update();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static ReadOnlyCollection<GlamourSet> BuildGlamourSets(IDataManager dataManager, HashSet<uint> armoireItems,
|
|
|
|
|
Dictionary<uint, SpecialShopItem> specialShopItems)
|
|
|
|
|
{
|
|
|
|
|
ExcelSheet<Item> itemSheet = dataManager.GetExcelSheet<Item>();
|
|
|
|
|
_glamourSets = dataManager.GetExcelSheet<MirageStoreSetItem>()
|
|
|
|
|
return dataManager.GetExcelSheet<MirageStoreSetItem>()
|
|
|
|
|
.Where(x => x.RowId > 0)
|
|
|
|
|
.Select(x =>
|
|
|
|
|
{
|
|
|
|
@ -154,20 +167,7 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
ItemId = x.RowId,
|
|
|
|
|
Name = itemSheet.GetRow(x.RowId).Name.ToString(),
|
|
|
|
|
Items = items,
|
|
|
|
|
SetType =
|
|
|
|
|
x.RowId == MostRecentPvpSet
|
|
|
|
|
? ESetType.PvP
|
|
|
|
|
: UnobtainableSets.Contains(x.RowId)
|
|
|
|
|
? ESetType.Unobtainable
|
|
|
|
|
: EternalBondingSets.Contains(x.RowId) || UndyedRathalosSets.Contains(x.RowId) || MgpMakaiSets.Contains(x.RowId)
|
|
|
|
|
? ESetType.Special
|
|
|
|
|
: items.FirstOrDefault()?.ShopItem?.CostItemId switch
|
|
|
|
|
{
|
|
|
|
|
ItemWolfMarks or ItemTrophyCrystals => ESetType.PvP,
|
|
|
|
|
ItemMgp => ESetType.MGP,
|
|
|
|
|
> 100 => ESetType.AlliedSociety,
|
|
|
|
|
_ => ESetType.Default,
|
|
|
|
|
},
|
|
|
|
|
SetType = DetermineSetType(x, items),
|
|
|
|
|
};
|
|
|
|
|
})
|
|
|
|
|
.Where(x => x.Items.Count > 0 && x.Items.Any(y => !armoireItems.Contains(y.ItemId)))
|
|
|
|
@ -175,16 +175,49 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
.ThenBy(x => x.ItemId)
|
|
|
|
|
.ToList()
|
|
|
|
|
.AsReadOnly();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_commandManager.AddHandler("/glamoursets", new CommandInfo(ProcessCommand)
|
|
|
|
|
private static ESetType DetermineSetType(MirageStoreSetItem item, ReadOnlyCollection<GlamourItem> items)
|
|
|
|
|
{
|
|
|
|
|
if (item.RowId == MostRecentPvpSet)
|
|
|
|
|
return ESetType.PvP;
|
|
|
|
|
|
|
|
|
|
if (UnobtainableSets.Contains(item.RowId))
|
|
|
|
|
return ESetType.Unobtainable;
|
|
|
|
|
|
|
|
|
|
if (EternalBondingSets.Contains(item.RowId) ||
|
|
|
|
|
UndyedRathalosSets.Contains(item.RowId) ||
|
|
|
|
|
MgpMakaiSets.Contains(item.RowId))
|
|
|
|
|
return ESetType.Special;
|
|
|
|
|
|
|
|
|
|
uint? costItemId = items.FirstOrDefault()?.ShopItem?.CostItemId;
|
|
|
|
|
if (AlliedSocietyCurrencies.Any(x => x.ItemId == costItemId))
|
|
|
|
|
return ESetType.AlliedSociety;
|
|
|
|
|
|
|
|
|
|
return costItemId switch
|
|
|
|
|
{
|
|
|
|
|
HelpMessage = "Shows the glamour set tracker"
|
|
|
|
|
});
|
|
|
|
|
_addonLifecycle.RegisterListener(AddonEvent.PostRefresh, "MiragePrismPrismBox", UpdateFromGlamourDresser);
|
|
|
|
|
_clientState.Logout += Reset;
|
|
|
|
|
ItemWolfMarks or ItemTrophyCrystals => ESetType.PvP,
|
|
|
|
|
ItemMgp => ESetType.MGP,
|
|
|
|
|
_ => ESetType.Default,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_clientState.IsLoggedIn)
|
|
|
|
|
Update();
|
|
|
|
|
private static Dictionary<uint, SpecialShopItem> BuildSpecialShopItems(IDataManager dataManager)
|
|
|
|
|
{
|
|
|
|
|
return dataManager.GetExcelSheet<SpecialShop>()
|
|
|
|
|
.Where(x => x.RowId > 0 && !string.IsNullOrEmpty(x.Name.ToString()))
|
|
|
|
|
.SelectMany(x => x.Item.SelectMany(y =>
|
|
|
|
|
y.ReceiveItems.Select(z => new SpecialShopItem
|
|
|
|
|
{
|
|
|
|
|
ItemId = z.Item.RowId,
|
|
|
|
|
CostItemId = y.ItemCosts[0].ItemCost.Value.RowId,
|
|
|
|
|
CostType = y.ItemCosts[0].ItemCost.Value.ItemUICategory.RowId,
|
|
|
|
|
CostName = y.ItemCosts[0].ItemCost.Value.Name.ToString(),
|
|
|
|
|
CostQuantity = y.ItemCosts[0].CurrencyCost,
|
|
|
|
|
})
|
|
|
|
|
.Where(z => z.ItemId > 0 && (z.CostItemId < 100 || z.CostType == 100))))
|
|
|
|
|
.GroupBy(x => x.ItemId)
|
|
|
|
|
.ToDictionary(x => x.Key, x => x.First());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ProcessCommand(string command, string arguments)
|
|
|
|
@ -227,6 +260,21 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
|
|
|
|
InventoryManager* inventoryManager = InventoryManager.Instance();
|
|
|
|
|
if (inventoryManager != null)
|
|
|
|
|
{
|
|
|
|
|
_ownedCurrencies[ItemMgp] = inventoryManager->GetItemCountInContainer(ItemMgp, InventoryType.Currency);
|
|
|
|
|
_ownedCurrencies[ItemWolfMarks] = (int)inventoryManager->GetWolfMarks();
|
|
|
|
|
_ownedCurrencies[ItemTrophyCrystals] = inventoryManager->GetInventoryItemCount(ItemTrophyCrystals);
|
|
|
|
|
foreach (var (itemId, _) in AlliedSocietyCurrencies)
|
|
|
|
|
_ownedCurrencies[itemId] = inventoryManager->GetInventoryItemCount(itemId);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
_ownedCurrencies.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var ownedSets = _glamourSets.Where(x => _characterData.GlamourDresserItems.Contains(x.ItemId)).ToList();
|
|
|
|
|
ImGui.Text(
|
|
|
|
|
$"Complete Sets: {ownedSets.Count} / {_glamourSets.Count(x => x.SetType != ESetType.Unobtainable || ownedSets.Contains(x))}");
|
|
|
|
@ -267,7 +315,9 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
|
|
|
|
|
var ownedItems = GetOwnedItems();
|
|
|
|
|
DrawMissingItemHeader(glamourSets, setType, ownedSets, ownedItems);
|
|
|
|
|
DrawSetRange(glamourSets, ownedSets, ownedItems);
|
|
|
|
|
|
|
|
|
|
using (ImRaii.Child("Sets"))
|
|
|
|
|
DrawSetRange(glamourSets, ownedSets, ownedItems);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void DrawSpecialtyTab(List<GlamourSet> ownedSets)
|
|
|
|
@ -290,7 +340,7 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
DrawSetRange(glamourSets.Where(x => UndyedRathalosSets.Contains(x.ItemId)).ToList(), ownedSets, ownedItems);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void DrawMissingItemHeader(List<GlamourSet> glamourSets, ESetType setType, List<GlamourSet> ownedSets,
|
|
|
|
|
private void DrawMissingItemHeader(List<GlamourSet> glamourSets, ESetType setType, List<GlamourSet> ownedSets,
|
|
|
|
|
HashSet<uint> ownedItems)
|
|
|
|
|
{
|
|
|
|
|
var missingItems = glamourSets
|
|
|
|
@ -300,15 +350,24 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
if (setType == ESetType.PvP)
|
|
|
|
|
{
|
|
|
|
|
ImGui.Text(
|
|
|
|
|
$"Wolf Marks: {missingItems.Where(x => x is { ShopItem.CostItemId: ItemWolfMarks }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
$"Wolf Marks: {_ownedCurrencies.GetValueOrDefault(ItemWolfMarks):N0} / {missingItems.Where(x => x is { ShopItem.CostItemId: ItemWolfMarks }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
ImGui.Text(
|
|
|
|
|
$"Trophy Crystals: {missingItems.Where(x => x is { ShopItem.CostItemId: ItemTrophyCrystals }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
$"Trophy Crystals: {_ownedCurrencies.GetValueOrDefault(ItemTrophyCrystals):N0} / {missingItems.Where(x => x is { ShopItem.CostItemId: ItemTrophyCrystals }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
ImGui.Separator();
|
|
|
|
|
}
|
|
|
|
|
else if (setType is ESetType.MGP or ESetType.Special)
|
|
|
|
|
{
|
|
|
|
|
ImGui.Text(
|
|
|
|
|
$"MGP: {missingItems.Where(x => x is { ShopItem.CostItemId: ItemMgp }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
$"MGP: {_ownedCurrencies.GetValueOrDefault(ItemMgp):N0} / {missingItems.Where(x => x is { ShopItem.CostItemId: ItemMgp }).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
ImGui.Separator();
|
|
|
|
|
}
|
|
|
|
|
else if (setType == ESetType.AlliedSociety)
|
|
|
|
|
{
|
|
|
|
|
foreach (var (itemId, name) in AlliedSocietyCurrencies)
|
|
|
|
|
{
|
|
|
|
|
ImGui.Text(
|
|
|
|
|
$"{name}: {_ownedCurrencies.GetValueOrDefault(itemId):N0} / {missingItems.Where(x => x is { ShopItem: { } shopItem } && shopItem.CostItemId == itemId).Sum(x => x.ShopItem!.CostQuantity):N0}");
|
|
|
|
|
}
|
|
|
|
|
ImGui.Separator();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -325,12 +384,13 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
int ownedCount = glamourSet.Items.Count(x => ownedItems.Contains(x.ItemId));
|
|
|
|
|
if (ownedCount == glamourSet.Items.Count)
|
|
|
|
|
ImGui.TextColored(ImGuiColors.ParsedBlue, $"{glamourSet.Name} (Can be completed)");
|
|
|
|
|
else if (CanAffordAllMissingGearPieces(glamourSet, ownedItems))
|
|
|
|
|
ImGui.TextColored(ImGuiColors.DalamudViolet, $"{glamourSet.Name} (Can afford)");
|
|
|
|
|
else if (ownedCount > 0)
|
|
|
|
|
ImGui.TextColored(ImGuiColors.DalamudYellow, glamourSet.Name);
|
|
|
|
|
else
|
|
|
|
|
ImGui.Text(glamourSet.Name);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
using (ImRaii.PushIndent())
|
|
|
|
|
{
|
|
|
|
|
foreach (var item in glamourSet.Items)
|
|
|
|
@ -371,6 +431,9 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
foreach (var inventoryType in _inventoryTypes)
|
|
|
|
|
{
|
|
|
|
|
var inventoryContainer = inventoryManager->GetInventoryContainer(inventoryType);
|
|
|
|
|
if (inventoryContainer == null)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < inventoryContainer->Size; ++i)
|
|
|
|
|
{
|
|
|
|
|
InventoryItem* item = inventoryContainer->GetInventorySlot(i);
|
|
|
|
@ -384,6 +447,25 @@ internal sealed class GlamourSetter : Window, IDisposable
|
|
|
|
|
return ownedItems;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool CanAffordAllMissingGearPieces(GlamourSet glamourSet, HashSet<uint> ownedItems)
|
|
|
|
|
{
|
|
|
|
|
uint costItemId = 0;
|
|
|
|
|
uint costQuantity = 0;
|
|
|
|
|
foreach (var item in glamourSet.Items)
|
|
|
|
|
{
|
|
|
|
|
if (ownedItems.Contains(item.ItemId))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (item.ShopItem == null)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
costItemId = item.ShopItem.CostItemId;
|
|
|
|
|
costQuantity += item.ShopItem.CostQuantity;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return costQuantity <= _ownedCurrencies.GetValueOrDefault(costItemId);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private unsafe void UpdateFromGlamourDresser(AddonEvent type, AddonArgs args)
|
|
|
|
|
{
|
|
|
|
|
if (_characterData == null)
|
|
|
|
|