forked from liza/ARControl
Initial Commit
This commit is contained in:
commit
161bf7e39c
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
/.idea
|
||||
*.user
|
16
ARControl.sln
Normal file
16
ARControl.sln
Normal file
@ -0,0 +1,16 @@
|
||||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ARControl", "ARControl\ARControl.csproj", "{B33BF820-56C2-45A1-AEEC-3DCF526DBF42}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{B33BF820-56C2-45A1-AEEC-3DCF526DBF42}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{B33BF820-56C2-45A1-AEEC-3DCF526DBF42}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{B33BF820-56C2-45A1-AEEC-3DCF526DBF42}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{B33BF820-56C2-45A1-AEEC-3DCF526DBF42}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
EndGlobal
|
3
ARControl/.gitignore
vendored
Normal file
3
ARControl/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
/dist
|
||||
/obj
|
||||
/bin
|
83
ARControl/ARControl.csproj
Normal file
83
ARControl/ARControl.csproj
Normal file
@ -0,0 +1,83 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net7.0-windows</TargetFramework>
|
||||
<Version>1.0</Version>
|
||||
<LangVersion>11.0</LangVersion>
|
||||
<Nullable>enable</Nullable>
|
||||
<CopyLocalLockFileAssemblies>true</CopyLocalLockFileAssemblies>
|
||||
<AppendTargetFrameworkToOutputPath>false</AppendTargetFrameworkToOutputPath>
|
||||
<AppendRuntimeIdentifierToOutputPath>false</AppendRuntimeIdentifierToOutputPath>
|
||||
<OutputPath>dist</OutputPath>
|
||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
||||
<DebugType>portable</DebugType>
|
||||
<PathMap Condition="$(SolutionDir) != ''">$(SolutionDir)=X:\</PathMap>
|
||||
<RestorePackagesWithLockFile>true</RestorePackagesWithLockFile>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<DalamudLibPath>$(appdata)\XIVLauncher\addon\Hooks\dev\</DalamudLibPath>
|
||||
<AutoRetainerLibPath>$(appdata)\XIVLauncher\installedPlugins\AutoRetainer\4.1.2.5\</AutoRetainerLibPath>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup Condition="'$([System.Runtime.InteropServices.RuntimeInformation]::IsOSPlatform($([System.Runtime.InteropServices.OSPlatform]::Linux)))'">
|
||||
<DalamudLibPath>$(DALAMUD_HOME)/</DalamudLibPath>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Dalamud.ContextMenu" Version="1.2.3"/>
|
||||
<PackageReference Include="DalamudPackager" Version="2.1.11"/>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Reference Include="Dalamud">
|
||||
<HintPath>$(DalamudLibPath)Dalamud.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="ImGui.NET">
|
||||
<HintPath>$(DalamudLibPath)ImGui.NET.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="ImGuiScene">
|
||||
<HintPath>$(DalamudLibPath)ImGuiScene.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="Lumina">
|
||||
<HintPath>$(DalamudLibPath)Lumina.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="Lumina.Excel">
|
||||
<HintPath>$(DalamudLibPath)Lumina.Excel.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="Newtonsoft.Json">
|
||||
<HintPath>$(DalamudLibPath)Newtonsoft.Json.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="FFXIVClientStructs">
|
||||
<HintPath>$(DalamudLibPath)FFXIVClientStructs.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="FFXIVClientStructs">
|
||||
<HintPath>$(DalamudLibPath)FFXIVClientStructs.dll</HintPath>
|
||||
<Private>false</Private>
|
||||
</Reference>
|
||||
<Reference Include="AutoRetainerAPI">
|
||||
<HintPath>$(AutoRetainerLibPath)AutoRetainerAPI.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="ECommons">
|
||||
<HintPath>$(AutoRetainerLibPath)ECommons.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="ClickLib">
|
||||
<HintPath>$(AutoRetainerLibPath)ClickLib.dll</HintPath>
|
||||
</Reference>
|
||||
</ItemGroup>
|
||||
|
||||
|
||||
<ItemGroup>
|
||||
<Folder Include="External\" />
|
||||
</ItemGroup>
|
||||
|
||||
<Target Name="RenameLatestZip" AfterTargets="PackagePlugin">
|
||||
<Exec Command="rename $(OutDir)$(AssemblyName)\latest.zip $(AssemblyName)-$(Version).zip"/>
|
||||
</Target>
|
||||
</Project>
|
7
ARControl/ARControl.json
Normal file
7
ARControl/ARControl.json
Normal file
@ -0,0 +1,7 @@
|
||||
{
|
||||
"Name": "ARC",
|
||||
"Author": "Liza Carvelli",
|
||||
"Punchline": "Better AutoRetainer Venture Distribution",
|
||||
"Description": "",
|
||||
"RepoUrl": "https://git.carvel.li/liza/ARControl"
|
||||
}
|
303
ARControl/AutoRetainerControlPlugin.cs
Normal file
303
ARControl/AutoRetainerControlPlugin.cs
Normal file
@ -0,0 +1,303 @@
|
||||
using System;
|
||||
using System.Linq;
|
||||
using ARControl.GameData;
|
||||
using ARControl.Windows;
|
||||
using AutoRetainerAPI;
|
||||
using Dalamud.Data;
|
||||
using Dalamud.Game.ClientState;
|
||||
using Dalamud.Game.Command;
|
||||
using Dalamud.Game.Gui;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Components;
|
||||
using Dalamud.Interface.Windowing;
|
||||
using Dalamud.Logging;
|
||||
using Dalamud.Plugin;
|
||||
using ECommons;
|
||||
using ImGuiNET;
|
||||
|
||||
namespace ARControl;
|
||||
|
||||
public sealed class AutoRetainerControlPlugin : IDalamudPlugin
|
||||
{
|
||||
private readonly WindowSystem _windowSystem = new(nameof(AutoRetainerControlPlugin));
|
||||
|
||||
private readonly DalamudPluginInterface _pluginInterface;
|
||||
private readonly DataManager _dataManager;
|
||||
private readonly ClientState _clientState;
|
||||
private readonly ChatGui _chatGui;
|
||||
private readonly CommandManager _commandManager;
|
||||
|
||||
private readonly Configuration _configuration;
|
||||
private readonly GameCache _gameCache;
|
||||
private readonly ConfigWindow _configWindow;
|
||||
private readonly AutoRetainerApi _autoRetainerApi;
|
||||
|
||||
public AutoRetainerControlPlugin(DalamudPluginInterface pluginInterface, DataManager dataManager,
|
||||
ClientState clientState, ChatGui chatGui, CommandManager commandManager)
|
||||
{
|
||||
_pluginInterface = pluginInterface;
|
||||
_dataManager = dataManager;
|
||||
_clientState = clientState;
|
||||
_chatGui = chatGui;
|
||||
_commandManager = commandManager;
|
||||
|
||||
_configuration = (Configuration?)_pluginInterface.GetPluginConfig() ?? new Configuration();
|
||||
|
||||
_gameCache = new GameCache(_dataManager);
|
||||
_configWindow = new ConfigWindow(_pluginInterface, _configuration, _gameCache, _clientState, _commandManager);
|
||||
_windowSystem.AddWindow(_configWindow);
|
||||
|
||||
ECommonsMain.Init(_pluginInterface, this);
|
||||
_autoRetainerApi = new();
|
||||
|
||||
_pluginInterface.UiBuilder.Draw += _windowSystem.Draw;
|
||||
_pluginInterface.UiBuilder.OpenConfigUi += _configWindow.Toggle;
|
||||
_autoRetainerApi.OnSendRetainerToVenture += SendRetainerToVenture;
|
||||
_autoRetainerApi.OnRetainerPostVentureTaskDraw += RetainerTaskButtonDraw;
|
||||
_clientState.TerritoryChanged += TerritoryChanged;
|
||||
_commandManager.AddHandler("/arc", new CommandInfo(ProcessCommand));
|
||||
|
||||
if (_autoRetainerApi.Ready)
|
||||
Sync();
|
||||
}
|
||||
|
||||
public string Name => "ARC";
|
||||
|
||||
private void SendRetainerToVenture(string retainerName)
|
||||
{
|
||||
var ch = _configuration.Characters.SingleOrDefault(x => x.LocalContentId == _clientState.LocalContentId);
|
||||
if (ch == null)
|
||||
{
|
||||
PluginLog.Information("No character information found");
|
||||
}
|
||||
else if (!ch.Managed)
|
||||
{
|
||||
PluginLog.Information("Character is not managed");
|
||||
}
|
||||
else
|
||||
{
|
||||
var retainer = ch.Retainers.SingleOrDefault(x => x.Name == retainerName);
|
||||
if (retainer == null)
|
||||
{
|
||||
PluginLog.Information("No retainer information found");
|
||||
}
|
||||
else if (!retainer.Managed)
|
||||
{
|
||||
PluginLog.Information("Retainer is not managed");
|
||||
}
|
||||
else
|
||||
{
|
||||
PluginLog.Information("Checking tasks...");
|
||||
Sync();
|
||||
foreach (var queuedItem in _configuration.QueuedItems.Where(x => x.RemainingQuantity > 0))
|
||||
{
|
||||
PluginLog.Information($"Checking venture info for itemId {queuedItem.ItemId}");
|
||||
var venture = _gameCache.Ventures
|
||||
.Where(x => retainer.Level >= x.Level)
|
||||
.FirstOrDefault(x => x.ItemId == queuedItem.ItemId && x.MatchesJob(retainer.Job));
|
||||
if (venture == null)
|
||||
{
|
||||
PluginLog.Information($"No applicable venture found for itemId {queuedItem.ItemId}");
|
||||
}
|
||||
else
|
||||
{
|
||||
var itemToGather = _gameCache.ItemsToGather.FirstOrDefault(x => x.ItemId == queuedItem.ItemId);
|
||||
if (itemToGather != null && !ch.GatheredItems.Contains(itemToGather.GatheredItemId))
|
||||
{
|
||||
PluginLog.Information($"Character hasn't gathered {venture.Name} yet");
|
||||
}
|
||||
else
|
||||
{
|
||||
PluginLog.Information(
|
||||
$"Found venture {venture.Name}, row = {venture.RowId}, checking if it is suitable");
|
||||
VentureReward? reward = null;
|
||||
if (venture.CategoryName is "MIN" or "BTN")
|
||||
{
|
||||
if (retainer.Gathering >= venture.RequiredGathering)
|
||||
reward = venture.Rewards.Last(
|
||||
x => retainer.Perception >= x.PerceptionMinerBotanist);
|
||||
}
|
||||
else if (venture.CategoryName == "FSH")
|
||||
{
|
||||
if (retainer.Gathering >= venture.RequiredGathering)
|
||||
reward = venture.Rewards.Last(
|
||||
x => retainer.Perception >= x.PerceptionFisher);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (retainer.ItemLevel >= venture.ItemLevelCombat)
|
||||
reward = venture.Rewards.Last(
|
||||
x => retainer.ItemLevel >= x.ItemLevelCombat);
|
||||
}
|
||||
|
||||
if (reward == null)
|
||||
{
|
||||
PluginLog.Information(
|
||||
"Retainer doesn't have enough stats for the venture and would return no items");
|
||||
}
|
||||
else
|
||||
{
|
||||
_chatGui.Print(
|
||||
$"ARC → Overriding venture to collect {reward.Quantity}x {venture.Name}.");
|
||||
PluginLog.Information(
|
||||
$"Setting AR to use venture {venture.RowId}, which should retrieve {reward.Quantity}x {venture.Name}");
|
||||
_autoRetainerApi.SetVenture(venture.RowId);
|
||||
|
||||
queuedItem.RemainingQuantity =
|
||||
Math.Max(0, queuedItem.RemainingQuantity - reward.Quantity);
|
||||
_pluginInterface.SavePluginConfig(_configuration);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// fallback: managed but no venture found
|
||||
if (retainer.LastVenture != 395)
|
||||
{
|
||||
_chatGui.Print("ARC → No tasks left, using QC");
|
||||
PluginLog.Information($"No tasks left (previous venture = {retainer.LastVenture}), using QC");
|
||||
_autoRetainerApi.SetVenture(395);
|
||||
}
|
||||
else
|
||||
PluginLog.Information("Not changing venture plan, already 395");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void RetainerTaskButtonDraw(ulong characterId, string retainerName)
|
||||
{
|
||||
Configuration.CharacterConfiguration? characterConfiguration = _configuration.Characters.FirstOrDefault(x => x.LocalContentId == characterId);
|
||||
if (characterConfiguration is not { Managed: true })
|
||||
return;
|
||||
|
||||
Configuration.RetainerConfiguration? retainer = characterConfiguration.Retainers.FirstOrDefault(x => x.Name == retainerName);
|
||||
if (retainer is not { Managed: true })
|
||||
return;
|
||||
|
||||
ImGui.SameLine();
|
||||
ImGuiComponents.IconButton(FontAwesomeIcon.Book);
|
||||
}
|
||||
|
||||
private void TerritoryChanged(object? sender, ushort e) => Sync();
|
||||
|
||||
public void Sync()
|
||||
{
|
||||
bool save = false;
|
||||
|
||||
// FIXME This should have a way to get blacklisted character ids
|
||||
foreach (ulong registeredCharacterId in _autoRetainerApi.GetRegisteredCharacters())
|
||||
{
|
||||
PluginLog.Information($"ch → {registeredCharacterId:X}");
|
||||
var offlineCharacterData = _autoRetainerApi.GetOfflineCharacterData(registeredCharacterId);
|
||||
if (offlineCharacterData.ExcludeRetainer)
|
||||
continue;
|
||||
|
||||
var character = _configuration.Characters.SingleOrDefault(x => x.LocalContentId == registeredCharacterId);
|
||||
if (character == null)
|
||||
{
|
||||
character = new Configuration.CharacterConfiguration
|
||||
{
|
||||
LocalContentId = registeredCharacterId,
|
||||
CharacterName = offlineCharacterData.Name,
|
||||
WorldName = offlineCharacterData.World,
|
||||
Managed = false,
|
||||
};
|
||||
|
||||
save = true;
|
||||
_configuration.Characters.Add(character);
|
||||
}
|
||||
|
||||
if (character.GatheredItems != offlineCharacterData.UnlockedGatheringItems)
|
||||
{
|
||||
character.GatheredItems = offlineCharacterData.UnlockedGatheringItems;
|
||||
save = true;
|
||||
}
|
||||
|
||||
foreach (var retainerData in offlineCharacterData.RetainerData)
|
||||
{
|
||||
var retainer = character.Retainers.SingleOrDefault(x => x.Name == retainerData.Name);
|
||||
if (retainer == null)
|
||||
{
|
||||
retainer = new Configuration.RetainerConfiguration
|
||||
{
|
||||
Name = retainerData.Name,
|
||||
Managed = false,
|
||||
};
|
||||
|
||||
save = true;
|
||||
character.Retainers.Add(retainer);
|
||||
}
|
||||
|
||||
if (retainer.DisplayOrder != retainerData.DisplayOrder)
|
||||
{
|
||||
retainer.DisplayOrder = retainerData.DisplayOrder;
|
||||
save = true;
|
||||
}
|
||||
|
||||
if (retainer.Level != retainerData.Level)
|
||||
{
|
||||
retainer.Level = retainerData.Level;
|
||||
save = true;
|
||||
}
|
||||
|
||||
if (retainer.Job != retainerData.Job)
|
||||
{
|
||||
retainer.Job = retainerData.Job;
|
||||
save = true;
|
||||
}
|
||||
|
||||
if (retainer.LastVenture != retainerData.VentureID)
|
||||
{
|
||||
retainer.LastVenture = retainerData.VentureID;
|
||||
save = true;
|
||||
}
|
||||
|
||||
var additionalData =
|
||||
_autoRetainerApi.GetAdditionalRetainerData(registeredCharacterId, retainerData.Name);
|
||||
if (retainer.ItemLevel != additionalData.Ilvl)
|
||||
{
|
||||
retainer.ItemLevel = additionalData.Ilvl;
|
||||
save = true;
|
||||
}
|
||||
|
||||
if (retainer.Gathering != additionalData.Gathering)
|
||||
{
|
||||
retainer.Gathering = additionalData.Gathering;
|
||||
save = true;
|
||||
}
|
||||
|
||||
if (retainer.Perception != additionalData.Perception)
|
||||
{
|
||||
retainer.Perception = additionalData.Perception;
|
||||
save = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (save)
|
||||
_pluginInterface.SavePluginConfig(_configuration);
|
||||
}
|
||||
|
||||
private void ProcessCommand(string command, string arguments)
|
||||
{
|
||||
if (arguments == "sync")
|
||||
Sync();
|
||||
else
|
||||
_configWindow.Toggle();
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_commandManager.RemoveHandler("/arc");
|
||||
_clientState.TerritoryChanged -= TerritoryChanged;
|
||||
_autoRetainerApi.OnRetainerPostVentureTaskDraw -= RetainerTaskButtonDraw;
|
||||
_autoRetainerApi.OnSendRetainerToVenture -= SendRetainerToVenture;
|
||||
_pluginInterface.UiBuilder.OpenConfigUi -= _configWindow.Toggle;
|
||||
_pluginInterface.UiBuilder.Draw -= _windowSystem.Draw;
|
||||
|
||||
_autoRetainerApi.Dispose();
|
||||
ECommonsMain.Dispose();
|
||||
|
||||
}
|
||||
}
|
45
ARControl/Configuration.cs
Normal file
45
ARControl/Configuration.cs
Normal file
@ -0,0 +1,45 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using Dalamud.Configuration;
|
||||
|
||||
namespace ARControl;
|
||||
|
||||
internal sealed class Configuration : IPluginConfiguration
|
||||
{
|
||||
public int Version { get; set; } = 1;
|
||||
|
||||
public List<QueuedItem> QueuedItems { get; set; } = new();
|
||||
public List<CharacterConfiguration> Characters { get; set; } = new();
|
||||
|
||||
public sealed class QueuedItem
|
||||
{
|
||||
public required uint ItemId { get; set; }
|
||||
public required int RemainingQuantity { get; set; }
|
||||
}
|
||||
|
||||
public sealed class CharacterConfiguration
|
||||
{
|
||||
public required ulong LocalContentId { get; set; }
|
||||
public required string CharacterName { get; set; }
|
||||
public required string WorldName { get; set; }
|
||||
public required bool Managed { get; set; }
|
||||
|
||||
public List<RetainerConfiguration> Retainers { get; set; } = new();
|
||||
public HashSet<uint> GatheredItems { get; set; } = new();
|
||||
|
||||
public override string ToString() => $"{CharacterName} @ {WorldName}";
|
||||
}
|
||||
|
||||
public sealed class RetainerConfiguration
|
||||
{
|
||||
public required string Name { get; set; }
|
||||
public required bool Managed { get; set; }
|
||||
public int DisplayOrder { get; set; }
|
||||
public int Level { get; set; }
|
||||
public uint Job { get; set; }
|
||||
public uint LastVenture { get; set; }
|
||||
public int ItemLevel { get; set; }
|
||||
public int Gathering { get; set; }
|
||||
public int Perception { get; set; }
|
||||
}
|
||||
}
|
21
ARControl/DalamudPackager.targets
Normal file
21
ARControl/DalamudPackager.targets
Normal file
@ -0,0 +1,21 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project>
|
||||
<Target Name="PackagePlugin" AfterTargets="Build" Condition="'$(Configuration)' == 'Debug'">
|
||||
<DalamudPackager
|
||||
ProjectDir="$(ProjectDir)"
|
||||
OutputPath="$(OutputPath)"
|
||||
AssemblyName="$(AssemblyName)"
|
||||
MakeZip="false"
|
||||
VersionComponents="2"/>
|
||||
</Target>
|
||||
|
||||
<Target Name="PackagePlugin" AfterTargets="Build" Condition="'$(Configuration)' == 'Release'">
|
||||
<DalamudPackager
|
||||
ProjectDir="$(ProjectDir)"
|
||||
OutputPath="$(OutputPath)"
|
||||
AssemblyName="$(AssemblyName)"
|
||||
MakeZip="true"
|
||||
VersionComponents="2"
|
||||
Exclude="ARDiscard.deps.json;AutoRetainerAPI.pdb;ClickLib.pdb;ClickLib.xml;ECommons.pdb;ECommons.xml"/>
|
||||
</Target>
|
||||
</Project>
|
30
ARControl/GameData/GameCache.cs
Normal file
30
ARControl/GameData/GameCache.cs
Normal file
@ -0,0 +1,30 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using Dalamud.Data;
|
||||
using Lumina.Excel.GeneratedSheets;
|
||||
|
||||
namespace ARControl.GameData;
|
||||
|
||||
internal sealed class GameCache
|
||||
{
|
||||
public GameCache(DataManager dataManager)
|
||||
{
|
||||
Jobs = dataManager.GetExcelSheet<ClassJob>()!.ToDictionary(x => x.RowId, x => x.Abbreviation.ToString());
|
||||
Ventures = dataManager.GetExcelSheet<RetainerTask>()!
|
||||
.Where(x => x.RowId > 0 && !x.IsRandom && x.Task != 0)
|
||||
.Select(x => new Venture(dataManager, x))
|
||||
.ToList()
|
||||
.AsReadOnly();
|
||||
ItemsToGather = dataManager.GetExcelSheet<GatheringItem>()!
|
||||
.Where(x => x.RowId > 0 && x.RowId < 10_000 && x.Item != 0 && x.Quest.Row == 0)
|
||||
.Where(x => Ventures.Any(y => y.ItemId == x.Item))
|
||||
.Select(x => new ItemToGather(dataManager, x))
|
||||
.OrderBy(x => x.Name)
|
||||
.ToList()
|
||||
.AsReadOnly();
|
||||
}
|
||||
|
||||
public IReadOnlyDictionary<uint, string> Jobs { get; }
|
||||
public IReadOnlyList<Venture> Ventures { get; }
|
||||
public IReadOnlyList<ItemToGather> ItemsToGather { get; }
|
||||
}
|
19
ARControl/GameData/ItemToGather.cs
Normal file
19
ARControl/GameData/ItemToGather.cs
Normal file
@ -0,0 +1,19 @@
|
||||
using Dalamud.Data;
|
||||
using Lumina.Excel.GeneratedSheets;
|
||||
|
||||
namespace ARControl.GameData;
|
||||
|
||||
internal sealed class ItemToGather
|
||||
{
|
||||
public ItemToGather(DataManager dataManager, GatheringItem item)
|
||||
{
|
||||
GatheredItemId = item.RowId;
|
||||
ItemId = item.Item;
|
||||
Name = dataManager.GetExcelSheet<Item>()!.GetRow((uint)item.Item)!.Name.ToString();
|
||||
}
|
||||
|
||||
|
||||
public uint GatheredItemId { get; }
|
||||
public int ItemId { get; }
|
||||
public string Name { get; }
|
||||
}
|
93
ARControl/GameData/Venture.cs
Normal file
93
ARControl/GameData/Venture.cs
Normal file
@ -0,0 +1,93 @@
|
||||
using System.Collections.Generic;
|
||||
using Dalamud.Data;
|
||||
using Lumina.Excel.GeneratedSheets;
|
||||
|
||||
namespace ARControl.GameData;
|
||||
|
||||
internal sealed class Venture
|
||||
{
|
||||
public Venture(DataManager dataManager, RetainerTask retainerTask)
|
||||
{
|
||||
RowId = retainerTask.RowId;
|
||||
Category = retainerTask.ClassJobCategory.Value!;
|
||||
|
||||
var taskDetails = dataManager.GetExcelSheet<RetainerTaskNormal>()!.GetRow(retainerTask.Task)!;
|
||||
var taskParameters = retainerTask.RetainerTaskParameter.Value!;
|
||||
ItemId = taskDetails.Item.Row;
|
||||
Name = taskDetails.Item.Value!.Name.ToString();
|
||||
Level = retainerTask.RetainerLevel;
|
||||
ItemLevelCombat = retainerTask.RequiredItemLevel;
|
||||
RequiredGathering = retainerTask.RequiredGathering;
|
||||
Rewards = new List<VentureReward>
|
||||
{
|
||||
new VentureReward
|
||||
{
|
||||
Quantity = taskDetails.Quantity[0],
|
||||
ItemLevelCombat = 0,
|
||||
PerceptionMinerBotanist = 0,
|
||||
PerceptionFisher = 0,
|
||||
},
|
||||
new VentureReward
|
||||
{
|
||||
Quantity = taskDetails.Quantity[1],
|
||||
ItemLevelCombat = taskParameters.ItemLevelDoW[0],
|
||||
PerceptionMinerBotanist = taskParameters.PerceptionDoL[0],
|
||||
PerceptionFisher = taskParameters.PerceptionFSH[0],
|
||||
},
|
||||
new VentureReward
|
||||
{
|
||||
Quantity = taskDetails.Quantity[2],
|
||||
ItemLevelCombat = taskParameters.ItemLevelDoW[1],
|
||||
PerceptionMinerBotanist = taskParameters.PerceptionDoL[1],
|
||||
PerceptionFisher = taskParameters.PerceptionFSH[1],
|
||||
},
|
||||
new VentureReward
|
||||
{
|
||||
Quantity = taskDetails.Quantity[3],
|
||||
ItemLevelCombat = taskParameters.ItemLevelDoW[2],
|
||||
PerceptionMinerBotanist = taskParameters.PerceptionDoL[2],
|
||||
PerceptionFisher = taskParameters.PerceptionFSH[2],
|
||||
},
|
||||
new VentureReward
|
||||
{
|
||||
Quantity = taskDetails.Quantity[4],
|
||||
ItemLevelCombat = taskParameters.ItemLevelDoW[3],
|
||||
PerceptionMinerBotanist = taskParameters.PerceptionDoL[3],
|
||||
PerceptionFisher = taskParameters.PerceptionFSH[3],
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
public uint RowId { get; }
|
||||
public ClassJobCategory Category { get; }
|
||||
|
||||
public string? CategoryName
|
||||
{
|
||||
get
|
||||
{
|
||||
return Category.RowId switch
|
||||
{
|
||||
17 => "MIN",
|
||||
18 => "BTN",
|
||||
19 => "FSH",
|
||||
_ => "DoWM",
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
public uint ItemId { get; }
|
||||
public string Name { get; }
|
||||
public byte Level { get; }
|
||||
public ushort ItemLevelCombat { get; }
|
||||
public ushort RequiredGathering { get; set; }
|
||||
|
||||
public List<VentureReward> Rewards { get; }
|
||||
|
||||
public bool MatchesJob(uint job)
|
||||
{
|
||||
if (Category.RowId >= 17 && Category.RowId <= 19)
|
||||
return Category.RowId == job + 1;
|
||||
else
|
||||
return job is < 16 or > 18;
|
||||
}
|
||||
}
|
9
ARControl/GameData/VentureReward.cs
Normal file
9
ARControl/GameData/VentureReward.cs
Normal file
@ -0,0 +1,9 @@
|
||||
namespace ARControl.GameData;
|
||||
|
||||
internal sealed class VentureReward
|
||||
{
|
||||
public required byte Quantity { get; init; }
|
||||
public required short ItemLevelCombat { get; init; }
|
||||
public required short PerceptionMinerBotanist { get; init; }
|
||||
public required short PerceptionFisher { get; init; }
|
||||
}
|
408
ARControl/Windows/ConfigWindow.cs
Normal file
408
ARControl/Windows/ConfigWindow.cs
Normal file
@ -0,0 +1,408 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Numerics;
|
||||
using ARControl.GameData;
|
||||
using Dalamud.Game.ClientState;
|
||||
using Dalamud.Game.Command;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Colors;
|
||||
using Dalamud.Interface.Components;
|
||||
using Dalamud.Interface.Style;
|
||||
using Dalamud.Interface.Windowing;
|
||||
using Dalamud.Logging;
|
||||
using Dalamud.Plugin;
|
||||
using ECommons.ImGuiMethods;
|
||||
using ImGuiNET;
|
||||
|
||||
namespace ARControl.Windows;
|
||||
|
||||
internal sealed class ConfigWindow : Window
|
||||
{
|
||||
private const byte MaxLevel = 90;
|
||||
|
||||
private static readonly Vector4 ColorGreen = ImGuiColors.HealerGreen;
|
||||
private static readonly Vector4 ColorRed = ImGuiColors.DalamudRed;
|
||||
private static readonly Vector4 ColorGrey = ImGuiColors.DalamudGrey;
|
||||
|
||||
private readonly DalamudPluginInterface _pluginInterface;
|
||||
private readonly Configuration _configuration;
|
||||
private readonly GameCache _gameCache;
|
||||
private readonly ClientState _clientState;
|
||||
private readonly CommandManager _commandManager;
|
||||
|
||||
private string _searchString = string.Empty;
|
||||
private Configuration.QueuedItem? _dragDropSource;
|
||||
private bool _enableDragDrop;
|
||||
private bool _checkPerCharacter = true;
|
||||
private bool _onlyShowMissing = true;
|
||||
|
||||
public ConfigWindow(
|
||||
DalamudPluginInterface pluginInterface,
|
||||
Configuration configuration,
|
||||
GameCache gameCache,
|
||||
ClientState clientState,
|
||||
CommandManager commandManager)
|
||||
: base("ARC###ARControlConfig")
|
||||
{
|
||||
_pluginInterface = pluginInterface;
|
||||
_configuration = configuration;
|
||||
_gameCache = gameCache;
|
||||
_clientState = clientState;
|
||||
_commandManager = commandManager;
|
||||
}
|
||||
|
||||
public override void Draw()
|
||||
{
|
||||
if (ImGui.BeginTabBar("ARConfigTabs"))
|
||||
{
|
||||
DrawItemQueue();
|
||||
DrawCharacters();
|
||||
DrawGatheredItemsToCheck();
|
||||
ImGui.EndTabBar();
|
||||
}
|
||||
}
|
||||
|
||||
private unsafe void DrawItemQueue()
|
||||
{
|
||||
if (ImGui.BeginTabItem("Venture Queue"))
|
||||
{
|
||||
if (ImGui.BeginCombo("Venture...##VentureSelection", ""))
|
||||
{
|
||||
ImGuiEx.SetNextItemFullWidth();
|
||||
ImGui.InputTextWithHint("", "Filter...", ref _searchString, 256);
|
||||
|
||||
foreach (var ventures in _gameCache.Ventures
|
||||
.Where(x => x.Name.ToLower().Contains(_searchString.ToLower()))
|
||||
.OrderBy(x => x.Level)
|
||||
.ThenBy(x => x.Name)
|
||||
.ThenBy(x => x.ItemId)
|
||||
.GroupBy(x => x.ItemId))
|
||||
{
|
||||
var venture = ventures.First();
|
||||
if (ImGui.Selectable(
|
||||
$"{venture.Name} ({string.Join(" ", ventures.Select(x => x.CategoryName))})##SelectVenture{venture.RowId}"))
|
||||
{
|
||||
_configuration.QueuedItems.Add(new Configuration.QueuedItem
|
||||
{
|
||||
ItemId = venture.ItemId,
|
||||
RemainingQuantity = 0,
|
||||
});
|
||||
_searchString = string.Empty;
|
||||
Save();
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.EndCombo();
|
||||
}
|
||||
|
||||
ImGui.Checkbox("Enable Drag&Drop", ref _enableDragDrop);
|
||||
ImGui.Separator();
|
||||
|
||||
ImGui.Indent(30);
|
||||
|
||||
Configuration.QueuedItem? itemToRemove = null;
|
||||
Configuration.QueuedItem? itemToAdd = null;
|
||||
int indexToAdd = 0;
|
||||
for (int i = 0; i < _configuration.QueuedItems.Count; ++i)
|
||||
{
|
||||
var item = _configuration.QueuedItems[i];
|
||||
ImGui.PushID($"QueueItem{i}");
|
||||
var ventures = _gameCache.Ventures.Where(x => x.ItemId == item.ItemId).ToList();
|
||||
var venture = ventures.First();
|
||||
|
||||
if (!_enableDragDrop)
|
||||
{
|
||||
ImGui.SetNextItemWidth(130);
|
||||
int quantity = item.RemainingQuantity;
|
||||
if (ImGui.InputInt($"{venture.Name} ({string.Join(" ", ventures.Select(x => x.CategoryName))})",
|
||||
ref quantity, 100))
|
||||
{
|
||||
item.RemainingQuantity = quantity;
|
||||
Save();
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.Selectable($"{item.RemainingQuantity}x {venture.Name}");
|
||||
|
||||
if (ImGui.BeginDragDropSource())
|
||||
{
|
||||
ImGui.SetDragDropPayload("ArcDragDrop", nint.Zero, 0);
|
||||
_dragDropSource = item;
|
||||
|
||||
ImGui.EndDragDropSource();
|
||||
}
|
||||
|
||||
if (ImGui.BeginDragDropTarget())
|
||||
{
|
||||
if (_dragDropSource != null && ImGui.AcceptDragDropPayload("ArcDragDrop").NativePtr != null)
|
||||
{
|
||||
itemToAdd = _dragDropSource;
|
||||
indexToAdd = i;
|
||||
|
||||
_dragDropSource = null;
|
||||
}
|
||||
|
||||
ImGui.EndDragDropTarget();
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.OpenPopupOnItemClick($"###ctx{i}", ImGuiPopupFlags.MouseButtonRight);
|
||||
if (ImGui.BeginPopup($"###ctx{i}"))
|
||||
{
|
||||
if (ImGui.Selectable($"Remove {venture.Name}"))
|
||||
itemToRemove = item;
|
||||
|
||||
ImGui.EndPopup();
|
||||
}
|
||||
|
||||
ImGui.PopID();
|
||||
}
|
||||
|
||||
if (itemToRemove != null)
|
||||
{
|
||||
_configuration.QueuedItems.Remove(itemToRemove);
|
||||
Save();
|
||||
}
|
||||
|
||||
if (itemToAdd != null)
|
||||
{
|
||||
PluginLog.Information($"Updating {itemToAdd.ItemId} → {indexToAdd}");
|
||||
_configuration.QueuedItems.Remove(itemToAdd);
|
||||
_configuration.QueuedItems.Insert(indexToAdd, itemToAdd);
|
||||
Save();
|
||||
}
|
||||
|
||||
ImGui.Unindent(30);
|
||||
ImGui.EndTabItem();
|
||||
}
|
||||
}
|
||||
|
||||
private void DrawCharacters()
|
||||
{
|
||||
if (ImGui.BeginTabItem("Retainers"))
|
||||
{
|
||||
foreach (var world in _configuration.Characters
|
||||
.Where(x => x.Retainers.Any(y => y.Job != 0))
|
||||
.OrderBy(x => x.LocalContentId)
|
||||
.GroupBy(x => x.WorldName))
|
||||
{
|
||||
ImGui.CollapsingHeader(world.Key, ImGuiTreeNodeFlags.DefaultOpen | ImGuiTreeNodeFlags.OpenOnArrow | ImGuiTreeNodeFlags.Bullet);
|
||||
foreach (var character in world)
|
||||
{
|
||||
ImGui.PushID($"Char{character.LocalContentId}");
|
||||
|
||||
ImGui.PushItemWidth(ImGui.GetFontSize() * 30);
|
||||
Vector4 buttonColor = new Vector4();
|
||||
if (character.Managed && character.Retainers.Count > 0)
|
||||
{
|
||||
if (character.Retainers.All(x => x.Managed))
|
||||
buttonColor = ImGuiColors.HealerGreen;
|
||||
else if (character.Retainers.All(x => !x.Managed))
|
||||
buttonColor = ImGuiColors.DalamudRed;
|
||||
else
|
||||
buttonColor = ImGuiColors.DalamudOrange;
|
||||
}
|
||||
|
||||
if (ImGuiComponents.IconButton(FontAwesomeIcon.Book, buttonColor))
|
||||
{
|
||||
character.Managed = !character.Managed;
|
||||
Save();
|
||||
}
|
||||
|
||||
ImGui.SameLine();
|
||||
|
||||
if (ImGui.CollapsingHeader(
|
||||
$"{character.CharacterName} {(character.Managed ? $"({character.Retainers.Count(x => x.Managed)} / {character.Retainers.Count})" : "")}###{character.LocalContentId}"))
|
||||
{
|
||||
ImGui.Indent(30);
|
||||
foreach (var retainer in character.Retainers.Where(x => x.Job > 0).OrderBy(x => x.DisplayOrder))
|
||||
{
|
||||
ImGui.BeginDisabled(retainer.Level < MaxLevel);
|
||||
|
||||
bool managed = retainer.Managed && retainer.Level == MaxLevel;
|
||||
ImGui.Text(_gameCache.Jobs[retainer.Job]);
|
||||
ImGui.SameLine();
|
||||
if (ImGui.Checkbox($"{retainer.Name}###Retainer{retainer.Name}{retainer.DisplayOrder}",
|
||||
ref managed))
|
||||
{
|
||||
retainer.Managed = managed;
|
||||
Save();
|
||||
}
|
||||
|
||||
ImGui.EndDisabled();
|
||||
}
|
||||
|
||||
ImGui.Unindent(30);
|
||||
}
|
||||
|
||||
ImGui.PopID();
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.EndTabItem();
|
||||
}
|
||||
}
|
||||
|
||||
private void DrawGatheredItemsToCheck()
|
||||
{
|
||||
if (ImGui.BeginTabItem("Locked Items"))
|
||||
{
|
||||
ImGui.Checkbox("Group by character", ref _checkPerCharacter);
|
||||
ImGui.Checkbox("Only show missing items", ref _onlyShowMissing);
|
||||
ImGui.Separator();
|
||||
|
||||
var itemsToCheck =
|
||||
_configuration.QueuedItems
|
||||
.Select(x => x.ItemId)
|
||||
.Distinct()
|
||||
.Select(itemId => new
|
||||
{
|
||||
GatheredItem = _gameCache.ItemsToGather.SingleOrDefault(x => x.ItemId == itemId),
|
||||
Ventures = _gameCache.Ventures.Where(x => x.ItemId == itemId).ToList()
|
||||
})
|
||||
.Where(x => x.GatheredItem != null && x.Ventures.Count > 0)
|
||||
.Select(x => new CheckedItem
|
||||
{
|
||||
GatheredItem = x.GatheredItem!,
|
||||
Ventures = x.Ventures,
|
||||
ItemId = x.Ventures.First().ItemId,
|
||||
})
|
||||
.ToList();
|
||||
|
||||
var charactersToCheck = _configuration.Characters
|
||||
.Where(x => x.Managed)
|
||||
.OrderBy(x => x.WorldName)
|
||||
.ThenBy(x => x.LocalContentId)
|
||||
.Select(x => new CheckedCharacter(x, itemsToCheck))
|
||||
.ToList();
|
||||
|
||||
if (_checkPerCharacter)
|
||||
{
|
||||
foreach (var ch in charactersToCheck.Where(x => x.ToCheck(_onlyShowMissing).Any()))
|
||||
{
|
||||
bool currentCharacter = _clientState.LocalContentId == ch.Character.LocalContentId;
|
||||
ImGui.BeginDisabled(currentCharacter);
|
||||
if (ImGuiComponents.IconButton($"SwitchChacters{ch.Character.LocalContentId}",
|
||||
FontAwesomeIcon.DoorOpen))
|
||||
{
|
||||
_commandManager.ProcessCommand($"/ays relog {ch.Character.CharacterName}@{ch.Character.WorldName}");
|
||||
}
|
||||
|
||||
ImGui.EndDisabled();
|
||||
ImGui.SameLine();
|
||||
|
||||
if (currentCharacter)
|
||||
ImGui.PushStyleColor(ImGuiCol.Text, ImGuiColors.HealerGreen);
|
||||
|
||||
bool expanded = ImGui.CollapsingHeader($"{ch.Character}###GatheredCh{ch.Character.LocalContentId}");
|
||||
if (currentCharacter)
|
||||
ImGui.PopStyleColor();
|
||||
|
||||
if (expanded)
|
||||
{
|
||||
ImGui.Indent(30);
|
||||
foreach (var item in itemsToCheck.Where(x =>
|
||||
ch.ToCheck(_onlyShowMissing).ContainsKey(x.ItemId)))
|
||||
{
|
||||
var color = ch.Items[item.ItemId];
|
||||
if (color != ColorGrey)
|
||||
{
|
||||
ImGui.PushStyleColor(ImGuiCol.Text, color);
|
||||
if (currentCharacter && color == ColorRed)
|
||||
{
|
||||
ImGui.Selectable(item.GatheredItem.Name);
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left))
|
||||
{
|
||||
uint classJob = _clientState.LocalPlayer!.ClassJob.Id;
|
||||
if (classJob == 16)
|
||||
_commandManager.ProcessCommand($"/gathermin {item.GatheredItem.Name}");
|
||||
else if (classJob == 17)
|
||||
_commandManager.ProcessCommand($"/gatherbtn {item.GatheredItem.Name}");
|
||||
else if (classJob == 18)
|
||||
_commandManager.ProcessCommand($"/gatherfish {item.GatheredItem.Name}");
|
||||
else
|
||||
_commandManager.ProcessCommand($"/gather {item.GatheredItem.Name}");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.Text(item.GatheredItem.Name);
|
||||
}
|
||||
ImGui.PopStyleColor();
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.Unindent(30);
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach (var item in itemsToCheck.Where(x =>
|
||||
charactersToCheck.Any(y => y.ToCheck(_onlyShowMissing).ContainsKey(x.ItemId))))
|
||||
{
|
||||
if (ImGui.CollapsingHeader($"{item.GatheredItem.Name}##Gathered{item.GatheredItem.ItemId}"))
|
||||
{
|
||||
ImGui.Indent(30);
|
||||
foreach (var ch in charactersToCheck)
|
||||
{
|
||||
var color = ch.Items[item.ItemId];
|
||||
if (color == ColorRed || (color == ColorGreen && !_onlyShowMissing))
|
||||
ImGui.TextColored(color, ch.Character.ToString());
|
||||
}
|
||||
|
||||
ImGui.Unindent(30);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.EndTabItem();
|
||||
}
|
||||
}
|
||||
|
||||
private void Save()
|
||||
{
|
||||
_pluginInterface.SavePluginConfig(_configuration);
|
||||
}
|
||||
|
||||
private sealed class CheckedCharacter
|
||||
{
|
||||
public CheckedCharacter(Configuration.CharacterConfiguration character,
|
||||
List<CheckedItem> itemsToCheck)
|
||||
{
|
||||
Character = character;
|
||||
|
||||
foreach (var item in itemsToCheck)
|
||||
{
|
||||
bool enabled = character.Retainers.Any(x => item.Ventures.Any(v => v.MatchesJob(x.Job)));
|
||||
if (enabled)
|
||||
{
|
||||
if (character.GatheredItems.Contains(item.GatheredItem.GatheredItemId))
|
||||
Items[item.ItemId] = ColorGreen;
|
||||
else
|
||||
Items[item.ItemId] = ColorRed;
|
||||
}
|
||||
else
|
||||
Items[item.ItemId] = ColorGrey;
|
||||
}
|
||||
}
|
||||
|
||||
public Configuration.CharacterConfiguration Character { get; }
|
||||
public Dictionary<uint, Vector4> Items { get; } = new();
|
||||
|
||||
public Dictionary<uint, Vector4> ToCheck(bool onlyShowMissing)
|
||||
{
|
||||
return Items
|
||||
.Where(x => x.Value == ColorRed || (x.Value == ColorGreen && !onlyShowMissing))
|
||||
.ToDictionary(x => x.Key, x => x.Value);
|
||||
}
|
||||
}
|
||||
|
||||
private sealed class CheckedItem
|
||||
{
|
||||
public required ItemToGather GatheredItem { get; init; }
|
||||
public required List<Venture> Ventures { get; init; }
|
||||
public required uint ItemId { get; init; }
|
||||
}
|
||||
}
|
19
ARControl/packages.lock.json
Normal file
19
ARControl/packages.lock.json
Normal file
@ -0,0 +1,19 @@
|
||||
{
|
||||
"version": 1,
|
||||
"dependencies": {
|
||||
"net7.0-windows7.0": {
|
||||
"Dalamud.ContextMenu": {
|
||||
"type": "Direct",
|
||||
"requested": "[1.2.3, )",
|
||||
"resolved": "1.2.3",
|
||||
"contentHash": "ydemplF7DNcA/LLeongDVzWUD/JV0Fw3EwA2+P0jYq3Le2ZYSt4U8qyJq4FyoChqt0lFG8BxYCAzfeWp4Jmnqw=="
|
||||
},
|
||||
"DalamudPackager": {
|
||||
"type": "Direct",
|
||||
"requested": "[2.1.11, )",
|
||||
"resolved": "2.1.11",
|
||||
"contentHash": "9qlAWoRRTiL/geAvuwR/g6Bcbrd/bJJgVnB/RurBiyKs6srsP0bvpoo8IK+Eg8EA6jWeM6/YJWs66w4FIAzqPw=="
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
13
README.md
Normal file
13
README.md
Normal file
@ -0,0 +1,13 @@
|
||||
# ARC
|
||||
|
||||
Instead of manually managing retainer plans for each individual character,
|
||||
which can get fairly tedious fairly quickly, we optimize this a bit and
|
||||
only manage "high-level" plans.
|
||||
|
||||
This means we create a list a la:
|
||||
|
||||
- 5000 Cobalt Ore
|
||||
- 2000 Gold Ore
|
||||
|
||||
... and ARC distributes each venture automatically amongst all characters
|
||||
which are included.
|
7
global.json
Normal file
7
global.json
Normal file
@ -0,0 +1,7 @@
|
||||
{
|
||||
"sdk": {
|
||||
"version": "7.0.0",
|
||||
"rollForward": "latestMinor",
|
||||
"allowPrerelease": false
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user