Compare commits
19 Commits
4495177f02
...
fr-transla
| Author | SHA1 | Date | |
|---|---|---|---|
|
6d8a8476b4
|
|||
|
0808266887
|
|||
|
a2071b9c05
|
|||
|
612e7c88a2
|
|||
|
1755b5cb54
|
|||
|
4a388dcfa9
|
|||
|
a0957715a5
|
|||
|
04a8ee3186
|
|||
|
b79a51748f
|
|||
|
95d9f65068
|
|||
|
a70968d30c
|
|||
|
6ebb73040b
|
|||
|
46f2443824
|
|||
|
eeab8354b6
|
|||
|
b5d8f288f9
|
|||
| 3c2dab4d21 | |||
| edb49f710a | |||
| 9ff21dc341 | |||
| 17962a37b3 |
6
.gitignore
vendored
6
.gitignore
vendored
@@ -2,15 +2,17 @@
|
||||
## files generated by popular Visual Studio add-ons.
|
||||
##
|
||||
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
||||
|
||||
.idea
|
||||
# User-specific files
|
||||
*.rsuser
|
||||
*.suo
|
||||
*.user
|
||||
*.userosscache
|
||||
*.sln.docstates
|
||||
*.bak
|
||||
.DS_Store
|
||||
MareSynchronos/.DS_Store
|
||||
*.zip
|
||||
UmbraServer_extracted/
|
||||
|
||||
# User-specific files (MonoDevelop/Xamarin Studio)
|
||||
*.userprefs
|
||||
|
||||
23
.gitmodules
vendored
23
.gitmodules
vendored
@@ -1,15 +1,12 @@
|
||||
[submodule "UmbraAPI"]
|
||||
path = UmbraAPI
|
||||
url = https://git.umbra-sync.net/SirConstance/UmbraAPI.git
|
||||
|
||||
[submodule "MareAPI"]
|
||||
path = MareAPI
|
||||
url = ssh://git@git.umbra-sync.net:1222/Keda/UmbraAPI.git
|
||||
branch = main
|
||||
[submodule "Penumbra.Api"]
|
||||
path = Penumbra.Api
|
||||
url = https://github.com/Ottermandias/Penumbra.Api.git
|
||||
|
||||
path = Penumbra.Api
|
||||
url = https://github.com/Ottermandias/Penumbra.Api.git
|
||||
branch = main
|
||||
[submodule "Glamourer.Api"]
|
||||
path = Glamourer.Api
|
||||
url = https://github.com/Ottermandias/Glamourer.Api
|
||||
|
||||
[submodule "UmbraCrypt"]
|
||||
path = UmbraCrypt
|
||||
url = https://git.umbra-sync.net/SirConstance/UmbraCrypt.git
|
||||
path = Glamourer.Api
|
||||
url = https://github.com/Ottermandias/Glamourer.Api.git
|
||||
branch = main
|
||||
|
||||
1
Glamourer.Api
Submodule
1
Glamourer.Api
Submodule
Submodule Glamourer.Api added at 54c1944dc7
File diff suppressed because it is too large
Load Diff
3
Glamourer.Api/.gitignore
vendored
3
Glamourer.Api/.gitignore
vendored
@@ -1,3 +0,0 @@
|
||||
bin/
|
||||
obj/
|
||||
.vs/
|
||||
@@ -1,14 +0,0 @@
|
||||
namespace Glamourer.Api.Api;
|
||||
|
||||
/// <summary> The full API available. </summary>
|
||||
public interface IGlamourerApi : IGlamourerApiBase
|
||||
{
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns"/>
|
||||
public IGlamourerApiDesigns Designs { get; }
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems"/>
|
||||
public IGlamourerApiItems Items { get; }
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState"/>
|
||||
public IGlamourerApiState State { get; }
|
||||
}
|
||||
@@ -1,11 +0,0 @@
|
||||
namespace Glamourer.Api.Api;
|
||||
|
||||
/// <summary> Basic API functions. </summary>
|
||||
public interface IGlamourerApiBase
|
||||
{
|
||||
/// <summary>
|
||||
/// Get the current API version of the Glamourer available in this installation.
|
||||
/// Major version changes indicate incompatibilities, minor version changes are backward-compatible additions.
|
||||
/// </summary>
|
||||
public (int Major, int Minor) ApiVersion { get; }
|
||||
}
|
||||
@@ -1,33 +0,0 @@
|
||||
using Glamourer.Api.Enums;
|
||||
|
||||
namespace Glamourer.Api.Api;
|
||||
|
||||
/// <summary> All functions related to Glamourer designs. </summary>
|
||||
public interface IGlamourerApiDesigns
|
||||
{
|
||||
/// <summary> Obtain a list of all available designs. </summary>
|
||||
/// <returns> A dictionary of all designs from their GUID to their current display name. </returns>
|
||||
public Dictionary<Guid, string> GetDesignList();
|
||||
|
||||
/// <summary> Apply an existing design to an actor. </summary>
|
||||
/// <param name="designId"> The GUID of the design to apply. </param>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once, Equipment, Customization, Lock (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> DesignNotFound, ActorNotFound, InvalidKey, Success. </returns>
|
||||
public GlamourerApiEc ApplyDesign(Guid designId, int objectIndex, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Apply an existing design to an actor. </summary>
|
||||
/// <param name="designId"> The GUID of the design to apply. </param>
|
||||
/// <param name="playerName"> The name of the players to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once, Equipment, Customization, Lock (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> DesignNotFound, ActorNotFound, InvalidKey, Success. </returns>
|
||||
/// /// <remarks>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are reverted.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc ApplyDesignName(Guid designId, string playerName, uint key, ApplyFlag flags);
|
||||
}
|
||||
@@ -1,80 +0,0 @@
|
||||
using Glamourer.Api.Enums;
|
||||
|
||||
namespace Glamourer.Api.Api;
|
||||
|
||||
/// <summary> All functions related to items. </summary>
|
||||
public interface IGlamourerApiItems
|
||||
{
|
||||
/// <summary> Set a single item on an actor. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="slot"> The slot to apply the item to. </param>
|
||||
/// <param name="itemId"> The (Custom) ID of the item to apply. </param>
|
||||
/// <param name="stains"> The IDs of the stains to apply to the item. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
/// <remarks> The item ID can be a custom item ID in Glamourer's format for models without an associated item, or a normal game item ID. </remarks>
|
||||
public GlamourerApiEc SetItem(int objectIndex, ApiEquipSlot slot, ulong itemId, IReadOnlyList<byte> stains, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Set a single item on players. </summary>
|
||||
/// <param name="playerName"> The name of the players to be manipulated. </param>
|
||||
/// <param name="slot"> The slot to apply the item to. </param>
|
||||
/// <param name="itemId"> The (Custom) ID of the item to apply. </param>
|
||||
/// <param name="stains"> The IDs of the stains to apply to the item. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
/// <remarks>
|
||||
/// The item ID can be a custom item ID in Glamourer's format for models without an associated item, or a normal game item ID.<br/>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are modified.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc SetItemName(string playerName, ApiEquipSlot slot, ulong itemId, IReadOnlyList<byte> stains, uint key,
|
||||
ApplyFlag flags);
|
||||
|
||||
/// <summary> Set a single bonus item on an actor. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="slot"> The bonus slot to apply the item to. </param>
|
||||
/// <param name="bonusItemId"> The bonus item sheet ID of the item to apply (including stain). </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
/// <remarks> The bonus item ID can currently not be a custom item ID in Glamourer's format for models without an associated item. Use 0 to remove the bonus item. </remarks>
|
||||
public GlamourerApiEc SetBonusItem(int objectIndex, ApiBonusSlot slot, ulong bonusItemId, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Set a single bonus item on an actor. </summary>
|
||||
/// <param name="playerName"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="slot"> The bonus slot to apply the item to. </param>
|
||||
/// <param name="bonusItemId"> The bonus item sheet ID of the item to apply (including stain). </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
/// <remarks>
|
||||
/// The bonus item ID can currently not be a custom item ID in Glamourer's format for models without an associated item. Use 0 to remove the bonus item. <br/>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are modified.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc SetBonusItemName(string playerName, ApiBonusSlot slot, ulong bonusItemId, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Set the defined Meta State flags to the active or inactive state on actor. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="types"> The flags defining which meta states to update to the new value. This can be multiple at once. </param>
|
||||
/// <param name="newValue"> The new value to update to. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag.Once"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
public GlamourerApiEc SetMetaState(int objectIndex, MetaFlag types, bool newValue, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Set the defined Meta State flags to the active or inactive state on actor (by name) </summary>
|
||||
/// <param name="playerName"> The name of the players to be manipulated. </param>
|
||||
/// <param name="types"> The flags defining which meta states to update to the new value. This can be multiple at once. </param>
|
||||
/// <param name="newValue"> The new value to update to. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once (see <see cref="ApplyFlag.Once"/>.)</param>
|
||||
/// <returns> ItemInvalid, ActorNotFound, ActorNotHuman, InvalidKey, Success. </returns>
|
||||
public GlamourerApiEc SetMetaStateName(string playerName, MetaFlag types, bool newValue, uint key, ApplyFlag flags);
|
||||
}
|
||||
@@ -1,124 +0,0 @@
|
||||
using Glamourer.Api.Enums;
|
||||
using Newtonsoft.Json.Linq;
|
||||
|
||||
namespace Glamourer.Api.Api;
|
||||
|
||||
/// <summary> Any functions related to Glamourer's state tracking. </summary>
|
||||
public interface IGlamourerApiState
|
||||
{
|
||||
/// <summary> Get the current Glamourer state of an actor. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the desired actor. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <returns> ActorNotFound, InvalidKey or Success, and the state on success. </returns>
|
||||
/// <remarks> The actor does not need to have a prior Glamourer state as long as it can be found. </remarks>
|
||||
public (GlamourerApiEc, JObject?) GetState(int objectIndex, uint key);
|
||||
|
||||
/// <summary> Get the current Glamourer state of a player character. </summary>
|
||||
/// <param name="playerName"> The name of the desired player. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <returns> ActorNotFound, InvalidKey or Success, and the state on success. </returns>
|
||||
/// <remarks>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.
|
||||
/// Only players are checked for name equality, no NPCs.
|
||||
/// If multiple players of the same name are found, the first is returned.
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public (GlamourerApiEc, JObject?) GetStateName(string playerName, uint key);
|
||||
|
||||
/// <inheritdoc cref="GetState"/>
|
||||
public (GlamourerApiEc, string?) GetStateBase64(int objectIndex, uint key);
|
||||
|
||||
/// <inheritdoc cref="GetStateName"/>
|
||||
public (GlamourerApiEc, string?) GetStateBase64Name(string objectName, uint key);
|
||||
|
||||
/// <summary> Apply a supplied state to an actor. </summary>
|
||||
/// <param name="applyState"> The state, which can be either a Glamourer-supplied JObject or a Base64 string. </param>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the application. Respects Once, Equipment, Customization and Lock (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, ActorNotHuman, Success. </returns>
|
||||
public GlamourerApiEc ApplyState(object applyState, int objectIndex, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Apply a supplied state to players. </summary>
|
||||
/// <param name="applyState"> The state, which can be either a Glamourer-supplied JObject or a Base64 string. </param>
|
||||
/// <param name="playerName"> The name of the player to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock or lock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the application. Respects Once, Equipment, Customization and Lock (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, ActorNotHuman, Success. </returns>
|
||||
/// <remarks>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are manipulated.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc ApplyStateName(object applyState, string playerName, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Revert the Glamourer state of an actor to Game state. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Equipment and Customization (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, Success, NothingDone. </returns>
|
||||
public GlamourerApiEc RevertState(int objectIndex, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Revert the Glamourer state of players to game state. </summary>
|
||||
/// <param name="playerName"> The name of the players to be reverted. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Equipment and Customization (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, Success, NothingDone. </returns>
|
||||
/// /// <remarks>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are reverted.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc RevertStateName(string playerName, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Unlock the Glamourer state of an actor with a key. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock the state. </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, Success, NothingDone. </returns>
|
||||
public GlamourerApiEc UnlockState(int objectIndex, uint key);
|
||||
|
||||
/// <summary> Unlock the Glamourer state of players with a key. </summary>
|
||||
/// <param name="playerName"> The name of the players to be unlocked. </param>
|
||||
/// <param name="key"> A key to unlock the state. </param>
|
||||
/// <returns> InvalidKey, Success, NothingDone. </returns>
|
||||
public GlamourerApiEc UnlockStateName(string playerName, uint key);
|
||||
|
||||
/// <summary> Unlock all active glamourer states with a key. </summary>
|
||||
/// <param name="key"> The key to unlock states with. </param>
|
||||
/// <returns> The number of unlocked states. </returns>
|
||||
public int UnlockAll(uint key);
|
||||
|
||||
/// <summary> Revert the Glamourer state of an actor to automation state. </summary>
|
||||
/// <param name="objectIndex"> The game object index of the actor to be manipulated. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once and Lock (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, Success, NothingDone. </returns>
|
||||
public GlamourerApiEc RevertToAutomation(int objectIndex, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Revert the Glamourer state of players to automation state. </summary>
|
||||
/// <param name="playerName"> The name of the players to be reverted. </param>
|
||||
/// <param name="key"> A key to unlock the state if necessary. </param>
|
||||
/// <param name="flags"> The flags used for the reversion. Respects Once and Lock (see <see cref="ApplyFlag"/>.) </param>
|
||||
/// <returns> ActorNotFound, InvalidKey, Success, NothingDone. </returns>
|
||||
/// /// <remarks>
|
||||
/// The player does not have to be currently available as long as he has a persisted Glamourer state.<br/>
|
||||
/// Only players are checked for name equality, no NPCs.<br/>
|
||||
/// If multiple players of the same name are found, all of them are reverted.<br/>
|
||||
/// Prefer to use the index-based function unless you need to get the state of someone currently unavailable.
|
||||
/// </remarks>
|
||||
public GlamourerApiEc RevertToAutomationName(string playerName, uint key, ApplyFlag flags);
|
||||
|
||||
/// <summary> Invoked with the game object pointer (if available) whenever an actors tracked state changes. </summary>
|
||||
public event Action<nint> StateChanged;
|
||||
|
||||
/// <summary> Invoked with the game object pointer (if available) whenever an actors tracked state changes, with the type of change. </summary>
|
||||
public event Action<nint, StateChangeType> StateChangedWithType;
|
||||
|
||||
/// <summary> Invoked with the game object pointer (if available) whenever an actors tracked state finalizes a grouped change consisting of multiple smaller changes. </summary>
|
||||
public event Action<nint, StateFinalizationType> StateFinalized;
|
||||
|
||||
/// <summary> Invoked when the player enters or leaves GPose (true => entered GPose, false => left GPose). </summary>
|
||||
public event Action<bool>? GPoseChanged;
|
||||
}
|
||||
@@ -1,11 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> Bonus item slots restricted to API-relevant slots. </summary>
|
||||
public enum ApiBonusSlot : byte
|
||||
{
|
||||
/// <summary> No slot. </summary>
|
||||
Unknown = 0,
|
||||
|
||||
/// <summary> The Glasses slot. </summary>
|
||||
Glasses = 1,
|
||||
}
|
||||
@@ -1,45 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> Equip slots restricted to API-relevant slots, but compatible with GameData.EquipSlots. </summary>
|
||||
public enum ApiEquipSlot : byte
|
||||
{
|
||||
/// <summary> No slot. </summary>
|
||||
Unknown = 0,
|
||||
|
||||
/// <summary> Mainhand, also used for both-handed weapons. </summary>
|
||||
MainHand = 1,
|
||||
|
||||
/// <summary> Offhand, used for shields or if you want to apply the offhand component of certain weapons. </summary>
|
||||
OffHand = 2,
|
||||
|
||||
/// <summary> Head. </summary>
|
||||
Head = 3,
|
||||
|
||||
/// <summary> Body. </summary>
|
||||
Body = 4,
|
||||
|
||||
/// <summary> Hands. </summary>
|
||||
Hands = 5,
|
||||
|
||||
/// <summary> Legs. </summary>
|
||||
Legs = 7,
|
||||
|
||||
/// <summary> Feet. </summary>
|
||||
Feet = 8,
|
||||
|
||||
/// <summary> Ears. </summary>
|
||||
Ears = 9,
|
||||
|
||||
/// <summary> Neck. </summary>
|
||||
Neck = 10,
|
||||
|
||||
/// <summary> Wrists. </summary>
|
||||
Wrists = 11,
|
||||
|
||||
/// <summary> Right Finger. </summary>
|
||||
RFinger = 12,
|
||||
|
||||
/// <summary> Left Finger. </summary>
|
||||
/// <remarks> Not officially existing, means "weapon could be equipped in either hand" for the game. </remarks>
|
||||
LFinger = 14,
|
||||
}
|
||||
@@ -1,31 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> Application flags that can be used in different situations. </summary>
|
||||
[Flags]
|
||||
public enum ApplyFlag : ulong
|
||||
{
|
||||
/// <summary> Apply the selected manipulation only once, without forcing the state into automation. </summary>
|
||||
Once = 0x01,
|
||||
|
||||
/// <summary> Apply the selected manipulation on the equipment (might be more or less supported). </summary>
|
||||
Equipment = 0x02,
|
||||
|
||||
/// <summary> Apply the selected manipulation on the customizations (might be more or less supported). </summary>
|
||||
Customization = 0x04,
|
||||
|
||||
/// <summary> Lock the state with the given key after applying the selected manipulation </summary>
|
||||
Lock = 0x08,
|
||||
}
|
||||
|
||||
/// <summary> Extensions for apply flags. </summary>
|
||||
public static class ApplyFlagEx
|
||||
{
|
||||
/// <summary> The default application flags for design-based manipulations. </summary>
|
||||
public const ApplyFlag DesignDefault = ApplyFlag.Once | ApplyFlag.Equipment | ApplyFlag.Customization;
|
||||
|
||||
/// <summary> The default application flags for state-based manipulations. </summary>
|
||||
public const ApplyFlag StateDefault = ApplyFlag.Equipment | ApplyFlag.Customization | ApplyFlag.Lock;
|
||||
|
||||
/// <summary> The default application flags for reverse manipulations. </summary>
|
||||
public const ApplyFlag RevertDefault = ApplyFlag.Equipment | ApplyFlag.Customization;
|
||||
}
|
||||
@@ -1,29 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> Return codes for API functions. </summary>
|
||||
public enum GlamourerApiEc
|
||||
{
|
||||
/// <summary> The function succeeded. </summary>
|
||||
Success = 0,
|
||||
|
||||
/// <summary> The function did not encounter a problem, but also did not do anything. </summary>
|
||||
NothingDone = 1,
|
||||
|
||||
/// <summary> The requested actor was not found. </summary>
|
||||
ActorNotFound = 2,
|
||||
|
||||
/// <summary> The requested actor was not human, but should have been. </summary>
|
||||
ActorNotHuman,
|
||||
|
||||
/// <summary> The requested design was not found. </summary>
|
||||
DesignNotFound,
|
||||
|
||||
/// <summary> The requested item was not found or could not be applied to the requested slot. </summary>
|
||||
ItemInvalid,
|
||||
|
||||
/// <summary> The state of an actor could not be manipulated because it was locked and the provided key could not unlock it. </summary>
|
||||
InvalidKey,
|
||||
|
||||
/// <summary> The provided object could not be converted into a valid Glamourer state to apply. </summary>
|
||||
InvalidState,
|
||||
}
|
||||
@@ -1,11 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> Application flags for setting the meta state of an actor. </summary>
|
||||
[Flags]
|
||||
public enum MetaFlag : ulong
|
||||
{
|
||||
Wetness = 0x01,
|
||||
HatState = 0x02,
|
||||
VisorState = 0x04,
|
||||
WeaponState = 0x08,
|
||||
}
|
||||
@@ -1,47 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> What type of information changed in a state. </summary>
|
||||
public enum StateChangeType
|
||||
{
|
||||
/// <summary> A characters saved state had the model id changed. This means everything may have changed. </summary>
|
||||
Model = 0,
|
||||
|
||||
/// <summary> A characters saved state had multiple customization values changed. </summary>
|
||||
EntireCustomize = 1,
|
||||
|
||||
/// <summary> A characters saved state had a customization value changed. </summary>
|
||||
Customize = 2,
|
||||
|
||||
/// <summary> A characters saved state had an equipment piece changed. </summary>
|
||||
Equip = 3,
|
||||
|
||||
/// <summary> A characters saved state had its weapons changed. </summary>
|
||||
Weapon = 4,
|
||||
|
||||
/// <summary> A characters saved state had a stain changed. </summary>
|
||||
Stains = 5,
|
||||
|
||||
/// <summary> A characters saved state had a crest visibility changed. </summary>
|
||||
Crest = 6,
|
||||
|
||||
/// <summary> A characters saved state had its customize parameter changed. </summary>
|
||||
Parameter = 7,
|
||||
|
||||
/// <summary> A characters saved state had a material color table value changed. </summary>
|
||||
MaterialValue = 8,
|
||||
|
||||
/// <summary> A characters saved state had a design applied. This means everything may have changed. </summary>
|
||||
Design = 9,
|
||||
|
||||
/// <summary> A characters saved state had its state reset to its game values. </summary>
|
||||
Reset = 10,
|
||||
|
||||
/// <summary> A characters saved state had a meta toggle changed. </summary>
|
||||
Other = 11,
|
||||
|
||||
/// <summary> A characters state was reapplied. Data is null. </summary>
|
||||
Reapply = 12,
|
||||
|
||||
/// <summary> A characters saved state had a bonus item changed. </summary>
|
||||
BonusItem = 13,
|
||||
}
|
||||
@@ -1,36 +0,0 @@
|
||||
namespace Glamourer.Api.Enums;
|
||||
|
||||
/// <summary> What type of Glamourer process was performed on the actors state to update it. </summary>
|
||||
public enum StateFinalizationType
|
||||
{
|
||||
/// <summary> A characters saved state had the model id altered. </summary>
|
||||
ModelChange = 0,
|
||||
|
||||
/// <summary> A singular Design was applied to an actors state. </summary>
|
||||
DesignApplied = 1,
|
||||
|
||||
/// <summary> A characters saved state had been reset to game values. </summary>
|
||||
Revert = 2,
|
||||
|
||||
/// <summary> A characters saved state had only its customization data reset to game state. </summary>
|
||||
RevertCustomize = 3,
|
||||
|
||||
/// <summary> A characters saved state had only its equipment data reset to game state. </summary>
|
||||
RevertEquipment = 4,
|
||||
|
||||
/// <summary> A characters saved state had its advanced values reverted to game state. </summary>
|
||||
RevertAdvanced = 5,
|
||||
|
||||
/// <summary> A characters saved state was reverted to automation state on top of their game state </summary>
|
||||
RevertAutomation = 6,
|
||||
|
||||
/// <summary> A characters saved state had a generic reapply as a single operation. </summary>
|
||||
Reapply = 7,
|
||||
|
||||
/// <summary> A characters saved state had their automation state reapplied over their existing state. </summary>
|
||||
ReapplyAutomation = 8,
|
||||
|
||||
/// <summary> A characters save state finished applying all updated slots for game state on gearset change or initial load. </summary>
|
||||
Gearset = 9,
|
||||
}
|
||||
|
||||
@@ -1,34 +0,0 @@
|
||||
<Project Sdk="Dalamud.NET.Sdk/13.0.0">
|
||||
<PropertyGroup>
|
||||
<AssemblyTitle>Glamourer.Api</AssemblyTitle>
|
||||
<Product>Glamourer</Product>
|
||||
<Copyright>Copyright © 2025</Copyright>
|
||||
<FileVersion>2.4.1.0</FileVersion>
|
||||
<AssemblyVersion>2.4.1.0</AssemblyVersion>
|
||||
<PackageVersion>2.4.1</PackageVersion>
|
||||
<PackageReadmeFile>README.md</PackageReadmeFile>
|
||||
<OutputPath>bin\$(Configuration)\</OutputPath>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||
<Title>Glamourer.Api</Title>
|
||||
<Authors>Ottermandias</Authors>
|
||||
<RepositoryUrl>https://github.com/Ottermandias/Glamourer</RepositoryUrl>
|
||||
<Description>Auxiliary functions for Glamourers external API.</Description>
|
||||
<PackageLicenseExpression>MIT</PackageLicenseExpression>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<Use_DalamudPackager>false</Use_DalamudPackager>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<NoWarn>1591</NoWarn>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Include="README.md" Pack="true" PackagePath="\" />
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
@@ -1,2 +0,0 @@
|
||||
<wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation">
|
||||
<s:Boolean x:Key="/Default/CodeInspection/NamespaceProvider/NamespaceFoldersToSkip/=ipc/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>
|
||||
@@ -1,4 +0,0 @@
|
||||
// Global using directives
|
||||
|
||||
global using System;
|
||||
global using System.Collections.Generic;
|
||||
@@ -1,114 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Ipc;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
/// <summary>
|
||||
/// Specialized subscriber only allowing to invoke actions.
|
||||
/// </summary>
|
||||
public class ActionSubscriber
|
||||
{
|
||||
private readonly ICallGateSubscriber<object?>? _subscriber;
|
||||
|
||||
/// <summary> Whether the subscriber could successfully be created. </summary>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
protected ActionSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<object?>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Invoke the action. See the source of the subscriber for details.</summary>
|
||||
protected void Invoke()
|
||||
=> _subscriber?.InvokeAction();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber"/>
|
||||
public class ActionSubscriber<T1>
|
||||
{
|
||||
private readonly ICallGateSubscriber<T1, object?>? _subscriber;
|
||||
|
||||
/// <summary> Whether the subscriber could successfully be created. </summary>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
protected ActionSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, object?>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Invoke the action. See the source of the subscriber for details.</summary>
|
||||
protected void Invoke(T1 a)
|
||||
=> _subscriber?.InvokeAction(a);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber"/>
|
||||
public class ActionSubscriber<T1, T2>
|
||||
{
|
||||
private readonly ICallGateSubscriber<T1, T2, object?>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber{T1}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
protected ActionSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, object?>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber.Invoke"/>
|
||||
protected void Invoke(T1 a, T2 b)
|
||||
=> _subscriber?.InvokeAction(a, b);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber"/>
|
||||
public class ActionSubscriber<T1, T2, T3>
|
||||
{
|
||||
private readonly ICallGateSubscriber<T1, T2, T3, object?>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber{T1}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
protected ActionSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, object?>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="ActionSubscriber.Invoke"/>
|
||||
protected void Invoke(T1 a, T2 b, T3 c)
|
||||
=> _subscriber?.InvokeAction(a, b, c);
|
||||
}
|
||||
@@ -1,234 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Ipc;
|
||||
using Dalamud.Plugin.Services;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
/// <summary>
|
||||
/// Specialized disposable Provider for Events.<para />
|
||||
/// Will execute the unsubscriber action on dispose if any is provided.<para />
|
||||
/// Can only be invoked and disposed.
|
||||
/// </summary>
|
||||
public sealed class EventProvider : IDisposable
|
||||
{
|
||||
private readonly IPluginLog _log;
|
||||
private ICallGateProvider<object?>? _provider;
|
||||
private Delegate? _unsubscriber;
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, (Action<Action> Add, Action<Action> Del)? subscribe = null)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<object?>(label);
|
||||
subscribe?.Add(Invoke);
|
||||
_unsubscriber = subscribe?.Del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, Action<EventProvider> add, Action<EventProvider> del)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<object?>(label);
|
||||
add(this);
|
||||
_unsubscriber = del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Invoke the event.</summary>
|
||||
public void Invoke()
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider?.SendMessage();
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception thrown on IPC event:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
switch (_unsubscriber)
|
||||
{
|
||||
case Action<Action> a:
|
||||
a(Invoke);
|
||||
break;
|
||||
case Action<EventProvider> b:
|
||||
b(this);
|
||||
break;
|
||||
}
|
||||
|
||||
_unsubscriber = null;
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~EventProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="EventProvider"/>
|
||||
public sealed class EventProvider<T1> : IDisposable
|
||||
{
|
||||
private readonly IPluginLog _log;
|
||||
private ICallGateProvider<T1, object?>? _provider;
|
||||
private Delegate? _unsubscriber;
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, (Action<Action<T1>> Add, Action<Action<T1>> Del)? subscribe = null)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, object?>(label);
|
||||
subscribe?.Add(Invoke);
|
||||
_unsubscriber = subscribe?.Del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, Action<EventProvider<T1>> add, Action<EventProvider<T1>> del)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, object?>(label);
|
||||
add(this);
|
||||
_unsubscriber = del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="EventProvider.Invoke"/>
|
||||
public void Invoke(T1 a)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider?.SendMessage(a);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception thrown on IPC event:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
switch (_unsubscriber)
|
||||
{
|
||||
case Action<Action<T1>> a:
|
||||
a(Invoke);
|
||||
break;
|
||||
case Action<EventProvider<T1>> b:
|
||||
b(this);
|
||||
break;
|
||||
}
|
||||
|
||||
_unsubscriber = null;
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~EventProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="EventProvider"/>
|
||||
public sealed class EventProvider<T1, T2> : IDisposable
|
||||
{
|
||||
private readonly IPluginLog _log;
|
||||
private ICallGateProvider<T1, T2, object?>? _provider;
|
||||
private Delegate? _unsubscriber;
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, (Action<Action<T1, T2>> Add, Action<Action<T1, T2>> Del)? subscribe = null)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, object?>(label);
|
||||
subscribe?.Add(Invoke);
|
||||
_unsubscriber = subscribe?.Del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
public EventProvider(IDalamudPluginInterface pi, string label, Action<EventProvider<T1, T2>> add, Action<EventProvider<T1, T2>> del)
|
||||
{
|
||||
_unsubscriber = null;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, object?>(label);
|
||||
add(this);
|
||||
_unsubscriber = del;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="EventProvider.Invoke"/>
|
||||
public void Invoke(T1 a, T2 b)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider?.SendMessage(a, b);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception thrown on IPC event:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
switch (_unsubscriber)
|
||||
{
|
||||
case Action<Action<T1, T2>> a:
|
||||
a(Invoke);
|
||||
break;
|
||||
case Action<EventProvider<T1, T2>> b:
|
||||
b(this);
|
||||
break;
|
||||
}
|
||||
|
||||
_unsubscriber = null;
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~EventProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
@@ -1,394 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Ipc;
|
||||
using Dalamud.Plugin.Services;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
/// <summary>
|
||||
/// Specialized disposable Subscriber for Events.<para />
|
||||
/// Subscriptions are wrapped to be individually exception-safe.<para/>
|
||||
/// Can be enabled and disabled.<para/>
|
||||
/// </summary>
|
||||
public sealed class EventSubscriber : IDisposable
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly IPluginLog _log;
|
||||
private readonly Dictionary<Action, Action> _delegates = new();
|
||||
private ICallGateSubscriber<object?>? _subscriber;
|
||||
private bool _disabled;
|
||||
|
||||
public EventSubscriber(IDalamudPluginInterface pi, string label, params Action[] actions)
|
||||
{
|
||||
_label = label;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<object?>(label);
|
||||
foreach (var action in actions)
|
||||
Event += action;
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Enable all currently subscribed actions registered with this EventSubscriber.
|
||||
/// Does nothing if it is already enabled.
|
||||
/// </summary>
|
||||
public void Enable()
|
||||
{
|
||||
if (_disabled && _subscriber != null)
|
||||
{
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Subscribe(action);
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Disable all subscribed actions registered with this EventSubscriber.
|
||||
/// Does nothing if it is already disabled.
|
||||
/// Does not forget the actions, only disables them.
|
||||
/// </summary>
|
||||
public void Disable()
|
||||
{
|
||||
if (!_disabled)
|
||||
{
|
||||
if (_subscriber != null)
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Unsubscribe(action);
|
||||
|
||||
_disabled = true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Add or remove an action to the IPC event, if it is valid.
|
||||
/// </summary>
|
||||
public event Action Event
|
||||
{
|
||||
add
|
||||
{
|
||||
if (_subscriber != null && !_delegates.ContainsKey(value))
|
||||
{
|
||||
void Action()
|
||||
{
|
||||
try
|
||||
{
|
||||
value();
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception invoking IPC event {_label}:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
if (_delegates.TryAdd(value, Action) && !_disabled)
|
||||
_subscriber.Subscribe(Action);
|
||||
}
|
||||
}
|
||||
remove
|
||||
{
|
||||
if (_subscriber != null && _delegates.Remove(value, out var action))
|
||||
_subscriber.Unsubscribe(action);
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
Disable();
|
||||
_subscriber = null;
|
||||
_delegates.Clear();
|
||||
}
|
||||
|
||||
~EventSubscriber()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber"/> </summary>
|
||||
public sealed class EventSubscriber<T1> : IDisposable
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly IPluginLog _log;
|
||||
private readonly Dictionary<Action<T1>, Action<T1>> _delegates = new();
|
||||
private ICallGateSubscriber<T1, object?>? _subscriber;
|
||||
private bool _disabled;
|
||||
|
||||
public EventSubscriber(IDalamudPluginInterface pi, string label, params Action<T1>[] actions)
|
||||
{
|
||||
_label = label;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, object?>(label);
|
||||
foreach (var action in actions)
|
||||
Event += action;
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Enable"/> </summary>
|
||||
public void Enable()
|
||||
{
|
||||
if (_disabled && _subscriber != null)
|
||||
{
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Subscribe(action);
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Disable"/> </summary>
|
||||
public void Disable()
|
||||
{
|
||||
if (!_disabled)
|
||||
{
|
||||
if (_subscriber != null)
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Unsubscribe(action);
|
||||
|
||||
_disabled = true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Event"/> </summary>
|
||||
public event Action<T1> Event
|
||||
{
|
||||
add
|
||||
{
|
||||
if (_subscriber != null && !_delegates.ContainsKey(value))
|
||||
{
|
||||
void Action(T1 a)
|
||||
{
|
||||
try
|
||||
{
|
||||
value(a);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception invoking IPC event {_label}:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
if (_delegates.TryAdd(value, Action) && !_disabled)
|
||||
_subscriber.Subscribe(Action);
|
||||
}
|
||||
}
|
||||
remove
|
||||
{
|
||||
if (_subscriber != null && _delegates.Remove(value, out var action))
|
||||
_subscriber.Unsubscribe(action);
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
Disable();
|
||||
_subscriber = null;
|
||||
_delegates.Clear();
|
||||
}
|
||||
|
||||
~EventSubscriber()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber"/> </summary>
|
||||
public sealed class EventSubscriber<T1, T2> : IDisposable
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly IPluginLog _log;
|
||||
private readonly Dictionary<Action<T1, T2>, Action<T1, T2>> _delegates = new();
|
||||
private ICallGateSubscriber<T1, T2, object?>? _subscriber;
|
||||
private bool _disabled;
|
||||
|
||||
public EventSubscriber(IDalamudPluginInterface pi, string label, params Action<T1, T2>[] actions)
|
||||
{
|
||||
_label = label;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, object?>(label);
|
||||
foreach (var action in actions)
|
||||
Event += action;
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Enable"/> </summary>
|
||||
public void Enable()
|
||||
{
|
||||
if (_disabled && _subscriber != null)
|
||||
{
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Subscribe(action);
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Disable"/> </summary>
|
||||
public void Disable()
|
||||
{
|
||||
if (!_disabled)
|
||||
{
|
||||
if (_subscriber != null)
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Unsubscribe(action);
|
||||
|
||||
_disabled = true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Event"/> </summary>
|
||||
public event Action<T1, T2> Event
|
||||
{
|
||||
add
|
||||
{
|
||||
if (_subscriber != null && !_delegates.ContainsKey(value))
|
||||
{
|
||||
void Action(T1 a, T2 b)
|
||||
{
|
||||
try
|
||||
{
|
||||
value(a, b);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception invoking IPC event {_label}:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
if (_delegates.TryAdd(value, Action) && !_disabled)
|
||||
_subscriber.Subscribe(Action);
|
||||
}
|
||||
}
|
||||
remove
|
||||
{
|
||||
if (_subscriber != null && _delegates.Remove(value, out var action))
|
||||
_subscriber.Unsubscribe(action);
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
Disable();
|
||||
_subscriber = null;
|
||||
_delegates.Clear();
|
||||
}
|
||||
|
||||
~EventSubscriber()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber"/> </summary>
|
||||
public sealed class EventSubscriber<T1, T2, T3> : IDisposable
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly IPluginLog _log;
|
||||
private readonly Dictionary<Action<T1, T2, T3>, Action<T1, T2, T3>> _delegates = new();
|
||||
private ICallGateSubscriber<T1, T2, T3, object?>? _subscriber;
|
||||
private bool _disabled;
|
||||
|
||||
public EventSubscriber(IDalamudPluginInterface pi, string label, params Action<T1, T2, T3>[] actions)
|
||||
{
|
||||
_label = label;
|
||||
_log = PluginLogHelper.GetLog(pi);
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, object?>(label);
|
||||
foreach (var action in actions)
|
||||
Event += action;
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Enable"/> </summary>
|
||||
public void Enable()
|
||||
{
|
||||
if (_disabled && _subscriber != null)
|
||||
{
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Subscribe(action);
|
||||
|
||||
_disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Disable"/> </summary>
|
||||
public void Disable()
|
||||
{
|
||||
if (!_disabled)
|
||||
{
|
||||
if (_subscriber != null)
|
||||
foreach (var action in _delegates.Values)
|
||||
_subscriber.Unsubscribe(action);
|
||||
|
||||
_disabled = true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary><inheritdoc cref="EventSubscriber.Event"/> </summary>
|
||||
public event Action<T1, T2, T3> Event
|
||||
{
|
||||
add
|
||||
{
|
||||
if (_subscriber != null && !_delegates.ContainsKey(value))
|
||||
{
|
||||
void Action(T1 a, T2 b, T3 c)
|
||||
{
|
||||
try
|
||||
{
|
||||
value(a, b, c);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_log.Error($"Exception invoking IPC event {_label}:\n{e}");
|
||||
}
|
||||
}
|
||||
|
||||
if (_delegates.TryAdd(value, Action) && !_disabled)
|
||||
_subscriber.Subscribe(Action);
|
||||
}
|
||||
}
|
||||
remove
|
||||
{
|
||||
if (_subscriber != null && _delegates.Remove(value, out var action))
|
||||
_subscriber.Unsubscribe(action);
|
||||
}
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
Disable();
|
||||
_subscriber = null;
|
||||
_delegates.Clear();
|
||||
}
|
||||
|
||||
~EventSubscriber()
|
||||
=> Dispose();
|
||||
}
|
||||
@@ -1,224 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Ipc;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
/// <summary>
|
||||
/// Specialized disposable Provider for Funcs.
|
||||
/// </summary>
|
||||
public sealed class FuncProvider<TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, T2, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, T2, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, T2, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, T2, T3, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, T2, T3, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, T2, T3, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, T3, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, T2, T3, T4, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, T2, T3, T4, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, T2, T3, T4, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, T3, T4, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, T2, T3, T4, T5, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, T2, T3, T4, T5, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, T2, T3, T4, T5, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, T3, T4, T5, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
|
||||
|
||||
/// <inheritdoc cref="FuncProvider{TRet}"/>
|
||||
public sealed class FuncProvider<T1, T2, T3, T4, T5, T6, TRet> : IDisposable
|
||||
{
|
||||
private ICallGateProvider<T1, T2, T3, T4, T5, T6, TRet>? _provider;
|
||||
|
||||
public FuncProvider(IDalamudPluginInterface pi, string label, Func<T1, T2, T3, T4, T5, T6, TRet> func)
|
||||
{
|
||||
try
|
||||
{
|
||||
_provider = pi.GetIpcProvider<T1, T2, T3, T4, T5, T6, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Provider for {label}\n{e}");
|
||||
_provider = null;
|
||||
}
|
||||
|
||||
_provider?.RegisterFunc(func);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
_provider?.UnregisterFunc();
|
||||
_provider = null;
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
~FuncProvider()
|
||||
=> Dispose();
|
||||
}
|
||||
@@ -1,217 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Ipc;
|
||||
using Dalamud.Plugin.Ipc.Exceptions;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
/// <summary>
|
||||
/// Specialized subscriber only allowing to invoke functions with a return.
|
||||
/// </summary>
|
||||
public class FuncSubscriber<TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<TRet>? _subscriber;
|
||||
|
||||
/// <summary> Whether the subscriber could successfully be created. </summary>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary> Invoke the function. See the source of the subscriber for details.</summary>
|
||||
protected TRet Invoke()
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc() : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, T2, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, T2, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a, T2 b)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a, b) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, T2, T3, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, T2, T3, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a, T2 b, T3 c)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a, b, c) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, T2, T3, T4, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, T2, T3, T4, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, T4, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a, T2 b, T3 c, T4 d)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a, b, c, d) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, T2, T3, T4, T5, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, T2, T3, T4, T5, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, T4, T5, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a, T2 b, T3 c, T4 d, T5 e)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a, b, c, d, e) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
public class FuncSubscriber<T1, T2, T3, T4, T5, T6, TRet>
|
||||
{
|
||||
private readonly string _label;
|
||||
private readonly ICallGateSubscriber<T1, T2, T3, T4, T5, T6, TRet>? _subscriber;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Valid"/>
|
||||
public bool Valid
|
||||
=> _subscriber != null;
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}"/>
|
||||
protected FuncSubscriber(IDalamudPluginInterface pi, string label)
|
||||
{
|
||||
_label = label;
|
||||
try
|
||||
{
|
||||
_subscriber = pi.GetIpcSubscriber<T1, T2, T3, T4, T5, T6, TRet>(label);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
PluginLogHelper.WriteError(pi, $"Error registering IPC Subscriber for {label}\n{e}");
|
||||
_subscriber = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="FuncSubscriber{TRet}.Invoke"/>
|
||||
protected TRet Invoke(T1 a, T2 b, T3 c, T4 d, T5 e, T6 f)
|
||||
=> _subscriber != null ? _subscriber.InvokeFunc(a, b, c, d, e, f) : throw new IpcNotReadyError(_label);
|
||||
}
|
||||
@@ -1,26 +0,0 @@
|
||||
using Dalamud.IoC;
|
||||
using Dalamud.Plugin;
|
||||
using Dalamud.Plugin.Services;
|
||||
|
||||
namespace Glamourer.Api.Helpers;
|
||||
|
||||
internal class PluginLogHelper
|
||||
{
|
||||
[PluginService]
|
||||
private static IPluginLog? _log { get; set; }
|
||||
|
||||
private PluginLogHelper(IDalamudPluginInterface pi)
|
||||
=> pi.Inject(this);
|
||||
|
||||
public static void WriteError(IDalamudPluginInterface pi, string errorMessage)
|
||||
=> GetLog(pi).Error(errorMessage);
|
||||
|
||||
public static IPluginLog GetLog(IDalamudPluginInterface pi)
|
||||
{
|
||||
if (_log != null)
|
||||
return _log;
|
||||
|
||||
_ = new PluginLogHelper(pi);
|
||||
return _log!;
|
||||
}
|
||||
}
|
||||
@@ -1,52 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Api;
|
||||
using Glamourer.Api.Enums;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers;
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.GetDesignList"/>
|
||||
public sealed class GetDesignList(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<Dictionary<Guid, string>>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(GetDesignList)}.V2";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.GetDesignList"/>
|
||||
public new Dictionary<Guid, string> Invoke()
|
||||
=> base.Invoke();
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<Dictionary<Guid, string>> Provider(IDalamudPluginInterface pi, IGlamourerApiDesigns api)
|
||||
=> new(pi, Label, api.GetDesignList);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.ApplyDesign"/>
|
||||
public sealed class ApplyDesign(IDalamudPluginInterface pi) : FuncSubscriber<Guid, int, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(ApplyDesign)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.ApplyDesign"/>
|
||||
public GlamourerApiEc Invoke(Guid designId, int objectIndex, uint key = 0, ApplyFlag flags = ApplyFlagEx.DesignDefault)
|
||||
=> (GlamourerApiEc)Invoke(designId, objectIndex, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<Guid, int, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiDesigns api)
|
||||
=> new(pi, Label, (a, b, c, d) => (int)api.ApplyDesign(a, b, c, (ApplyFlag)d));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.ApplyDesignName"/>
|
||||
public sealed class ApplyDesignName(IDalamudPluginInterface pi) : FuncSubscriber<Guid, string, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(ApplyDesignName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiDesigns.ApplyDesignName"/>
|
||||
public GlamourerApiEc Invoke(Guid designId, string objectName, uint key = 0, ApplyFlag flags = ApplyFlagEx.DesignDefault)
|
||||
=> (GlamourerApiEc)Invoke(designId, objectName, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<Guid, string, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiDesigns api)
|
||||
=> new(pi, Label, (a, b, c, d) => (int)api.ApplyDesignName(a, b, c, (ApplyFlag)d));
|
||||
}
|
||||
@@ -1,110 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Api;
|
||||
using Glamourer.Api.Enums;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers;
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetItem"/>
|
||||
public sealed class SetItem(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, byte, ulong, IReadOnlyList<byte>, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetItem)}.V3";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetItem"/>
|
||||
public GlamourerApiEc Invoke(int objectIndex, ApiEquipSlot slot, ulong itemId, IReadOnlyList<byte> stain, uint key = 0,
|
||||
ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, (byte)slot, itemId, stain, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, byte, ulong, IReadOnlyList<byte>, uint, ulong, int> Provider(IDalamudPluginInterface pi,
|
||||
IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e, f) => (int)api.SetItem(a, (ApiEquipSlot)b, c, d, e, (ApplyFlag)f));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetItemName"/>
|
||||
public sealed class SetItemName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, byte, ulong, IReadOnlyList<byte>, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetItemName)}.V2";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetItem"/>
|
||||
public GlamourerApiEc Invoke(string objectName, ApiEquipSlot slot, ulong itemId, IReadOnlyList<byte> stain, uint key = 0,
|
||||
ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectName, (byte)slot, itemId, stain, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, byte, ulong, IReadOnlyList<byte>, uint, ulong, int> Provider(IDalamudPluginInterface pi,
|
||||
IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e, f) => (int)api.SetItemName(a, (ApiEquipSlot)b, c, d, e, (ApplyFlag)f));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetBonusItem"/>
|
||||
public sealed class SetBonusItem(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, byte, ulong, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetBonusItem)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetBonusItem"/>
|
||||
public GlamourerApiEc Invoke(int objectIndex, ApiBonusSlot slot, ulong itemId, uint key = 0, ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, (byte)slot, itemId, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, byte, ulong, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e) => (int)api.SetBonusItem(a, (ApiBonusSlot)b, c, d, (ApplyFlag)e));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetBonusItemName"/>
|
||||
public sealed class SetBonusItemName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, byte, ulong, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetBonusItemName)}.V2";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetBonusItemName"/>
|
||||
public GlamourerApiEc Invoke(string objectName, ApiBonusSlot slot, ulong itemId, uint key = 0, ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectName, (byte)slot, itemId, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, byte, ulong, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e) => (int)api.SetBonusItemName(a, (ApiBonusSlot)b, c, d, (ApplyFlag)e));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetMetaState"/>
|
||||
public sealed class SetMetaState(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, ulong, bool, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetMetaState)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetMetaState"/>
|
||||
public GlamourerApiEc Invoke(int objectIndex, MetaFlag types, bool newValue, uint key = 0,
|
||||
ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, (ulong)types, newValue, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, ulong, bool, uint, ulong, int> Provider(IDalamudPluginInterface pi,
|
||||
IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e) => (int)api.SetMetaState(a, (MetaFlag)b, c, d, (ApplyFlag)e));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetMetaStateName"/>
|
||||
public sealed class SetMetaStateName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, ulong, bool, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(SetMetaStateName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiItems.SetMetaStateName"/>
|
||||
public GlamourerApiEc Invoke(string objectName, MetaFlag types, bool newValue, uint key = 0,
|
||||
ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectName, (ulong)types, newValue, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, ulong, bool, uint, ulong, int> Provider(IDalamudPluginInterface pi,
|
||||
IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e) => (int)api.SetMetaStateName(a, (MetaFlag)b, c, d, (ApplyFlag)e));
|
||||
}
|
||||
@@ -1,52 +0,0 @@
|
||||
using Dalamud.Game.ClientState.Objects.Types;
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers.Legacy;
|
||||
|
||||
public sealed class GetDesignList(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<(string Name, Guid Identifier)[]>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(GetDesignList)}";
|
||||
|
||||
public new (string Name, Guid Identifier)[] Invoke()
|
||||
=> base.Invoke();
|
||||
}
|
||||
|
||||
public sealed class ApplyByGuid(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<Guid, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyByGuid)}";
|
||||
|
||||
public new void Invoke(Guid design, string name)
|
||||
=> base.Invoke(design, name);
|
||||
}
|
||||
|
||||
public sealed class ApplyByGuidOnce(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<Guid, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyByGuidOnce)}";
|
||||
|
||||
public new void Invoke(Guid design, string name)
|
||||
=> base.Invoke(design, name);
|
||||
}
|
||||
|
||||
public sealed class ApplyByGuidToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<Guid, ICharacter?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyByGuidToCharacter)}";
|
||||
|
||||
public new void Invoke(Guid design, ICharacter? character)
|
||||
=> base.Invoke(design, character);
|
||||
}
|
||||
|
||||
public sealed class ApplyByGuidOnceToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<Guid, ICharacter?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyByGuidOnceToCharacter)}";
|
||||
|
||||
public new void Invoke(Guid design, ICharacter? character)
|
||||
=> base.Invoke(design, character);
|
||||
}
|
||||
@@ -1,66 +0,0 @@
|
||||
using Dalamud.Game.ClientState.Objects.Types;
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Api;
|
||||
using Glamourer.Api.Enums;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers.Legacy;
|
||||
|
||||
public sealed class SetItem(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, byte, ulong, byte, uint, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItem)}";
|
||||
|
||||
public new GlamourerApiEc Invoke(ICharacter? character, byte slot, ulong itemId, byte stainId, uint key)
|
||||
=> (GlamourerApiEc)base.Invoke(character, slot, itemId, stainId, key);
|
||||
}
|
||||
|
||||
public sealed class SetItemOnce(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, byte, ulong, byte, uint, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItemOnce)}";
|
||||
|
||||
public new GlamourerApiEc Invoke(ICharacter? character, byte slot, ulong itemId, byte stainId, uint key)
|
||||
=> (GlamourerApiEc)base.Invoke(character, slot, itemId, stainId, key);
|
||||
}
|
||||
|
||||
public sealed class SetItemByActorName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, byte, ulong, byte, uint, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItemByActorName)}";
|
||||
|
||||
public new GlamourerApiEc Invoke(string actorName, byte slot, ulong itemId, byte stainId, uint key)
|
||||
=> (GlamourerApiEc)base.Invoke(actorName, slot, itemId, stainId, key);
|
||||
}
|
||||
|
||||
public sealed class SetItemOnceByActorName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, byte, ulong, byte, uint, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItemOnceByActorName)}";
|
||||
|
||||
public new GlamourerApiEc Invoke(string actorName, byte slot, ulong itemId, byte stainId, uint key)
|
||||
=> (GlamourerApiEc)base.Invoke(actorName, slot, itemId, stainId, key);
|
||||
}
|
||||
|
||||
public sealed class SetItemV2(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, byte, ulong, byte, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItem)}.V2";
|
||||
|
||||
public GlamourerApiEc Invoke(int objectIndex, ApiEquipSlot slot, ulong itemId, byte stain, uint key = 0, ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, (byte)slot, itemId, stain, key, (ulong)flags);
|
||||
}
|
||||
|
||||
public sealed class SetItemName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, byte, ulong, byte, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(SetItemName)}";
|
||||
|
||||
public GlamourerApiEc Invoke(string objectName, ApiEquipSlot slot, ulong itemId, byte stain, uint key = 0, ApplyFlag flags = ApplyFlag.Once)
|
||||
=> (GlamourerApiEc)Invoke(objectName, (byte)slot, itemId, stain, key, (ulong)flags);
|
||||
|
||||
public static FuncProvider<string, byte, ulong, byte, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiItems api)
|
||||
=> new(pi, Label, (a, b, c, d, e, f) => (int)api.SetItemName(a, (ApiEquipSlot)b, c, [d], e, (ApplyFlag)f));
|
||||
}
|
||||
@@ -1,15 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers.Legacy;
|
||||
|
||||
public sealed class ApiVersions(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<(int, int)>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApiVersions)}";
|
||||
|
||||
public new (int Major, int Minor) Invoke()
|
||||
=> base.Invoke();
|
||||
}
|
||||
@@ -1,250 +0,0 @@
|
||||
using Dalamud.Game.ClientState.Objects.Types;
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers.Legacy;
|
||||
|
||||
public sealed class Revert(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(Revert)}";
|
||||
|
||||
public new void Invoke(string characterName)
|
||||
=> base.Invoke(characterName);
|
||||
}
|
||||
|
||||
public sealed class RevertCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(RevertCharacter)}";
|
||||
|
||||
public new void Invoke(ICharacter? character)
|
||||
=> base.Invoke(character);
|
||||
}
|
||||
|
||||
public sealed class RevertLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(RevertLock)}";
|
||||
|
||||
public new void Invoke(string characterName, uint key)
|
||||
=> base.Invoke(characterName, key);
|
||||
}
|
||||
|
||||
public sealed class RevertCharacterLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(RevertCharacterLock)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, uint key)
|
||||
=> base.Invoke(character, key);
|
||||
}
|
||||
|
||||
public sealed class RevertToAutomation(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, bool>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(RevertToAutomation)}";
|
||||
|
||||
public new bool Invoke(string characterName, uint key)
|
||||
=> base.Invoke(characterName, key);
|
||||
}
|
||||
|
||||
public sealed class RevertToAutomationCharacter(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, uint, bool>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(RevertToAutomationCharacter)}";
|
||||
|
||||
public new bool Invoke(ICharacter? character, uint key)
|
||||
=> base.Invoke(character, key);
|
||||
}
|
||||
|
||||
public sealed class Unlock(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, uint, bool>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(Unlock)}";
|
||||
|
||||
public new bool Invoke(ICharacter? character, uint key)
|
||||
=> base.Invoke(character, key);
|
||||
}
|
||||
|
||||
public sealed class UnlockName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, bool>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(UnlockName)}";
|
||||
|
||||
public new bool Invoke(string characterName, uint key)
|
||||
=> base.Invoke(characterName, key);
|
||||
}
|
||||
|
||||
public static class StateChanged
|
||||
{
|
||||
public const string Label = $"Penumbra.{nameof(StateChanged)}";
|
||||
|
||||
public static EventSubscriber<int, nint, Lazy<string>> Subscriber(IDalamudPluginInterface pi,
|
||||
params Action<int, nint, Lazy<string>>[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
}
|
||||
|
||||
public sealed class GetAllCustomization(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, string?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(GetAllCustomization)}";
|
||||
|
||||
public new string? Invoke(string characterName)
|
||||
=> base.Invoke(characterName);
|
||||
}
|
||||
|
||||
public sealed class GetAllCustomizationFromCharacter(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, string?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(GetAllCustomizationFromCharacter)}";
|
||||
|
||||
public new string? Invoke(ICharacter? character)
|
||||
=> base.Invoke(character);
|
||||
}
|
||||
|
||||
public sealed class GetAllCustomizationLocked(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, string?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(GetAllCustomizationLocked)}";
|
||||
|
||||
public new string? Invoke(string characterName, uint key)
|
||||
=> base.Invoke(characterName, key);
|
||||
}
|
||||
|
||||
public sealed class GetAllCustomizationFromLockedCharacter(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<ICharacter?, uint, string?>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(GetAllCustomizationFromLockedCharacter)}";
|
||||
|
||||
public new string? Invoke(ICharacter? character, uint key)
|
||||
=> base.Invoke(character, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyAll(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAll)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64)
|
||||
=> base.Invoke(characterName, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyAllOnce(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAllOnce)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64)
|
||||
=> base.Invoke(characterName, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyAllToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAllToCharacter)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64)
|
||||
=> base.Invoke(character, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyAllOnceToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAllOnceToCharacter)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64)
|
||||
=> base.Invoke(character, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyEquipment(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyEquipment)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64)
|
||||
=> base.Invoke(characterName, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyEquipmentToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyEquipmentToCharacter)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64)
|
||||
=> base.Invoke(character, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyCustomization(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyCustomization)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64)
|
||||
=> base.Invoke(characterName, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyCustomizationToCharacter(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyCustomizationToCharacter)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64)
|
||||
=> base.Invoke(character, stateBase64);
|
||||
}
|
||||
|
||||
public sealed class ApplyAllLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAllLock)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64, uint key)
|
||||
=> base.Invoke(characterName, stateBase64, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyAllToCharacterLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyAllToCharacterLock)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64, uint key)
|
||||
=> base.Invoke(character, stateBase64, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyEquipmentLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyEquipmentLock)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64, uint key)
|
||||
=> base.Invoke(characterName, stateBase64, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyEquipmentToCharacterLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyEquipmentToCharacterLock)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64, uint key)
|
||||
=> base.Invoke(character, stateBase64, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyCustomizationLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<string, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyCustomizationLock)}";
|
||||
|
||||
public new void Invoke(string characterName, string stateBase64, uint key)
|
||||
=> base.Invoke(characterName, stateBase64, key);
|
||||
}
|
||||
|
||||
public sealed class ApplyOnlyCustomizationToCharacterLock(IDalamudPluginInterface pi)
|
||||
: ActionSubscriber<ICharacter?, string, uint>(pi, Label)
|
||||
{
|
||||
public const string Label = $"Glamourer.{nameof(ApplyOnlyCustomizationToCharacterLock)}";
|
||||
|
||||
public new void Invoke(ICharacter? character, string stateBase64, uint key)
|
||||
=> base.Invoke(character, stateBase64, key);
|
||||
}
|
||||
@@ -1,51 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Api;
|
||||
using Glamourer.Api.Helpers;
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers;
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiBase.ApiVersion"/>
|
||||
public sealed class ApiVersion(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<(int, int)>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(ApiVersion)}.V2";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiBase.ApiVersion"/>
|
||||
public new (int Major, int Minor) Invoke()
|
||||
=> base.Invoke();
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<(int, int)> Provider(IDalamudPluginInterface pi, IGlamourerApiBase api)
|
||||
=> new(pi, Label, () => api.ApiVersion);
|
||||
}
|
||||
|
||||
/// <summary> Triggered when the Glamourer API is initialized and ready. </summary>
|
||||
public static class Initialized
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(Initialized)}";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber Subscriber(IDalamudPluginInterface pi, params Action[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider Provider(IDalamudPluginInterface pi)
|
||||
=> new(pi, Label);
|
||||
}
|
||||
|
||||
/// <summary> Triggered when the Glamourer API is fully disposed and unavailable. </summary>
|
||||
public static class Disposed
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(Disposed)}";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber Subscriber(IDalamudPluginInterface pi, params Action[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider Provider(IDalamudPluginInterface pi)
|
||||
=> new(pi, Label);
|
||||
}
|
||||
@@ -1,311 +0,0 @@
|
||||
using Dalamud.Plugin;
|
||||
using Glamourer.Api.Api;
|
||||
using Glamourer.Api.Enums;
|
||||
using Glamourer.Api.Helpers;
|
||||
using Newtonsoft.Json.Linq;
|
||||
|
||||
namespace Glamourer.Api.IpcSubscribers;
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetState"/>
|
||||
public sealed class GetState(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, uint, (int, JObject?)>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(GetState)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetState"/>
|
||||
public new (GlamourerApiEc, JObject?) Invoke(int objectIndex, uint key = 0)
|
||||
{
|
||||
var (ec, data) = base.Invoke(objectIndex, key);
|
||||
return ((GlamourerApiEc)ec, data);
|
||||
}
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, uint, (int, JObject?)> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b) =>
|
||||
{
|
||||
var (ec, data) = api.GetState(a, b);
|
||||
return ((int)ec, data);
|
||||
});
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateName"/>
|
||||
public sealed class GetStateName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, (int, JObject?)>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(GetStateName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateName"/>
|
||||
public new (GlamourerApiEc, JObject?) Invoke(string objectName, uint key = 0)
|
||||
{
|
||||
var (ec, data) = base.Invoke(objectName, key);
|
||||
return ((GlamourerApiEc)ec, data);
|
||||
}
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, uint, (int, JObject?)> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (i, k) =>
|
||||
{
|
||||
var (ec, data) = api.GetStateName(i, k);
|
||||
return ((int)ec, data);
|
||||
});
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateBase64"/>
|
||||
public sealed class GetStateBase64(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, uint, (int, string?)>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(GetStateBase64)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateBase64"/>
|
||||
public new (GlamourerApiEc, string?) Invoke(int objectIndex, uint key = 0)
|
||||
{
|
||||
var (ec, data) = base.Invoke(objectIndex, key);
|
||||
return ((GlamourerApiEc)ec, data);
|
||||
}
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, uint, (int, string?)> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b) =>
|
||||
{
|
||||
var (ec, data) = api.GetStateBase64(a, b);
|
||||
return ((int)ec, data);
|
||||
});
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateBase64Name"/>
|
||||
public sealed class GetStateBase64Name(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, (int, string?)>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(GetStateBase64Name)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GetStateBase64Name"/>
|
||||
public new (GlamourerApiEc, string?) Invoke(string objectName, uint key = 0)
|
||||
{
|
||||
var (ec, data) = base.Invoke(objectName, key);
|
||||
return ((GlamourerApiEc)ec, data);
|
||||
}
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, uint, (int, string?)> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (i, k) =>
|
||||
{
|
||||
var (ec, data) = api.GetStateBase64Name(i, k);
|
||||
return ((int)ec, data);
|
||||
});
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyState"/>
|
||||
public sealed class ApplyState(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<object, int, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(ApplyState)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyState"/>
|
||||
public GlamourerApiEc Invoke(JObject state, int objectIndex, uint key = 0, ApplyFlag flags = ApplyFlagEx.StateDefault)
|
||||
=> (GlamourerApiEc)Invoke(state, objectIndex, key, (ulong)flags);
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyState"/>
|
||||
public GlamourerApiEc Invoke(string base64State, int objectIndex, uint key = 0, ApplyFlag flags = ApplyFlagEx.StateDefault)
|
||||
=> (GlamourerApiEc)Invoke(base64State, objectIndex, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<object, int, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c, d) => (int)api.ApplyState(a, b, c, (ApplyFlag)d));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyStateName"/>
|
||||
public sealed class ApplyStateName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<object, string, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(ApplyStateName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyState"/>
|
||||
public GlamourerApiEc Invoke(JObject state, string objectName, uint key = 0, ApplyFlag flags = ApplyFlagEx.StateDefault)
|
||||
=> (GlamourerApiEc)Invoke(state, objectName, key, (ulong)flags);
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.ApplyState"/>
|
||||
public GlamourerApiEc Invoke(string base64State, string objectName, uint key = 0, ApplyFlag flags = ApplyFlagEx.StateDefault)
|
||||
=> (GlamourerApiEc)Invoke(base64State, objectName, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<object, string, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c, d) => (int)api.ApplyStateName(a, b, c, (ApplyFlag)d));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertState"/>
|
||||
public sealed class RevertState(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(RevertState)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertState"/>
|
||||
public GlamourerApiEc Invoke(int objectIndex, uint key = 0, ApplyFlag flags = ApplyFlagEx.RevertDefault)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c) => (int)api.RevertState(a, b, (ApplyFlag)c));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertStateName"/>
|
||||
public sealed class RevertStateName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(RevertStateName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertStateName"/>
|
||||
public GlamourerApiEc Invoke(string objectName, uint key = 0, ApplyFlag flags = ApplyFlagEx.RevertDefault)
|
||||
=> (GlamourerApiEc)Invoke(objectName, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c) => (int)api.RevertStateName(a, b, (ApplyFlag)c));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockState"/>
|
||||
public sealed class UnlockState(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, uint, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(UnlockState)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockState"/>
|
||||
public new GlamourerApiEc Invoke(int objectIndex, uint key = 0)
|
||||
=> (GlamourerApiEc)base.Invoke(objectIndex, key);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, uint, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b) => (int)api.UnlockState(a, b));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockStateName"/>
|
||||
public sealed class UnlockStateName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(UnlockStateName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockStateName"/>
|
||||
public new GlamourerApiEc Invoke(string objectName, uint key = 0)
|
||||
=> (GlamourerApiEc)base.Invoke(objectName, key);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, uint, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b) => (int)api.UnlockStateName(a, b));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockAll"/>
|
||||
public sealed class UnlockAll(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<uint, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(UnlockAll)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.UnlockAll"/>
|
||||
public new int Invoke(uint key)
|
||||
=> base.Invoke(key);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<uint, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, api.UnlockAll);
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertToAutomation"/>
|
||||
public sealed class RevertToAutomation(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<int, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(RevertToAutomation)}.V2";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertToAutomation"/>
|
||||
public GlamourerApiEc Invoke(int objectIndex, uint key = 0, ApplyFlag flags = ApplyFlagEx.RevertDefault)
|
||||
=> (GlamourerApiEc)Invoke(objectIndex, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<int, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c) => (int)api.RevertToAutomation(a, b, (ApplyFlag)c));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertToAutomationName"/>
|
||||
public sealed class RevertToAutomationName(IDalamudPluginInterface pi)
|
||||
: FuncSubscriber<string, uint, ulong, int>(pi, Label)
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Glamourer.{nameof(RevertToAutomationName)}";
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.RevertToAutomationName"/>
|
||||
public GlamourerApiEc Invoke(string objectName, uint key = 0, ApplyFlag flags = ApplyFlagEx.RevertDefault)
|
||||
=> (GlamourerApiEc)Invoke(objectName, key, (ulong)flags);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static FuncProvider<string, uint, ulong, int> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (a, b, c) => (int)api.RevertToAutomationName(a, b, (ApplyFlag)c));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.StateChanged" />
|
||||
public static class StateChanged
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Penumbra.{nameof(StateChanged)}.V2";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber<nint> Subscriber(IDalamudPluginInterface pi, params Action<nint>[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider<nint> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (t => api.StateChanged += t, t => api.StateChanged -= t));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.StateChangedWithType" />
|
||||
public static class StateChangedWithType
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Penumbra.{nameof(StateChangedWithType)}";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber<nint, StateChangeType> Subscriber(IDalamudPluginInterface pi, params Action<nint, StateChangeType>[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider<nint, StateChangeType> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (t => api.StateChangedWithType += t, t => api.StateChangedWithType -= t));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.StateFinalized" />
|
||||
public static class StateFinalized
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Penumbra.{nameof(StateFinalized)}";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber<nint, StateFinalizationType> Subscriber(IDalamudPluginInterface pi, params Action<nint, StateFinalizationType>[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider<nint, StateFinalizationType> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (t => api.StateFinalized += t, t => api.StateFinalized -= t));
|
||||
}
|
||||
|
||||
/// <inheritdoc cref="IGlamourerApiState.GPoseChanged" />
|
||||
public static class GPoseChanged
|
||||
{
|
||||
/// <summary> The label. </summary>
|
||||
public const string Label = $"Penumbra.{nameof(GPoseChanged)}";
|
||||
|
||||
/// <summary> Create a new event subscriber. </summary>
|
||||
public static EventSubscriber<bool> Subscriber(IDalamudPluginInterface pi, params Action<bool>[] actions)
|
||||
=> new(pi, Label, actions);
|
||||
|
||||
/// <summary> Create a provider. </summary>
|
||||
public static EventProvider<bool> Provider(IDalamudPluginInterface pi, IGlamourerApiState api)
|
||||
=> new(pi, Label, (t => api.GPoseChanged += t, t => api.GPoseChanged -= t));
|
||||
}
|
||||
@@ -1,4 +0,0 @@
|
||||
# Glamourer
|
||||
|
||||
This is an auxiliary repository for Glamourers external API.
|
||||
For more information, see the [main repo](https://github.com/Ottermandias/Glamourer).
|
||||
@@ -1,13 +0,0 @@
|
||||
{
|
||||
"version": 1,
|
||||
"dependencies": {
|
||||
"net9.0-windows7.0": {
|
||||
"DotNet.ReproducibleBuilds": {
|
||||
"type": "Direct",
|
||||
"requested": "[1.2.25, )",
|
||||
"resolved": "1.2.25",
|
||||
"contentHash": "xCXiw7BCxHJ8pF6wPepRUddlh2dlQlbr81gXA72hdk4FLHkKXas7EH/n+fk5UCA/YfMqG1Z6XaPiUjDbUNBUzg=="
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2022 Mare Synchronos
|
||||
Copyright (c) 2022 Penumbra-Sync
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
1
MareAPI
Submodule
1
MareAPI
Submodule
Submodule MareAPI added at fa9b7bce43
@@ -5,7 +5,7 @@ VisualStudioVersion = 17.1.32328.378
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "MareSynchronos", "MareSynchronos\MareSynchronos.csproj", "{13C812E9-0D42-4B95-8646-40EEBF30636F}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "MareSynchronos.API", "UmbraAPI\MareSynchronosAPI\MareSynchronos.API.csproj", "{5A0B7434-8D89-4E90-B55C-B4A7AE1A6ADE}"
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "MareSynchronos.API", "MareAPI\MareSynchronosAPI\MareSynchronos.API.csproj", "{5A0B7434-8D89-4E90-B55C-B4A7AE1A6ADE}"
|
||||
EndProject
|
||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{585B740D-BA2C-429B-9CF3-B2D223423748}"
|
||||
ProjectSection(SolutionItems) = preProject
|
||||
|
||||
@@ -236,7 +236,6 @@ public sealed class FileCacheManager : IHostedService
|
||||
|
||||
foreach (var entry in cleanedPaths)
|
||||
{
|
||||
//_logger.LogDebug("Checking {path}", entry.Value);
|
||||
|
||||
if (dict.TryGetValue(entry.Value, out var entity))
|
||||
{
|
||||
@@ -366,8 +365,7 @@ public sealed class FileCacheManager : IHostedService
|
||||
|
||||
if (!entries.Exists(u => string.Equals(u.PrefixedFilePath, fileCache.PrefixedFilePath, StringComparison.OrdinalIgnoreCase)))
|
||||
{
|
||||
//_logger.LogTrace("Adding to DB: {hash} => {path}", fileCache.Hash, fileCache.PrefixedFilePath);
|
||||
entries.Add(fileCache);
|
||||
entries.Add(fileCache);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -389,7 +387,6 @@ public sealed class FileCacheManager : IHostedService
|
||||
private FileCacheEntity? GetValidatedFileCache(FileCacheEntity fileCache)
|
||||
{
|
||||
var resultingFileCache = ReplacePathPrefixes(fileCache);
|
||||
//_logger.LogTrace("Validating {path}", fileCache.PrefixedFilePath);
|
||||
resultingFileCache = Validate(resultingFileCache);
|
||||
return resultingFileCache;
|
||||
}
|
||||
|
||||
@@ -95,8 +95,6 @@ public sealed class IpcCallerBrio : IIpcCaller
|
||||
if (gameObject == null) return default;
|
||||
var data = await _dalamudUtilService.RunOnFrameworkThread(() => _brioGetModelTransform.InvokeFunc(gameObject)).ConfigureAwait(false);
|
||||
if (data.Item1 == null || data.Item2 == null || data.Item3 == null) return default;
|
||||
//_logger.LogDebug("Getting Transform from Actor {actor}", gameObject.Name.TextValue);
|
||||
|
||||
return new WorldData()
|
||||
{
|
||||
PositionX = data.Item1.Value.X,
|
||||
|
||||
48
MareSynchronos/Localization/LocalizationExtensions.cs
Normal file
48
MareSynchronos/Localization/LocalizationExtensions.cs
Normal file
@@ -0,0 +1,48 @@
|
||||
using System.Globalization;
|
||||
|
||||
namespace MareSynchronos.Localization;
|
||||
|
||||
public static class LocalizationExtensions
|
||||
{
|
||||
public static string Loc(this string fallbackEnglish, params object[] formatArgs)
|
||||
{
|
||||
var service = LocalizationService.Instance;
|
||||
if (service == null) return FormatFallback(fallbackEnglish, formatArgs);
|
||||
return service.GetString(fallbackEnglish, formatArgs);
|
||||
}
|
||||
|
||||
public static string LocKey(this string key, string fallbackEnglish, params object[] formatArgs)
|
||||
{
|
||||
var service = LocalizationService.Instance;
|
||||
if (service == null) return FormatFallback(fallbackEnglish, formatArgs);
|
||||
return service.GetString(key, fallbackEnglish, formatArgs);
|
||||
}
|
||||
|
||||
public static string LocLabel(this string labelWithId, params object[] formatArgs)
|
||||
{
|
||||
if (string.IsNullOrEmpty(labelWithId)) return labelWithId;
|
||||
|
||||
var separatorIndex = labelWithId.IndexOf("##", StringComparison.Ordinal);
|
||||
if (separatorIndex < 0)
|
||||
{
|
||||
return labelWithId.Loc(formatArgs);
|
||||
}
|
||||
|
||||
var label = labelWithId[..separatorIndex];
|
||||
var id = labelWithId[separatorIndex..];
|
||||
return string.Concat(label.Loc(formatArgs), id);
|
||||
}
|
||||
|
||||
private static string FormatFallback(string fallback, params object[] args)
|
||||
{
|
||||
if (args == null || args.Length == 0) return fallback;
|
||||
try
|
||||
{
|
||||
return string.Format(CultureInfo.CurrentCulture, fallback, args);
|
||||
}
|
||||
catch
|
||||
{
|
||||
return fallback;
|
||||
}
|
||||
}
|
||||
}
|
||||
231
MareSynchronos/Localization/LocalizationService.cs
Normal file
231
MareSynchronos/Localization/LocalizationService.cs
Normal file
@@ -0,0 +1,231 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.IO;
|
||||
using System.Reflection;
|
||||
using System.Text.Json;
|
||||
using System.Threading;
|
||||
using Dalamud.Plugin;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.MareConfiguration.Models;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace MareSynchronos.Localization;
|
||||
|
||||
public class LocalizationService
|
||||
{
|
||||
private readonly ILogger<LocalizationService> _logger;
|
||||
private readonly IDalamudPluginInterface _pluginInterface;
|
||||
private readonly MareConfigService _configService;
|
||||
private readonly Dictionary<LocalizationLanguage, Dictionary<string, string>> _translations = new();
|
||||
private readonly HashSet<string> _missingLocalizationsLogged = new(StringComparer.Ordinal);
|
||||
private readonly Lock _missingLocalizationsLock = new();
|
||||
|
||||
private static readonly JsonSerializerOptions JsonOptions = new()
|
||||
{
|
||||
AllowTrailingCommas = true
|
||||
};
|
||||
|
||||
public static LocalizationService? Instance { get; private set; }
|
||||
|
||||
public LocalizationService(ILogger<LocalizationService> logger, IDalamudPluginInterface pluginInterface, MareConfigService configService)
|
||||
{
|
||||
_logger = logger;
|
||||
_pluginInterface = pluginInterface;
|
||||
_configService = configService;
|
||||
|
||||
Instance = this;
|
||||
|
||||
foreach (var language in Enum.GetValues<LocalizationLanguage>())
|
||||
{
|
||||
_translations[language] = LoadLanguage(language);
|
||||
}
|
||||
}
|
||||
|
||||
public LocalizationLanguage CurrentLanguage => _configService.Current.Language;
|
||||
|
||||
public static IEnumerable<LocalizationLanguage> SupportedLanguages => Enum.GetValues<LocalizationLanguage>();
|
||||
|
||||
public string GetString(string fallbackEnglish, params object[] formatArgs)
|
||||
{
|
||||
return GetStringInternal(null, fallbackEnglish, formatArgs);
|
||||
}
|
||||
|
||||
public string GetString(string key, string fallbackEnglish, params object[] formatArgs)
|
||||
{
|
||||
return GetStringInternal(key, fallbackEnglish, formatArgs);
|
||||
}
|
||||
|
||||
public string GetLanguageDisplayName(LocalizationLanguage language)
|
||||
{
|
||||
var fallback = language switch
|
||||
{
|
||||
LocalizationLanguage.French => "Français",
|
||||
LocalizationLanguage.English => "English",
|
||||
_ => language.ToString()
|
||||
};
|
||||
|
||||
return GetRawString($"Language.DisplayName.{language}", fallback);
|
||||
}
|
||||
|
||||
public void ReloadLanguage(LocalizationLanguage language)
|
||||
{
|
||||
_translations[language] = LoadLanguage(language);
|
||||
}
|
||||
|
||||
public void ReloadAll()
|
||||
{
|
||||
foreach (var language in Enum.GetValues<LocalizationLanguage>())
|
||||
{
|
||||
ReloadLanguage(language);
|
||||
}
|
||||
}
|
||||
|
||||
private string GetStringInternal(string? key, string fallbackEnglish, params object[] formatArgs)
|
||||
{
|
||||
var usedKey = string.IsNullOrWhiteSpace(key) ? fallbackEnglish : key;
|
||||
var text = GetRawString(usedKey, fallbackEnglish);
|
||||
|
||||
if (formatArgs != null && formatArgs.Length > 0)
|
||||
{
|
||||
try
|
||||
{
|
||||
return string.Format(CultureInfo.CurrentCulture, text, formatArgs);
|
||||
}
|
||||
catch (FormatException ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Localization format mismatch for key {Key} with text '{Text}'", usedKey, text);
|
||||
try
|
||||
{
|
||||
return string.Format(CultureInfo.CurrentCulture, fallbackEnglish, formatArgs);
|
||||
}
|
||||
catch
|
||||
{
|
||||
return fallbackEnglish;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return text;
|
||||
}
|
||||
|
||||
private string GetRawString(string key, string fallbackEnglish)
|
||||
{
|
||||
if (TryGetString(CurrentLanguage, key, out var localized))
|
||||
{
|
||||
return localized;
|
||||
}
|
||||
|
||||
LogMissingLocalization(CurrentLanguage, key, fallbackEnglish);
|
||||
|
||||
if (TryGetString(LocalizationLanguage.English, key, out var english))
|
||||
{
|
||||
return english;
|
||||
}
|
||||
|
||||
if (CurrentLanguage != LocalizationLanguage.English)
|
||||
{
|
||||
LogMissingLocalization(LocalizationLanguage.English, key, fallbackEnglish);
|
||||
}
|
||||
|
||||
return fallbackEnglish;
|
||||
}
|
||||
|
||||
private bool TryGetString(LocalizationLanguage language, string key, out string value)
|
||||
{
|
||||
var dictionary = GetDictionary(language);
|
||||
if (dictionary.TryGetValue(key, out var text) && !string.IsNullOrWhiteSpace(text))
|
||||
{
|
||||
value = text;
|
||||
return true;
|
||||
}
|
||||
|
||||
value = string.Empty;
|
||||
return false;
|
||||
}
|
||||
|
||||
private Dictionary<string, string> GetDictionary(LocalizationLanguage language)
|
||||
{
|
||||
if (!_translations.TryGetValue(language, out var dictionary))
|
||||
{
|
||||
dictionary = LoadLanguage(language);
|
||||
_translations[language] = dictionary;
|
||||
}
|
||||
|
||||
return dictionary;
|
||||
}
|
||||
|
||||
private Dictionary<string, string> LoadLanguage(LocalizationLanguage language)
|
||||
{
|
||||
var baseDirectory = _pluginInterface.AssemblyLocation.DirectoryName ?? string.Empty;
|
||||
var localizationDirectory = Path.Combine(baseDirectory, "Localization");
|
||||
var languageCode = GetLanguageCode(language);
|
||||
var filePath = Path.Combine(localizationDirectory, $"{languageCode}.json");
|
||||
|
||||
Dictionary<string, string>? translations = null;
|
||||
|
||||
if (File.Exists(filePath))
|
||||
{
|
||||
try
|
||||
{
|
||||
using var stream = File.OpenRead(filePath);
|
||||
translations = DeserializeTranslations(stream);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Failed to load localization data from {FilePath}", filePath);
|
||||
}
|
||||
}
|
||||
|
||||
if (translations == null)
|
||||
{
|
||||
var assembly = Assembly.GetExecutingAssembly();
|
||||
var resourceName = $"{assembly.GetName().Name}.Localization.{languageCode}.json";
|
||||
using var resourceStream = assembly.GetManifestResourceStream(resourceName);
|
||||
if (resourceStream != null)
|
||||
{
|
||||
try
|
||||
{
|
||||
translations = DeserializeTranslations(resourceStream);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Failed to load embedded localization resource {Resource}", resourceName);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (translations == null)
|
||||
{
|
||||
_logger.LogDebug("Localization data for {Language} not found on disk or embedded, using empty dictionary.", language);
|
||||
translations = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
||||
}
|
||||
|
||||
return translations;
|
||||
}
|
||||
|
||||
private void LogMissingLocalization(LocalizationLanguage language, string key, string fallback)
|
||||
{
|
||||
var marker = $"{language}:{key}";
|
||||
using var scope = _missingLocalizationsLock.EnterScope();
|
||||
if (_missingLocalizationsLogged.Contains(marker)) return;
|
||||
_missingLocalizationsLogged.Add(marker);
|
||||
|
||||
_logger.LogDebug("Missing localization for {Language} ({Key}). Using fallback '{Fallback}'.", language, key, fallback);
|
||||
}
|
||||
|
||||
private static string GetLanguageCode(LocalizationLanguage language) => language switch
|
||||
{
|
||||
LocalizationLanguage.French => "fr",
|
||||
LocalizationLanguage.English => "en",
|
||||
_ => language.ToString().ToLowerInvariant(),
|
||||
};
|
||||
|
||||
private static Dictionary<string, string>? DeserializeTranslations(Stream stream)
|
||||
{
|
||||
var translations = JsonSerializer.Deserialize<Dictionary<string, string>>(stream, JsonOptions);
|
||||
return translations != null
|
||||
? new Dictionary<string, string>(translations, StringComparer.OrdinalIgnoreCase)
|
||||
: null;
|
||||
}
|
||||
}
|
||||
579
MareSynchronos/Localization/en.json
Normal file
579
MareSynchronos/Localization/en.json
Normal file
@@ -0,0 +1,579 @@
|
||||
{
|
||||
"Language.DisplayName.French": "French",
|
||||
"Language.DisplayName.English": "English",
|
||||
"Settings.Plugins.MandatoryHeading": "Mandatory Plugins",
|
||||
"Settings.Plugins.MandatoryLabel": "Mandatory Plugins:",
|
||||
"Settings.Plugins.OptionalHeading": "Optional Addons",
|
||||
"Settings.Plugins.OptionalLabel": "Optional Addons:",
|
||||
"Settings.Plugins.OptionalDescription": "These addons are not required for basic operation, but without them you may not see others as intended.",
|
||||
"Settings.Plugins.Tooltip.Available": "{0} is available and up to date.",
|
||||
"Settings.Plugins.Tooltip.Unavailable": "{0} is unavailable or not up to date.",
|
||||
"Settings.Plugins.WarningMandatoryMissing": "You need to install both Penumbra and Glamourer and keep them up to date to use Umbra.",
|
||||
"Settings.General.LocalizationHeading": "Localization",
|
||||
"Settings.General.Language": "Language",
|
||||
"Settings.General.Language.Description": "Select the plugin language. Any missing translations will be shown in English.",
|
||||
"Settings.General.NotesHeading": "Notes",
|
||||
"Settings.General.Notes.Export": "Export all your user notes to clipboard",
|
||||
"Settings.General.Notes.Import": "Import notes from clipboard",
|
||||
"Settings.General.Notes.Overwrite": "Overwrite existing notes",
|
||||
"Settings.General.Notes.Overwrite.Description": "If this option is selected all already existing notes for UIDs will be overwritten by the imported notes.",
|
||||
"Settings.General.Notes.Import.Success": "User Notes successfully imported",
|
||||
"Settings.General.Notes.Import.Failure": "Attempt to import notes from clipboard failed. Check formatting and try again",
|
||||
"Settings.General.Notes.OpenPopup": "Open Notes Popup on user addition",
|
||||
"Settings.General.Notes.OpenPopup.Description": "This will open a popup that allows you to set the notes for a user after successfully adding them to your individual pairs.",
|
||||
"Settings.Transfers.Blocked.Description": "Files that you attempted to upload or download that were forbidden to be transferred by their creators will appear here. If you see file paths from your drive here, then those files were not allowed to be uploaded. If you see hashes, those files were not allowed to be downloaded. Ask your paired friend to send you the mod in question through other means or acquire the mod yourself.",
|
||||
"Settings.Transfers.Blocked.Column.Hash": "Hash/Filename",
|
||||
"Settings.Transfers.Blocked.Column.ForbiddenBy": "Forbidden by",
|
||||
"Settings.Transfers.Blocked.Tab": "Blocked Transfers",
|
||||
"Settings.Transfers.Heading": "Transfer Settings",
|
||||
"Settings.Transfers.GlobalLimit.Label": "Global Download Speed Limit",
|
||||
"Settings.Transfers.GlobalLimit.Unit.BytePerSec": "Byte/s",
|
||||
"Settings.Transfers.GlobalLimit.Unit.KiloBytePerSec": "KB/s",
|
||||
"Settings.Transfers.GlobalLimit.Unit.MegaBytePerSec": "MB/s",
|
||||
"Settings.Transfers.GlobalLimit.Hint": "0 = No limit/infinite",
|
||||
"Settings.Transfers.MaxParallelDownloads": "Maximum Parallel Downloads",
|
||||
"Settings.Transfers.AutoDetect.Heading": "AutoDetect",
|
||||
"Settings.Transfers.AutoDetect.EnableNearby": "Enable Nearby detection (beta)",
|
||||
"Settings.Transfers.AutoDetect.AllowRequests": "Allow pair requests",
|
||||
"Settings.Transfers.AutoDetect.Notification.Title": "Nearby Detection",
|
||||
"Settings.Transfers.AutoDetect.Notification.Enabled": "Pair requests enabled: others can invite you.",
|
||||
"Settings.Transfers.AutoDetect.Notification.Disabled": "Pair requests disabled: others cannot invite you.",
|
||||
"Settings.Transfers.AutoDetect.MaxDistance": "Max distance (meters)",
|
||||
"Settings.Transfers.UI.Heading": "Transfer UI",
|
||||
"Settings.Transfers.UI.ShowWindow": "Show separate transfer window",
|
||||
"Settings.Transfers.UI.ShowWindow.Description": "The download window will show the current progress of outstanding downloads.\n\nWhat do W/Q/P/D stand for?\nW = Waiting for Slot (see Maximum Parallel Downloads)\nQ = Queued on Server, waiting for queue ready signal\nP = Processing download (aka downloading)\nD = Decompressing download",
|
||||
"Settings.Transfers.UI.EditWindowPosition": "Edit Transfer Window position",
|
||||
"Settings.Transfers.UI.ShowTransferBars": "Show transfer bars rendered below players",
|
||||
"Settings.Transfers.UI.ShowTransferBars.Description": "This will render a progress bar during the download at the feet of the player you are downloading from.",
|
||||
"Settings.Transfers.UI.ShowDownloadText": "Show Download Text",
|
||||
"Settings.Transfers.UI.ShowDownloadText.Description": "Shows download text (amount of MiB downloaded) in the transfer bars",
|
||||
"Settings.Transfers.UI.BarWidth": "Transfer Bar Width",
|
||||
"Settings.Transfers.UI.BarWidth.Description": "Width of the displayed transfer bars (will never be less wide than the displayed text)",
|
||||
"Settings.Transfers.UI.BarHeight": "Transfer Bar Height",
|
||||
"Settings.Transfers.UI.BarHeight.Description": "Height of the displayed transfer bars (will never be less tall than the displayed text)",
|
||||
"Settings.Transfers.UI.ShowUploading": "Show 'Uploading' text below players that are currently uploading",
|
||||
"Settings.Transfers.UI.ShowUploading.Description": "This will render an 'Uploading' text at the feet of the player that is in progress of uploading data.",
|
||||
"Settings.Transfers.UI.ShowUploadingBigText": "Large font for 'Uploading' text",
|
||||
"Settings.Transfers.UI.ShowUploadingBigText.Description": "This will render an 'Uploading' text in a larger font.",
|
||||
"Settings.Transfers.Current.Heading": "Current Transfers",
|
||||
"Settings.Transfers.Current.Tab": "Transfers",
|
||||
"Settings.Transfers.Current.Uploads": "Uploads",
|
||||
"Settings.Transfers.Current.Uploads.Column.File": "File",
|
||||
"Settings.Transfers.Current.Uploads.Column.Uploaded": "Uploaded",
|
||||
"Settings.Transfers.Current.Uploads.Column.Size": "Size",
|
||||
"Settings.Transfers.Current.Downloads": "Downloads",
|
||||
"Settings.Transfers.Current.Downloads.Column.User": "User",
|
||||
"Settings.Transfers.Current.Downloads.Column.Server": "Server",
|
||||
"Settings.Transfers.Current.Downloads.Column.Files": "Files",
|
||||
"Settings.Transfers.Current.Downloads.Column.Download": "Download",
|
||||
"Settings.Storage.Heading": "Storage",
|
||||
"Settings.Storage.Description": "Umbra stores downloaded files from paired people permanently. This is to improve loading performance and requiring less downloads. The storage governs itself by clearing data beyond the set storage size. Please set the storage size accordingly. It is not necessary to manually clear the storage.",
|
||||
"Settings.Service.ActionsHeading": "Service Actions",
|
||||
"Settings.Service.Actions.DeleteAccount": "Delete account",
|
||||
"Settings.Service.Actions.DeleteAccountPopup": "Delete your account?",
|
||||
"Settings.Service.Actions.DeleteAccount.Description": "Completely deletes your currently connected account.",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Body1": "Your account and all associated files and data on the service will be deleted.",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Body2": "Your UID will be removed from all pairing lists.",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Confirm": "Are you sure you want to continue?",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Cancel": "Cancel",
|
||||
"Settings.Service.SettingsHeading": "Service & Character Settings",
|
||||
"Settings.Service.ReconnectWarning": "For any changes to be applied to the current service you need to reconnect to the service.",
|
||||
"Settings.Service.Tabs.CharacterAssignments": "Character Assignments",
|
||||
"Settings.Service.Tabs.SecretKey": "Secret Key Management",
|
||||
"Settings.Service.Tabs.ServiceSettings": "Service Settings",
|
||||
"Settings.Service.Character.Assignments.Description": "Characters listed here will connect with the specified secret key.",
|
||||
"Settings.Service.Character.Assignments.TooltipCurrent": "Current character",
|
||||
"Settings.Service.Character.Assignments.DeleteTooltip": "Delete character assignment",
|
||||
"Settings.Service.Character.Assignments.AddCurrent": "Add current character",
|
||||
"Settings.Service.Character.Assignments.NoKeys": "You need to add a Secret Key first before adding Characters.",
|
||||
"Settings.Service.SecretKey.DisplayName": "Secret Key Display Name",
|
||||
"Settings.Service.SecretKey.Value": "Secret Key",
|
||||
"Settings.Service.SecretKey.AssignCurrent": "Assign current character",
|
||||
"Settings.Service.SecretKey.AssignTooltip": "Use this secret key for {0} @ {1}",
|
||||
"Settings.Service.SecretKey.Delete": "Delete Secret Key",
|
||||
"Settings.Service.SecretKey.DeleteTooltip": "Hold CTRL to delete this secret key entry",
|
||||
"Settings.Service.SecretKey.InUse": "This key is currently assigned to a character and cannot be edited or deleted.",
|
||||
"Settings.Service.SecretKey.Add": "Add new Secret Key",
|
||||
"Settings.Service.SecretKey.NewFriendlyName": "New Secret Key",
|
||||
"Settings.Service.SecretKey.RegisterAccount": "Register a new Umbra account",
|
||||
"Settings.Service.SecretKey.RegisterFailed": "An unknown error occured. Please try again later.",
|
||||
"Settings.Service.SecretKey.RegisterSuccess": "New account registered.\nPlease keep a copy of your secret key in case you need to reset your plugins, or to use it on another PC.",
|
||||
"Settings.Service.SecretKey.RegisteredFriendlyName": "{0} (registered {1})",
|
||||
"Settings.Service.SecretKey.Registering": "Sending request...",
|
||||
"Settings.Service.ServiceTab.Uri": "Service URI",
|
||||
"Settings.Service.ServiceTab.UriReadOnlyHint": "You cannot edit the URI of the main service.",
|
||||
"Settings.Service.ServiceTab.Name": "Service Name",
|
||||
"Settings.Service.ServiceTab.NameReadOnlyHint": "You cannot edit the name of the main service.",
|
||||
"Settings.Service.ServiceTab.Delete": "Delete Service",
|
||||
"Settings.Service.ServiceTab.DeleteHint": "Hold CTRL to delete this service",
|
||||
"Settings.Advanced.Heading": "Advanced",
|
||||
"Settings.Advanced.Tab": "Advanced",
|
||||
"Settings.Advanced.Api.Enable": "Enable Umbra Sync API",
|
||||
"Settings.Advanced.Api.Description": "Enables handling of the Umbra Sync API. This currently includes:\n\n - MCDF loading support for other plugins\n - Blocking Moodles applications to paired users\n\nIf the Umbra Sync plugin is loaded while this option is enabled, control of its API will be relinquished.",
|
||||
"Settings.Advanced.Api.Status.Active": "Umbra API active!",
|
||||
"Settings.Advanced.Api.Status.Disabled": "Umbra API inactive: Option is disabled",
|
||||
"Settings.Advanced.Api.Status.PluginLoaded": "Umbra API inactive: Umbra plugin is loaded",
|
||||
"Settings.Advanced.Api.Status.Unknown": "Umbra API inactive: Unknown reason",
|
||||
"Settings.Advanced.EventViewer.LogToDisk": "Log Event Viewer data to disk",
|
||||
"Settings.Advanced.EventViewer.Open": "Open Event Viewer",
|
||||
"Settings.Advanced.HoldCombat": "Hold application during combat",
|
||||
"Settings.Advanced.SerializedApplications": "Serialized player applications",
|
||||
"Settings.Advanced.SerializedApplications.Description": "Experimental - May reduce issues in crowded areas",
|
||||
"Settings.Advanced.DebugHeading": "Debug",
|
||||
"Settings.Advanced.Debug.LastCreatedTree": "Last created character data",
|
||||
"Settings.Advanced.Debug.CopyButton": "[DEBUG] Copy Last created Character Data to clipboard",
|
||||
"Settings.Advanced.Debug.CopyError": "ERROR: No created character data, cannot copy.",
|
||||
"Settings.Advanced.Debug.CopyTooltip": "Use this when reporting mods being rejected from the server.",
|
||||
"Settings.Advanced.LogLevel": "Log Level",
|
||||
"Settings.Advanced.Performance.LogCounters": "Log Performance Counters",
|
||||
"Settings.Advanced.Performance.LogCounters.Description": "Enabling this can incur a (slight) performance impact. Enabling this for extended periods of time is not recommended.",
|
||||
"Settings.Advanced.Performance.PrintStats": "Print Performance Stats to /xllog",
|
||||
"Settings.Advanced.Performance.PrintStatsRecent": "Print Performance Stats (last 60s) to /xllog",
|
||||
"Settings.Advanced.ActiveBlocks": "Active Character Blocks",
|
||||
"Settings.UI.Heading": "UI",
|
||||
"Settings.UI.EnableRightClick": "Enable Game Right Click Menu Entries",
|
||||
"Settings.UI.EnableRightClick.Description": "This will add Umbra related right click menu entries in the game UI on paired players.",
|
||||
"Settings.UI.EnableDtrEntry": "Display status and visible pair count in Server Info Bar",
|
||||
"Settings.UI.EnableDtrEntry.Description": "This will add Umbra connection status and visible pair count in the Server Info Bar.\nYou can further configure this through your Dalamud Settings.",
|
||||
"Settings.UI.Dtr.ShowUid": "Show visible character's UID in tooltip",
|
||||
"Settings.UI.Dtr.PreferNotes": "Prefer notes over player names in tooltip",
|
||||
"Settings.UI.Dtr.UseColors": "Color-code the Server Info Bar entry according to status",
|
||||
"Settings.UI.Dtr.ColorDefault": "Default",
|
||||
"Settings.UI.Dtr.ColorNotConnected": "Not Connected",
|
||||
"Settings.UI.Dtr.ColorPairsInRange": "Pairs in Range",
|
||||
"Settings.UI.NameColors.Enable": "Color nameplates of paired players",
|
||||
"Settings.UI.NameColors.Character": "Character Name Color",
|
||||
"Settings.UI.NameColors.Blocked": "Blocked Character Color",
|
||||
"Settings.UI.VisibleGroup": "Show separate Visible group",
|
||||
"Settings.UI.VisibleGroup.Description": "This will show all currently visible users in a special 'Visible' group in the main UI.",
|
||||
"Settings.UI.OfflineGroup": "Show separate Offline group",
|
||||
"Settings.UI.OfflineGroup.Description": "This will show all currently offline users in a special 'Offline' group in the main UI.",
|
||||
"Settings.UI.ShowPlayerNames": "Show player names",
|
||||
"Settings.UI.ShowPlayerNames.Description": "This will show character names instead of UIDs when possible",
|
||||
"Settings.UI.Profiles.Show": "Show Profiles on Hover",
|
||||
"Settings.UI.Profiles.Show.Description": "This will show the configured user profile after a set delay",
|
||||
"Settings.UI.Profiles.PopoutRight": "Popout profiles on the right",
|
||||
"Settings.UI.Profiles.PopoutRight.Description": "Will show profiles on the right side of the main UI",
|
||||
"Settings.UI.Profiles.HoverDelay": "Hover Delay",
|
||||
"Settings.UI.Profiles.HoverDelay.Description": "Delay until the profile should be displayed",
|
||||
"Settings.UI.Profiles.ShowNsfw": "Show profiles marked as NSFW",
|
||||
"Settings.UI.Profiles.ShowNsfw.Description": "Will show profiles that have the NSFW tag enabled",
|
||||
"Settings.Notifications.Heading": "Notifications",
|
||||
"Settings.Notifications.InfoDisplay": "Info Notification Display",
|
||||
"Settings.Notifications.InfoDisplay.Description": "The location where \"Info\" notifications will display.\n'Nowhere' will not show any Info notifications\n'Chat' will print Info notifications in chat\n'Toast' will show Warning toast notifications in the bottom right corner\n'Both' will show chat as well as the toast notification",
|
||||
"Settings.Notifications.WarningDisplay": "Warning Notification Display",
|
||||
"Settings.Notifications.WarningDisplay.Description": "The location where \"Warning\" notifications will display.\n'Nowhere' will not show any Warning notifications\n'Chat' will print Warning notifications in chat\n'Toast' will show Warning toast notifications in the bottom right corner\n'Both' will show chat as well as the toast notification",
|
||||
"Settings.Notifications.ErrorDisplay": "Error Notification Display",
|
||||
"Settings.Notifications.ErrorDisplay.Description": "The location where \"Error\" notifications will display.\n'Nowhere' will not show any Error notifications\n'Chat' will print Error notifications in chat\n'Toast' will show Error toast notifications in the bottom right corner\n'Both' will show chat as well as the toast notification",
|
||||
"Settings.Notifications.Location.Nowhere": "Nowhere",
|
||||
"Settings.Notifications.Location.Chat": "Chat",
|
||||
"Settings.Notifications.Location.Toast": "Toast",
|
||||
"Settings.Notifications.Location.Both": "Both",
|
||||
"Settings.Notifications.DisableOptionalWarnings": "Disable optional plugin warnings",
|
||||
"Settings.Notifications.DisableOptionalWarnings.Description": "Enabling this will not show any \"Warning\" labeled messages for missing optional plugins.",
|
||||
"Settings.Notifications.EnableOnlineNotifications": "Enable online notifications",
|
||||
"Settings.Notifications.EnableOnlineNotifications.Description": "Enabling this will show a small notification (type: Info) in the bottom right corner when pairs go online.",
|
||||
"Settings.Notifications.IndividualPairsOnly": "Notify only for individual pairs",
|
||||
"Settings.Notifications.IndividualPairsOnly.Description": "Enabling this will only show online notifications (type: Info) for individual pairs.",
|
||||
"Settings.Notifications.NamedPairsOnly": "Notify only for named pairs",
|
||||
"Settings.Notifications.NamedPairsOnly.Description": "Enabling this will only show online notifications (type: Info) for pairs where you have set an individual note.",
|
||||
"Compact.Version.UnsupportedTitle": "UNSUPPORTED VERSION",
|
||||
"Compact.Version.Outdated": "Your UmbraSync installation is out of date, the current version is {0}.{1}.{2}. It is highly recommended to keep UmbraSync up to date. Open /xlplugins and update the plugin.",
|
||||
"Compact.Toggle.IndividualPairs": "Individual pairs",
|
||||
"Compact.Toggle.Syncshells": "Syncshells",
|
||||
"Compact.AddUser.ModalTitle": "Set Notes for New User",
|
||||
"Compact.AddUser.Description": "You have successfully added {0}. Set a local note for the user in the field below:",
|
||||
"Compact.AddUser.NoteHint": "Note for {0}",
|
||||
"Compact.AddUser.Save": "Save Note",
|
||||
"Compact.AddCharacter.Button": "Add current character with secret key",
|
||||
"Compact.AddCharacter.SecretKeyLabel": "Secret Key",
|
||||
"Compact.AddCharacter.NoKeys": "No secret keys are configured for the current server.",
|
||||
"Compact.AddPair.Hint": "Other player's UID/Alias",
|
||||
"Compact.AddPair.Tooltip": "Pair with {0}",
|
||||
"Compact.AddPair.Tooltip.DefaultUser": "other user",
|
||||
"Compact.Filter.Hint": "Filter for UID/notes",
|
||||
"Compact.Filter.ToggleTooltip": "Hold Control to {0} pairing with {1} out of {2} displayed users.",
|
||||
"Compact.Filter.ToggleTooltip.Resume": "resume",
|
||||
"Compact.Filter.ToggleTooltip.Pause": "pause",
|
||||
"Compact.Filter.CooldownTooltip": "Next execution is available at {0} seconds",
|
||||
"Compact.Nearby.Title": "Nearby ({0})",
|
||||
"Compact.Nearby.Button": "Nearby",
|
||||
"Compact.Nearby.None": "No nearby players detected.",
|
||||
"Compact.Nearby.Tooltip.AlreadyPaired": "Already paired on Umbra",
|
||||
"Compact.Nearby.Tooltip.RequestsDisabled": "Pair requests are disabled for this player",
|
||||
"Compact.Nearby.Tooltip.SendInvite": "Send Umbra invitation",
|
||||
"Compact.Nearby.Tooltip.CannotInvite": "Unable to invite this player",
|
||||
"Compact.Nearby.Incoming": "Incoming requests",
|
||||
"Compact.Nearby.Incoming.Entry": "{0} [{1}]",
|
||||
"Compact.Nearby.Incoming.Accept": "Accept and add as pair",
|
||||
"Compact.Nearby.Incoming.Dismiss": "Dismiss request",
|
||||
"Compact.Header.SettingsTooltip": "Open the UmbraSync settings",
|
||||
"Compact.Header.CopyUid": "Copy your UID to clipboard",
|
||||
"Compact.ServerStatus.UsersOnline": "Users Online",
|
||||
"Compact.ServerStatus.Shard": "Shard: {0}",
|
||||
"Compact.ServerStatus.NotConnected": "Not connected to any server",
|
||||
"Compact.ServerStatus.EditProfile": "Edit your Profile",
|
||||
"Compact.ServerStatus.Disconnect": "Disconnect from {0}",
|
||||
"Compact.ServerStatus.Connect": "Connect to {0}",
|
||||
"Compact.ServerError.Connecting": "Attempting to connect to the server.",
|
||||
"Compact.ServerError.Reconnecting": "Connection to server interrupted, attempting to reconnect to the server.",
|
||||
"Compact.ServerError.Disconnected": "You are currently disconnected from the sync server.",
|
||||
"Compact.ServerError.Disconnecting": "Disconnecting from the server",
|
||||
"Compact.ServerError.Unauthorized": "Server Response: {0}",
|
||||
"Compact.ServerError.Offline": "Your selected sync server is currently offline.",
|
||||
"Compact.ServerError.VersionMismatch": "Your plugin or the server you are connecting to is out of date. Please update your plugin now. If you already did so, contact the server provider to update their server to the latest version.",
|
||||
"Compact.ServerError.RateLimited": "You are rate limited for (re)connecting too often. Disconnect, wait 10 minutes and try again.",
|
||||
"Compact.ServerError.NoSecretKey": "You have no secret key set for this current character. Use the button below or open the settings and set a secret key for the current character. You can reuse the same secret key for multiple characters.",
|
||||
"Compact.ServerError.MultiChara": "Your Character Configuration has multiple characters configured with same name and world. You will not be able to connect until you fix this issue. Remove the duplicates from the configuration in Settings -> Service Settings -> Character Management and reconnect manually after.",
|
||||
"Compact.Transfers.CharacterAnalysis": "Character Analysis",
|
||||
"Compact.Transfers.CharacterDataHub": "Character Data Hub",
|
||||
"Compact.UidText.Reconnecting": "Reconnecting",
|
||||
"Compact.UidText.Connecting": "Connecting",
|
||||
"Compact.UidText.Disconnected": "Disconnected",
|
||||
"Compact.UidText.Disconnecting": "Disconnecting",
|
||||
"Compact.UidText.Unauthorized": "Unauthorized",
|
||||
"Compact.UidText.VersionMismatch": "Version mismatch",
|
||||
"Compact.UidText.Offline": "Unavailable",
|
||||
"Compact.UidText.RateLimited": "Rate Limited",
|
||||
"Compact.UidText.NoSecretKey": "No Secret Key",
|
||||
"Compact.UidText.MultiChara": "Duplicate Characters",
|
||||
"UserPair.Status.Online": "User is online",
|
||||
"UserPair.Status.Offline": "User is offline",
|
||||
"UserPair.Tooltip.NotAddedBack": "{0} has not added you back",
|
||||
"UserPair.Tooltip.Paused": "Pairing with {0} is paused",
|
||||
"UserPair.Tooltip.Visible": "{0} is visible: {1}\nClick to target this player",
|
||||
"UserPair.Tooltip.Visible.LastPrefix": "(Last) ",
|
||||
"UserPair.Tooltip.Visible.ModsInfo": "Mods Info",
|
||||
"UserPair.Tooltip.Visible.FilesSize": "Files Size: {0}",
|
||||
"UserPair.Tooltip.Visible.Vram": "Approx. VRAM Usage: {0}",
|
||||
"UserPair.Tooltip.Visible.Tris": "Triangle Count (excl. Vanilla): {0}",
|
||||
"UserPair.Tooltip.Pause": "Pause pairing with {0}",
|
||||
"UserPair.Tooltip.Resume": "Resume pairing with {0}",
|
||||
"UserPair.Tooltip.Permission.Header": "Individual user permissions",
|
||||
"UserPair.Tooltip.Permission.Sound": "Sound sync disabled with {0}",
|
||||
"UserPair.Tooltip.Permission.Animation": "Animation sync disabled with {0}",
|
||||
"UserPair.Tooltip.Permission.Vfx": "VFX sync disabled with {0}",
|
||||
"UserPair.Tooltip.Permission.Status": "You: {0}, They: {1}",
|
||||
"UserPair.Tooltip.Permission.State.Disabled": "Disabled",
|
||||
"UserPair.Tooltip.Permission.State.Enabled": "Enabled",
|
||||
"UserPair.Tooltip.SharedData": "This user has shared {0} Character Data Sets with you.",
|
||||
"UserPair.Tooltip.SharedData.OpenHub": "Click to open the Character Data Hub and show the entries.",
|
||||
"UserPair.Menu.Target": "Target player",
|
||||
"UserPair.Menu.OpenProfile": "Open Profile",
|
||||
"UserPair.Menu.OpenProfile.Tooltip": "Opens the profile for this user in a new window",
|
||||
"UserPair.Menu.OpenAnalysis": "Open Analysis",
|
||||
"UserPair.Menu.ReloadData": "Reload last data",
|
||||
"UserPair.Menu.ReloadData.Tooltip": "This reapplies the last received character data to this character",
|
||||
"UserPair.Menu.CyclePause": "Cycle pause state",
|
||||
"UserPair.Menu.PairGroups": "Pair Groups",
|
||||
"UserPair.Menu.PairGroups.Tooltip": "Choose pair groups for {0}",
|
||||
"UserPair.Menu.EnableSoundSync": "Enable sound sync",
|
||||
"UserPair.Menu.DisableSoundSync": "Disable sound sync",
|
||||
"UserPair.Menu.EnableAnimationSync": "Enable animation sync",
|
||||
"UserPair.Menu.DisableAnimationSync": "Disable animation sync",
|
||||
"UserPair.Menu.EnableVfxSync": "Enable VFX sync",
|
||||
"UserPair.Menu.DisableVfxSync": "Disable VFX sync",
|
||||
"UserPair.Menu.Unpair": "Unpair Permanently",
|
||||
"UserPair.Menu.Unpair.Tooltip": "Hold CTRL and click to unpair permanently from {0}",
|
||||
"Popup.Generic.Close": "Close",
|
||||
"Popup.BanUser.Description": "User {0} will be banned and removed from this Syncshell.",
|
||||
"Popup.BanUser.ReasonHint": "Ban Reason",
|
||||
"Popup.BanUser.Button": "Ban User",
|
||||
"Popup.BanUser.ReasonNote": "The reason will be displayed in the banlist. The current server-side alias if present (Vanity ID) will automatically be attached to the reason.",
|
||||
"Popup.Report.Title": "Report {0} Profile",
|
||||
"Popup.Report.Note": "Note: Sending a report will disable the offending profile globally.\nThe report will be sent to the team of your currently connected server.\nDepending on the severity of the offense the users profile or account can be permanently disabled or banned.",
|
||||
"Popup.Report.Warning": "Report spam and wrong reports will not be tolerated and can lead to permanent account suspension.",
|
||||
"Popup.Report.Scope": "This is not for reporting misbehavior but solely for the actual profile. Reports that are not solely for the profile will be ignored.",
|
||||
"Popup.Report.Button": "Send Report",
|
||||
"PairGroups.Popup.Title": "Choose Groups for {0}",
|
||||
"PairGroups.Popup.SelectPrompt": "Select the groups you want {0} to be in.",
|
||||
"PairGroups.Popup.CreatePrompt": "Create a new group for {0}.",
|
||||
"PairGroups.Popup.NewGroupHint": "New Group",
|
||||
"PairGroups.SelectPairs.Title": "Choose Users for Group {0}",
|
||||
"PairGroups.SelectPairs.SelectPrompt": "Select users for group {0}",
|
||||
"PairGroups.SelectPairs.FilterHint": "Filter",
|
||||
"UidDisplay.Tooltip": "Left click to switch between UID display and nick\nRight click to change nick for {0}\nMiddle Mouse Button to open their profile in a separate window",
|
||||
"UidDisplay.EditNotes.Hint": "Nick/Notes",
|
||||
"DataAnalysis.WindowTitle": "Character Data Analysis",
|
||||
"DataAnalysis.Bc7.ModalTitle": "BC7 Conversion in Progress",
|
||||
"DataAnalysis.Bc7.Status": "BC7 Conversion in progress: {0}/{1}",
|
||||
"DataAnalysis.Bc7.CurrentFile": "Current file: {0}",
|
||||
"DataAnalysis.Bc7.Cancel": "Cancel conversion",
|
||||
"DataAnalysis.Description": "This window shows you all files and their sizes that are currently in use through your character and associated entities",
|
||||
"DataAnalysis.Analyzing": "Analyzing {0}/{1}",
|
||||
"DataAnalysis.Button.CancelAnalysis": "Cancel analysis",
|
||||
"DataAnalysis.Analyze.MissingNotice": "Some entries in the analysis have file size not determined yet, press the button below to analyze your current data",
|
||||
"DataAnalysis.Button.StartMissing": "Start analysis (missing entries)",
|
||||
"DataAnalysis.Button.StartAll": "Start analysis (recalculate all entries)",
|
||||
"DataAnalysis.TotalFiles": "Total files:",
|
||||
"DataAnalysis.Tooltip.FileSummary": "{0}: {1} files, size: {2}, compressed: {3}",
|
||||
"DataAnalysis.TotalSizeActual": "Total size (actual):",
|
||||
"DataAnalysis.TotalSizeDownload": "Total size (download size):",
|
||||
"DataAnalysis.Tooltip.CalculateDownloadSize": "Click \"Start analysis\" to calculate download size",
|
||||
"DataAnalysis.TotalTriangles": "Total modded model triangles: {0}",
|
||||
"DataAnalysis.FilesFor": "Files for {0}",
|
||||
"DataAnalysis.Object.SizeActual": "{0} size (actual):",
|
||||
"DataAnalysis.Object.SizeDownload": "{0} size (download size):",
|
||||
"DataAnalysis.Object.Vram": "{0} VRAM usage:",
|
||||
"DataAnalysis.Object.Triangles": "{0} modded model triangles: {1}",
|
||||
"DataAnalysis.FileGroup.Count": "{0} files",
|
||||
"DataAnalysis.FileGroup.SizeActual": "{0} files size (actual):",
|
||||
"DataAnalysis.FileGroup.SizeDownload": "{0} files size (download size):",
|
||||
"DataAnalysis.Bc7.EnableMode": "Enable BC7 Conversion Mode",
|
||||
"DataAnalysis.Bc7.WarningTitle": "WARNING BC7 CONVERSION:",
|
||||
"DataAnalysis.Bc7.WarningIrreversible": "Converting textures to BC7 is irreversible!",
|
||||
"DataAnalysis.Bc7.WarningDetails": "- Converting textures to BC7 will reduce their size (compressed and uncompressed) drastically. It is recommended to be used for large (4k+) textures.\n- Some textures, especially ones utilizing colorsets, might not be suited for BC7 conversion and might produce visual artifacts.\n- Before converting textures, make sure to have the original files of the mod you are converting so you can reimport it in case of issues.\n- Conversion will convert all found texture duplicates (entries with more than 1 file path) automatically.\n- Converting textures to BC7 is a very expensive operation and, depending on the amount of textures to convert, will take a while to complete.",
|
||||
"DataAnalysis.Bc7.StartConversion": "Start conversion of {0} texture(s)",
|
||||
"DataAnalysis.Table.Hash": "Hash",
|
||||
"DataAnalysis.Table.Filepaths": "Filepaths",
|
||||
"DataAnalysis.Table.Gamepaths": "Gamepaths",
|
||||
"DataAnalysis.Table.FileSize": "File Size",
|
||||
"DataAnalysis.Table.DownloadSize": "Download Size",
|
||||
"DataAnalysis.Table.Format": "Format",
|
||||
"DataAnalysis.Table.ConvertToBc7": "Convert to BC7",
|
||||
"DataAnalysis.Table.Triangles": "Triangles",
|
||||
"DataAnalysis.SelectedFile": "Selected file:",
|
||||
"DataAnalysis.LocalFilePath": "Local file path:",
|
||||
"DataAnalysis.MoreCount": "(and {0} more)",
|
||||
"DataAnalysis.GamePath": "Used by game path:",
|
||||
"DownloadUi.WindowTitle": "Umbra Downloads",
|
||||
"DownloadUi.UploadStatus": "Compressing+Uploading {0}/{1}",
|
||||
"DownloadUi.DownloadStatus": "{0} [W:{1}/Q:{2}/P:{3}/D:{4}]",
|
||||
"DownloadUi.UploadingLabel": "Uploading",
|
||||
"EventViewer.WindowTitle": "Event Viewer",
|
||||
"EventViewer.Button.Unfreeze": "Unfreeze View",
|
||||
"EventViewer.Button.Freeze": "Freeze View",
|
||||
"EventViewer.Tooltip.NewEvents": "New events are available. Click to resume updating.",
|
||||
"EventViewer.FilterLabel": "Filter lines",
|
||||
"EventViewer.Button.OpenLog": "Open EventLog folder",
|
||||
"EventViewer.Column.Time": "Time",
|
||||
"EventViewer.Column.Source": "Source",
|
||||
"EventViewer.Column.Uid": "UID",
|
||||
"EventViewer.Column.Character": "Character",
|
||||
"EventViewer.Column.Event": "Event",
|
||||
"EventViewer.Severity.Informational": "Informational",
|
||||
"EventViewer.Severity.Warning": "Warning",
|
||||
"EventViewer.Severity.Error": "Error",
|
||||
"EventViewer.NoValue": "--",
|
||||
"DtrEntry.EntryName": "Umbra",
|
||||
"DtrEntry.Tooltip.Connected": "Umbra: Connected",
|
||||
"DtrEntry.Tooltip.Disconnected": "Umbra: Not Connected",
|
||||
"PermissionWindow.Title": "Permissions for {0}",
|
||||
"PermissionWindow.Pause.Label": "Pause Sync",
|
||||
"PermissionWindow.Pause.HelpMain": "Pausing will completely cease any sync with this user.",
|
||||
"PermissionWindow.Pause.HelpNote": "Note: this is bidirectional, either user pausing will cease sync completely.",
|
||||
"PermissionWindow.OtherPaused.True": "{0} has paused you",
|
||||
"PermissionWindow.OtherPaused.False": "{0} has not paused you",
|
||||
"PermissionWindow.Sounds.Label": "Disable Sounds",
|
||||
"PermissionWindow.Sounds.HelpMain": "Disabling sounds will remove all sounds synced with this user on both sides.",
|
||||
"PermissionWindow.Sounds.HelpNote": "Note: this is bidirectional, either user disabling sound sync will stop sound sync on both sides.",
|
||||
"PermissionWindow.OtherSoundDisabled.True": "{0} has disabled sound sync with you",
|
||||
"PermissionWindow.OtherSoundDisabled.False": "{0} has not disabled sound sync with you",
|
||||
"PermissionWindow.Animations.Label": "Disable Animations",
|
||||
"PermissionWindow.Animations.HelpMain": "Disabling sounds will remove all animations synced with this user on both sides.",
|
||||
"PermissionWindow.Animations.HelpNote": "Note: this is bidirectional, either user disabling animation sync will stop animation sync on both sides.",
|
||||
"PermissionWindow.OtherAnimationDisabled.True": "{0} has disabled animation sync with you",
|
||||
"PermissionWindow.OtherAnimationDisabled.False": "{0} has not disabled animation sync with you",
|
||||
"PermissionWindow.Vfx.Label": "Disable VFX",
|
||||
"PermissionWindow.Vfx.HelpMain": "Disabling sounds will remove all VFX synced with this user on both sides.",
|
||||
"PermissionWindow.Vfx.HelpNote": "Note: this is bidirectional, either user disabling VFX sync will stop VFX sync on both sides.",
|
||||
"PermissionWindow.OtherVfxDisabled.True": "{0} has disabled VFX sync with you",
|
||||
"PermissionWindow.OtherVfxDisabled.False": "{0} has not disabled VFX sync with you",
|
||||
"PermissionWindow.Button.Save": "Save",
|
||||
"PermissionWindow.Tooltip.Save": "Save and apply all changes",
|
||||
"PermissionWindow.Button.Revert": "Revert",
|
||||
"PermissionWindow.Tooltip.Revert": "Revert all changes",
|
||||
"PermissionWindow.Button.Reset": "Reset to Default",
|
||||
"PermissionWindow.Tooltip.Reset": "This will set all permissions to their default setting",
|
||||
"EditProfile.WindowTitle": "Umbra Edit Profile",
|
||||
"EditProfile.CurrentProfile": "Current Profile (as saved on server)",
|
||||
"EditProfile.Button.UploadPicture": "Upload new profile picture",
|
||||
"EditProfile.Dialog.PictureTitle": "Select new Profile picture",
|
||||
"EditProfile.Tooltip.UploadPicture": "Select and upload a new profile picture",
|
||||
"EditProfile.Button.ClearPicture": "Clear uploaded profile picture",
|
||||
"EditProfile.Tooltip.ClearPicture": "Clear your currently uploaded profile picture",
|
||||
"EditProfile.Error.PictureTooLarge": "The profile picture must be a PNG file with a maximum height and width of 256px and 250KiB size",
|
||||
"EditProfile.Checkbox.Nsfw": "Profile is NSFW",
|
||||
"EditProfile.Help.Nsfw": "If your profile description or image can be considered NSFW, toggle this to ON",
|
||||
"EditProfile.DescriptionCounter": "Description {0}/1500",
|
||||
"EditProfile.PreviewLabel": "Preview (approximate)",
|
||||
"EditProfile.Button.SaveDescription": "Save Description",
|
||||
"EditProfile.Tooltip.SaveDescription": "Sets your profile description text",
|
||||
"EditProfile.Button.ClearDescription": "Clear Description",
|
||||
"EditProfile.Tooltip.ClearDescription": "Clears your profile description text",
|
||||
"Intro.Welcome.Title": "Welcome to Umbra",
|
||||
"Intro.Welcome.Paragraph1": "Umbra is a plugin that will replicate your full current character state including all Penumbra mods to other paired users. Note that you will have to have Penumbra as well as Glamourer installed to use this plugin.",
|
||||
"Intro.Welcome.Paragraph2": "We will have to setup a few things first before you can start using this plugin. Click on next to continue.",
|
||||
"Intro.Welcome.Note": "Note: Any modifications you have applied through anything but Penumbra cannot be shared and your character state on other clients might look broken because of this or others players mods might not apply on your end altogether. If you want to use this plugin you will have to move your mods to Penumbra.",
|
||||
"Intro.Welcome.Next": "Next",
|
||||
"Intro.Agreement.Title": "Agreement of Usage of Service",
|
||||
"Intro.Agreement.Callout": "READ THIS CAREFULLY",
|
||||
"Intro.Agreement.Timeout": "'I agree' button will be available in {0}s",
|
||||
"Intro.Agreement.Paragraph1": "To use Umbra, you must be over the age of 18, or 21 in some jurisdictions.",
|
||||
"Intro.Agreement.Paragraph2": "All of the mod files currently active on your character as well as your current character state will be uploaded to the service you registered yourself at automatically. The plugin will exclusively upload the necessary mod files and not the whole mod.",
|
||||
"Intro.Agreement.Paragraph3": "If you are on a data capped internet connection, higher fees due to data usage depending on the amount of downloaded and uploaded mod files might occur. Mod files will be compressed on up- and download to save on bandwidth usage. Due to varying up- and download speeds, changes in characters might not be visible immediately. Files present on the service that already represent your active mod files will not be uploaded again.",
|
||||
"Intro.Agreement.Paragraph4": "The mod files you are uploading are confidential and will not be distributed to parties other than the ones who are requesting the exact same mod files. Please think about who you are going to pair since it is unavoidable that they will receive and locally cache the necessary mod files that you have currently in use. Locally cached mod files will have arbitrary file names to discourage attempts at replicating the original mod.",
|
||||
"Intro.Agreement.Paragraph5": "The plugin creator tried their best to keep you secure. However, there is no guarantee for 100% security. Do not blindly pair your client with everyone.",
|
||||
"Intro.Agreement.Paragraph6": "Mod files that are saved on the service will remain on the service as long as there are requests for the files from clients. After a period of not being used, the mod files will be automatically deleted.",
|
||||
"Intro.Agreement.Paragraph7": "Accounts that are inactive for ninety (90) days will be deleted for privacy reasons.",
|
||||
"Intro.Agreement.Paragraph8": "Umbra is operated from servers located in the European Union. You agree not to upload any content to the service that violates EU law; and more specifically, German law.",
|
||||
"Intro.Agreement.Paragraph9": "You may delete your account at any time from within the Settings panel of the plugin. Any mods unique to you will then be removed from the server within 14 days.",
|
||||
"Intro.Agreement.Paragraph10": "This service is provided as-is.",
|
||||
"Intro.Agreement.Accept": "I agree",
|
||||
"Intro.Storage.Title": "File Storage Setup",
|
||||
"Intro.Storage.Description": "To not unnecessarily download files already present on your computer, Umbra will have to scan your Penumbra mod directory. Additionally, a local storage folder must be set where Umbra will download other character files to. Once the storage folder is set and the scan complete, this page will automatically forward to registration at a service.",
|
||||
"Intro.Storage.ScanNote": "Note: The initial scan, depending on the amount of mods you have, might take a while. Please wait until it is completed.",
|
||||
"Intro.Storage.Warning.FileCache": "Warning: once past this step you should not delete the FileCache.csv of Umbra in the Plugin Configurations folder of Dalamud. Otherwise on the next launch a full re-scan of the file cache database will be initiated.",
|
||||
"Intro.Storage.Warning.ScanHang": "Warning: if the scan is hanging and does nothing for a long time, chances are high your Penumbra folder is not set up properly.",
|
||||
"Intro.Storage.NoPenumbra": "You do not have a valid Penumbra path set. Open Penumbra and set up a valid path for the mod directory.",
|
||||
"Intro.Storage.StartScan": "Start Scan",
|
||||
"Intro.Storage.UseCompactor": "Use File Compactor",
|
||||
"Intro.Storage.CompactorDescription": "The File Compactor can save a tremendous amount of space on the hard disk for downloads through Umbra. It will incur a minor CPU penalty on download but can speed up loading of other characters. It is recommended to keep it enabled. You can change this setting later anytime in the Umbra settings.",
|
||||
"Intro.Registration.Title": "Service Registration",
|
||||
"Intro.Registration.Description": "To be able to use Umbra you will have to register an account.",
|
||||
"Intro.Registration.Support": "Refer to the instructions at the location you obtained this plugin for more information or support.",
|
||||
"Intro.Registration.NewAccountInfo": "If you have not used Umbra before, click below to register a new account.",
|
||||
"Intro.Registration.RegisterButton": "Register a new Umbra account",
|
||||
"Intro.Registration.SendingRequest": "Sending request...",
|
||||
"Intro.Registration.Success": "New account registered.\nPlease keep a copy of your secret key in case you need to reset your plugins, or to use it on another PC.",
|
||||
"Intro.Registration.UnknownError": "An unknown error occured. Please try again later.",
|
||||
"Intro.Registration.SecretKeyLabel": "Enter Secret Key",
|
||||
"Intro.Registration.SecretKeyLabelRegistered": "Secret Key",
|
||||
"Intro.Registration.SecretKeyInstructions": "If you already have a registered account, you can enter its secret key below to use it instead.",
|
||||
"Intro.Registration.SecretKeyLength": "Your secret key must be exactly 64 characters long.",
|
||||
"Intro.Registration.SecretKeyCharacters": "Your secret key can only contain ABCDEF and the numbers 0-9.",
|
||||
"Intro.Registration.SaveAndConnect": "Save and Connect",
|
||||
"Intro.Registration.SavedKeyRegistered": "(registered {0})",
|
||||
"Intro.Registration.SavedKeySetup": "Secret Key added on Setup ({0})",
|
||||
"Intro.ConnectionStatus.Connected": "Connected",
|
||||
"AutoDetect.Disabled": "Nearby detection is disabled. Enable it in Settings to start detecting nearby Umbra users.",
|
||||
"AutoDetect.MaxDistance": "Max distance (m)",
|
||||
"AutoDetect.Table.Name": "Name",
|
||||
"AutoDetect.Table.World": "World",
|
||||
"AutoDetect.Table.Distance": "Distance",
|
||||
"AutoDetect.Table.Status": "Status",
|
||||
"AutoDetect.Table.Action": "Action",
|
||||
"AutoDetect.World.Unknown": "-",
|
||||
"AutoDetect.Distance.Unknown": "-",
|
||||
"AutoDetect.Distance.Format": "{0:0.0} m",
|
||||
"AutoDetect.Status.Paired": "Paired",
|
||||
"AutoDetect.Status.RequestsDisabled": "Requests disabled",
|
||||
"AutoDetect.Status.OnUmbra": "On Umbra",
|
||||
"AutoDetect.Action.AlreadySynced": "Already sync",
|
||||
"AutoDetect.Action.RequestsDisabled": "Requests disabled",
|
||||
"AutoDetect.Action.SendRequest": "Send request",
|
||||
"PairGroups.ResumeAll": "Resume pairing with all pairs in {0}",
|
||||
"PairGroups.PauseAll": "Pause pairing with all pairs in {0}",
|
||||
"PairGroups.Menu.Title": "Group Flyout Menu",
|
||||
"PairGroups.Menu.AddPeople": "Add people to {0}",
|
||||
"PairGroups.Menu.AddPeople.Tooltip": "Add more users to Group {0}",
|
||||
"PairGroups.Menu.Delete": "Delete {0}",
|
||||
"PairGroups.Menu.Delete.Tooltip": "Delete Group {0} (Will not delete the pairs)\nHold CTRL to delete",
|
||||
"PairGroups.Tag.Unpaired": "Unpaired",
|
||||
"PairGroups.Tag.Offline": "Offline",
|
||||
"PairGroups.Tag.Online": "Online",
|
||||
"PairGroups.Tag.Contacts": "Contacts",
|
||||
"PairGroups.Tag.Visible": "Visible",
|
||||
"PairGroups.Header.WithCounts": "{0} ({1}/{2}/{3} Pairs)",
|
||||
"PairGroups.Header.Special": "{0} ({1} Pairs)",
|
||||
"PairGroups.Tooltip.Title": "Group {0}",
|
||||
"PairGroups.Tooltip.Visible": "{0} Pairs visible",
|
||||
"PairGroups.Tooltip.Online": "{0} Pairs online/paused",
|
||||
"PairGroups.Tooltip.Total": "{0} Pairs total",
|
||||
"GroupPanel.Join.InputHint": "Syncshell GID/Alias (leave empty to create)",
|
||||
"GroupPanel.Join.PasswordPopup": "Enter Syncshell Password",
|
||||
"GroupPanel.Create.PopupTitle": "Create Syncshell",
|
||||
"GroupPanel.Create.Tooltip": "Create Syncshell",
|
||||
"GroupPanel.Create.TooMany": "You cannot create more than {0} Syncshells",
|
||||
"GroupPanel.Join.Tooltip": "Join Syncshell {0}",
|
||||
"GroupPanel.Join.TooMany": "You cannot join more than {0} Syncshells",
|
||||
"GroupPanel.Join.Warning": "Before joining any Syncshells please be aware that you will be automatically paired with everyone in the Syncshell.",
|
||||
"GroupPanel.Join.EnterPassword": "Enter the password for Syncshell {0}:",
|
||||
"GroupPanel.Join.PasswordHint": "{0} Password",
|
||||
"GroupPanel.Join.Error": "An error occured during joining of this Syncshell: you either have joined the maximum amount of Syncshells ({0}), it does not exist, the password you entered is wrong, you already joined the Syncshell, the Syncshell is full ({1} users) or the Syncshell has closed invites.",
|
||||
"GroupPanel.Join.Button": "Join {0}",
|
||||
"GroupPanel.Create.ChooseType": "Choisissez le type de Syncshell \u00e0 cr\u00e9er.",
|
||||
"GroupPanel.Create.Permanent": "Permanente",
|
||||
"GroupPanel.Create.Temporary": "Temporaire",
|
||||
"GroupPanel.Create.AliasPrompt": "Donnez un nom \u00e0 votre Syncshell (optionnel) puis cr\u00e9ez-la.",
|
||||
"GroupPanel.Create.AliasHint": "Nom du Syncshell",
|
||||
"GroupPanel.Create.TempMaxDuration": "Dur\u00e9e maximale d'une Syncshell temporaire : 7 jours.",
|
||||
"GroupPanel.Create.TempExpires": "Expiration le {0:g} (heure locale).",
|
||||
"GroupPanel.Create.Instruction": "Appuyez sur le bouton ci-dessous pour cr\u00e9er une nouvelle Syncshell.",
|
||||
"GroupPanel.Create.Button": "Create Syncshell",
|
||||
"GroupPanel.Create.Error.NameInUse": "Le nom de la Syncshell est d\u00e9j\u00e0 utilis\u00e9.",
|
||||
"GroupPanel.Create.Result.Name": "Syncshell Name: {0}",
|
||||
"GroupPanel.Create.Result.Id": "Syncshell ID: {0}",
|
||||
"GroupPanel.Create.Result.Password": "Syncshell Password: {0}",
|
||||
"GroupPanel.Create.Result.ChangeLater": "You can change the Syncshell password later at any time.",
|
||||
"GroupPanel.Create.Result.TempExpires": "Cette Syncshell expirera le {0:g} (heure locale).",
|
||||
"GroupPanel.Create.Error.Generic": "You are already owner of the maximum amount of Syncshells (3) or joined the maximum amount of Syncshells (6). Relinquish ownership of your own Syncshells to someone else or leave existing Syncshells.",
|
||||
"GroupPanel.CommentHint": "Comment/Notes",
|
||||
"GroupPanel.CommentTooltip": "Hit ENTER to save\\nRight click to cancel",
|
||||
"GroupPanel.Banlist.Title": "Manage Banlist for {0}",
|
||||
"GroupPanel.Banlist.Refresh": "Refresh Banlist from Server",
|
||||
"GroupPanel.Banlist.Column.Uid": "UID",
|
||||
"GroupPanel.Banlist.Column.Alias": "Alias",
|
||||
"GroupPanel.Banlist.Column.By": "By",
|
||||
"GroupPanel.Banlist.Column.Date": "Date",
|
||||
"GroupPanel.Banlist.Column.Reason": "Reason",
|
||||
"GroupPanel.Banlist.Column.Actions": "Actions",
|
||||
"GroupPanel.Banlist.Unban": "Unban",
|
||||
"GroupPanel.Password.Title": "Change Syncshell Password",
|
||||
"GroupPanel.Password.Description": "Enter the new Syncshell password for Syncshell {0} here.",
|
||||
"GroupPanel.Password.Warning": "This action is irreversible",
|
||||
"GroupPanel.Password.Hint": "New password for {0}",
|
||||
"GroupPanel.Password.Button": "Change password",
|
||||
"GroupPanel.Password.Error.TooShort": "The selected password is too short. It must be at least 10 characters.",
|
||||
"GroupPanel.Invites.Title": "Create Bulk One-Time Invites",
|
||||
"GroupPanel.Invites.Description": "This allows you to create up to 100 one-time invites at once for the Syncshell {0}.\\nThe invites are valid for 24h after creation and will automatically expire.",
|
||||
"GroupPanel.Invites.CreateButton": "Create invites",
|
||||
"GroupPanel.Invites.Result": "A total of {0} invites have been created.",
|
||||
"GroupPanel.Invites.Copy": "Copy invites to clipboard",
|
||||
"GroupPanel.List.Visible": "Visible",
|
||||
"GroupPanel.List.Online": "Online",
|
||||
"GroupPanel.List.Offline": "Offline/Unknown",
|
||||
"GroupPanel.List.OfflineOmitted": "{0} offline users omitted from display.",
|
||||
"GroupPanel.Permissions.Header": "Syncshell permissions",
|
||||
"GroupPanel.Permissions.InvitesDisabled": "Syncshell is closed for joining",
|
||||
"GroupPanel.Permissions.SoundDisabledOwner": "Sound sync disabled through owner",
|
||||
"GroupPanel.Permissions.AnimationDisabledOwner": "Animation sync disabled through owner",
|
||||
"GroupPanel.Permissions.VfxDisabledOwner": "VFX sync disabled through owner",
|
||||
"GroupPanel.Permissions.OwnHeader": "Your permissions",
|
||||
"GroupPanel.Permissions.SoundDisabledSelf": "Sound sync disabled through you",
|
||||
"GroupPanel.Permissions.AnimationDisabledSelf": "Animation sync disabled through you",
|
||||
"GroupPanel.Permissions.VfxDisabledSelf": "VFX sync disabled through you",
|
||||
"GroupPanel.Permissions.NotePriority": "Note that syncshell permissions for disabling take precedence over your own set permissions",
|
||||
"GroupPanel.PauseToggle.Tooltip": "{0} pairing with all users in this Syncshell",
|
||||
"GroupPanel.PauseToggle.Resume": "Resume",
|
||||
"GroupPanel.PauseToggle.Pause": "Pause",
|
||||
"GroupPanel.Popup.Leave": "Leave Syncshell",
|
||||
"GroupPanel.Popup.LeaveTooltip": "Hold CTRL and click to leave this Syncshell{0}",
|
||||
"GroupPanel.Popup.LeaveWarning": "WARNING: This action is irreversible\\nLeaving an owned Syncshell will transfer the ownership to a random person in the Syncshell.",
|
||||
"GroupPanel.Popup.CopyId": "Copy ID",
|
||||
"GroupPanel.Popup.CopyIdTooltip": "Copy Syncshell ID to Clipboard",
|
||||
"GroupPanel.Popup.CopyNotes": "Copy Notes",
|
||||
"GroupPanel.Popup.CopyNotesTooltip": "Copies all your notes for all users in this Syncshell to the clipboard.\\nThey can be imported via Settings -> General -> Notes -> Import notes from clipboard",
|
||||
"GroupPanel.Popup.EnableSound": "Enable sound sync",
|
||||
"GroupPanel.Popup.DisableSound": "Disable sound sync",
|
||||
"GroupPanel.Popup.SoundTooltip": "Sets your allowance for sound synchronization for users of this syncshell.\\nDisabling the synchronization will stop applying sound modifications for users of this syncshell.\\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\\nNote: this setting does not apply to individual pairs that are also in the syncshell.",
|
||||
"GroupPanel.Popup.EnableAnimations": "Enable animations sync",
|
||||
"GroupPanel.Popup.DisableAnimations": "Disable animations sync",
|
||||
"GroupPanel.Popup.AnimTooltip": "Sets your allowance for animations synchronization for users of this syncshell.\\nDisabling the synchronization will stop applying animations modifications for users of this syncshell.\\nNote: this setting might also affect sound synchronization\\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\\nNote: this setting does not apply to individual pairs that are also in the syncshell.",
|
||||
"GroupPanel.Popup.EnableVfx": "Enable VFX sync",
|
||||
"GroupPanel.Popup.DisableVfx": "Disable VFX sync",
|
||||
"GroupPanel.Popup.VfxTooltip": "Sets your allowance for VFX synchronization for users of this syncshell.\\nDisabling the synchronization will stop applying VFX modifications for users of this syncshell.\\nNote: this setting might also affect animation synchronization to some degree\\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\\nNote: this setting does not apply to individual pairs that are also in the syncshell.",
|
||||
"GroupPanel.Syncshell.OwnerTooltip": "You are the owner of Syncshell {0}",
|
||||
"GroupPanel.Syncshell.ModeratorTooltip": "You are a moderator of Syncshell {0}",
|
||||
"GroupPanel.Syncshell.MemberCount": "{0}/{1}",
|
||||
"GroupPanel.Syncshell.MemberCountTooltip": "Membres connect\u00e9s / membres totaux\\nCapacit\u00e9 maximale : {0}\\nSyncshell ID: {1}",
|
||||
"GroupPanel.Syncshell.NameTooltip": "Left click to switch between GID display and comment\\nRight click to change comment for {0}\\n\\nUsers: {1}, Owner: {2}",
|
||||
"GroupPanel.Syncshell.TempTag": "(Temp)",
|
||||
"GroupPanel.Syncshell.TempExpires": "Expire le {0:g}",
|
||||
"GroupPanel.Syncshell.TempTooltip": "Syncshell temporaire",
|
||||
"GroupPanel.Create.Duration.SingleDay": "24h",
|
||||
"GroupPanel.Create.Duration.Days": "{0}j",
|
||||
"GroupPanel.Create.Duration.Hours": "{0}h",
|
||||
"GroupPanel.Invites.AmountLabel": "Amount",
|
||||
"GroupPanel.Popup.OpenAdmin": "Open Admin Panel"
|
||||
}
|
||||
579
MareSynchronos/Localization/fr.json
Normal file
579
MareSynchronos/Localization/fr.json
Normal file
@@ -0,0 +1,579 @@
|
||||
{
|
||||
"Language.DisplayName.French": "Français",
|
||||
"Language.DisplayName.English": "Anglais",
|
||||
"Settings.Plugins.MandatoryHeading": "",
|
||||
"Settings.Plugins.MandatoryLabel": "",
|
||||
"Settings.Plugins.OptionalHeading": "",
|
||||
"Settings.Plugins.OptionalLabel": "",
|
||||
"Settings.Plugins.OptionalDescription": "",
|
||||
"Settings.Plugins.Tooltip.Available": "",
|
||||
"Settings.Plugins.Tooltip.Unavailable": "",
|
||||
"Settings.Plugins.WarningMandatoryMissing": "",
|
||||
"Settings.General.LocalizationHeading": "Langue du plugin",
|
||||
"Settings.General.Language": "Langue",
|
||||
"Settings.General.Language.Description": "Sélectionnez la langue du plugin. Les traductions manquantes seront affichées en anglais.",
|
||||
"Settings.General.NotesHeading": "",
|
||||
"Settings.General.Notes.Export": "",
|
||||
"Settings.General.Notes.Import": "",
|
||||
"Settings.General.Notes.Overwrite": "",
|
||||
"Settings.General.Notes.Overwrite.Description": "",
|
||||
"Settings.General.Notes.Import.Success": "",
|
||||
"Settings.General.Notes.Import.Failure": "",
|
||||
"Settings.General.Notes.OpenPopup": "",
|
||||
"Settings.General.Notes.OpenPopup.Description": "",
|
||||
"Settings.Transfers.Blocked.Description": "",
|
||||
"Settings.Transfers.Blocked.Column.Hash": "",
|
||||
"Settings.Transfers.Blocked.Column.ForbiddenBy": "",
|
||||
"Settings.Transfers.Blocked.Tab": "",
|
||||
"Settings.Transfers.Heading": "",
|
||||
"Settings.Transfers.GlobalLimit.Label": "",
|
||||
"Settings.Transfers.GlobalLimit.Unit.BytePerSec": "",
|
||||
"Settings.Transfers.GlobalLimit.Unit.KiloBytePerSec": "",
|
||||
"Settings.Transfers.GlobalLimit.Unit.MegaBytePerSec": "",
|
||||
"Settings.Transfers.GlobalLimit.Hint": "",
|
||||
"Settings.Transfers.MaxParallelDownloads": "",
|
||||
"Settings.Transfers.AutoDetect.Heading": "",
|
||||
"Settings.Transfers.AutoDetect.EnableNearby": "",
|
||||
"Settings.Transfers.AutoDetect.AllowRequests": "",
|
||||
"Settings.Transfers.AutoDetect.Notification.Title": "",
|
||||
"Settings.Transfers.AutoDetect.Notification.Enabled": "",
|
||||
"Settings.Transfers.AutoDetect.Notification.Disabled": "",
|
||||
"Settings.Transfers.AutoDetect.MaxDistance": "",
|
||||
"Settings.Transfers.UI.Heading": "",
|
||||
"Settings.Transfers.UI.ShowWindow": "",
|
||||
"Settings.Transfers.UI.ShowWindow.Description": "",
|
||||
"Settings.Transfers.UI.EditWindowPosition": "",
|
||||
"Settings.Transfers.UI.ShowTransferBars": "",
|
||||
"Settings.Transfers.UI.ShowTransferBars.Description": "",
|
||||
"Settings.Transfers.UI.ShowDownloadText": "",
|
||||
"Settings.Transfers.UI.ShowDownloadText.Description": "",
|
||||
"Settings.Transfers.UI.BarWidth": "",
|
||||
"Settings.Transfers.UI.BarWidth.Description": "",
|
||||
"Settings.Transfers.UI.BarHeight": "",
|
||||
"Settings.Transfers.UI.BarHeight.Description": "",
|
||||
"Settings.Transfers.UI.ShowUploading": "",
|
||||
"Settings.Transfers.UI.ShowUploading.Description": "",
|
||||
"Settings.Transfers.UI.ShowUploadingBigText": "",
|
||||
"Settings.Transfers.UI.ShowUploadingBigText.Description": "",
|
||||
"Settings.Transfers.Current.Heading": "",
|
||||
"Settings.Transfers.Current.Tab": "",
|
||||
"Settings.Transfers.Current.Uploads": "",
|
||||
"Settings.Transfers.Current.Uploads.Column.File": "",
|
||||
"Settings.Transfers.Current.Uploads.Column.Uploaded": "",
|
||||
"Settings.Transfers.Current.Uploads.Column.Size": "",
|
||||
"Settings.Transfers.Current.Downloads": "",
|
||||
"Settings.Transfers.Current.Downloads.Column.User": "",
|
||||
"Settings.Transfers.Current.Downloads.Column.Server": "",
|
||||
"Settings.Transfers.Current.Downloads.Column.Files": "",
|
||||
"Settings.Transfers.Current.Downloads.Column.Download": "",
|
||||
"Settings.Storage.Heading": "",
|
||||
"Settings.Storage.Description": "",
|
||||
"Settings.Service.ActionsHeading": "",
|
||||
"Settings.Service.Actions.DeleteAccount": "",
|
||||
"Settings.Service.Actions.DeleteAccountPopup": "",
|
||||
"Settings.Service.Actions.DeleteAccount.Description": "",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Body1": "",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Body2": "",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Confirm": "",
|
||||
"Settings.Service.Actions.DeleteAccount.Popup.Cancel": "",
|
||||
"Settings.Service.SettingsHeading": "",
|
||||
"Settings.Service.ReconnectWarning": "",
|
||||
"Settings.Service.Tabs.CharacterAssignments": "",
|
||||
"Settings.Service.Tabs.SecretKey": "",
|
||||
"Settings.Service.Tabs.ServiceSettings": "",
|
||||
"Settings.Service.Character.Assignments.Description": "",
|
||||
"Settings.Service.Character.Assignments.TooltipCurrent": "",
|
||||
"Settings.Service.Character.Assignments.DeleteTooltip": "",
|
||||
"Settings.Service.Character.Assignments.AddCurrent": "",
|
||||
"Settings.Service.Character.Assignments.NoKeys": "",
|
||||
"Settings.Service.SecretKey.DisplayName": "",
|
||||
"Settings.Service.SecretKey.Value": "",
|
||||
"Settings.Service.SecretKey.AssignCurrent": "",
|
||||
"Settings.Service.SecretKey.AssignTooltip": "",
|
||||
"Settings.Service.SecretKey.Delete": "",
|
||||
"Settings.Service.SecretKey.DeleteTooltip": "",
|
||||
"Settings.Service.SecretKey.InUse": "",
|
||||
"Settings.Service.SecretKey.Add": "",
|
||||
"Settings.Service.SecretKey.NewFriendlyName": "",
|
||||
"Settings.Service.SecretKey.RegisterAccount": "",
|
||||
"Settings.Service.SecretKey.RegisterFailed": "",
|
||||
"Settings.Service.SecretKey.RegisterSuccess": "",
|
||||
"Settings.Service.SecretKey.RegisteredFriendlyName": "",
|
||||
"Settings.Service.SecretKey.Registering": "",
|
||||
"Settings.Service.ServiceTab.Uri": "",
|
||||
"Settings.Service.ServiceTab.UriReadOnlyHint": "",
|
||||
"Settings.Service.ServiceTab.Name": "",
|
||||
"Settings.Service.ServiceTab.NameReadOnlyHint": "",
|
||||
"Settings.Service.ServiceTab.Delete": "",
|
||||
"Settings.Service.ServiceTab.DeleteHint": "",
|
||||
"Settings.Advanced.Heading": "",
|
||||
"Settings.Advanced.Tab": "",
|
||||
"Settings.Advanced.Api.Enable": "",
|
||||
"Settings.Advanced.Api.Description": "",
|
||||
"Settings.Advanced.Api.Status.Active": "",
|
||||
"Settings.Advanced.Api.Status.Disabled": "",
|
||||
"Settings.Advanced.Api.Status.PluginLoaded": "",
|
||||
"Settings.Advanced.Api.Status.Unknown": "",
|
||||
"Settings.Advanced.EventViewer.LogToDisk": "",
|
||||
"Settings.Advanced.EventViewer.Open": "",
|
||||
"Settings.Advanced.HoldCombat": "",
|
||||
"Settings.Advanced.SerializedApplications": "",
|
||||
"Settings.Advanced.SerializedApplications.Description": "",
|
||||
"Settings.Advanced.DebugHeading": "",
|
||||
"Settings.Advanced.Debug.LastCreatedTree": "",
|
||||
"Settings.Advanced.Debug.CopyButton": "",
|
||||
"Settings.Advanced.Debug.CopyError": "",
|
||||
"Settings.Advanced.Debug.CopyTooltip": "",
|
||||
"Settings.Advanced.LogLevel": "",
|
||||
"Settings.Advanced.Performance.LogCounters": "",
|
||||
"Settings.Advanced.Performance.LogCounters.Description": "",
|
||||
"Settings.Advanced.Performance.PrintStats": "",
|
||||
"Settings.Advanced.Performance.PrintStatsRecent": "",
|
||||
"Settings.Advanced.ActiveBlocks": "",
|
||||
"Settings.UI.Heading": "",
|
||||
"Settings.UI.EnableRightClick": "",
|
||||
"Settings.UI.EnableRightClick.Description": "",
|
||||
"Settings.UI.EnableDtrEntry": "",
|
||||
"Settings.UI.EnableDtrEntry.Description": "",
|
||||
"Settings.UI.Dtr.ShowUid": "",
|
||||
"Settings.UI.Dtr.PreferNotes": "",
|
||||
"Settings.UI.Dtr.UseColors": "",
|
||||
"Settings.UI.Dtr.ColorDefault": "",
|
||||
"Settings.UI.Dtr.ColorNotConnected": "",
|
||||
"Settings.UI.Dtr.ColorPairsInRange": "",
|
||||
"Settings.UI.NameColors.Enable": "",
|
||||
"Settings.UI.NameColors.Character": "",
|
||||
"Settings.UI.NameColors.Blocked": "",
|
||||
"Settings.UI.VisibleGroup": "",
|
||||
"Settings.UI.VisibleGroup.Description": "",
|
||||
"Settings.UI.OfflineGroup": "",
|
||||
"Settings.UI.OfflineGroup.Description": "",
|
||||
"Settings.UI.ShowPlayerNames": "",
|
||||
"Settings.UI.ShowPlayerNames.Description": "",
|
||||
"Settings.UI.Profiles.Show": "",
|
||||
"Settings.UI.Profiles.Show.Description": "",
|
||||
"Settings.UI.Profiles.PopoutRight": "",
|
||||
"Settings.UI.Profiles.PopoutRight.Description": "",
|
||||
"Settings.UI.Profiles.HoverDelay": "",
|
||||
"Settings.UI.Profiles.HoverDelay.Description": "",
|
||||
"Settings.UI.Profiles.ShowNsfw": "",
|
||||
"Settings.UI.Profiles.ShowNsfw.Description": "",
|
||||
"Settings.Notifications.Heading": "",
|
||||
"Settings.Notifications.InfoDisplay": "",
|
||||
"Settings.Notifications.InfoDisplay.Description": "",
|
||||
"Settings.Notifications.WarningDisplay": "",
|
||||
"Settings.Notifications.WarningDisplay.Description": "",
|
||||
"Settings.Notifications.ErrorDisplay": "",
|
||||
"Settings.Notifications.ErrorDisplay.Description": "",
|
||||
"Settings.Notifications.Location.Nowhere": "",
|
||||
"Settings.Notifications.Location.Chat": "",
|
||||
"Settings.Notifications.Location.Toast": "",
|
||||
"Settings.Notifications.Location.Both": "",
|
||||
"Settings.Notifications.DisableOptionalWarnings": "",
|
||||
"Settings.Notifications.DisableOptionalWarnings.Description": "",
|
||||
"Settings.Notifications.EnableOnlineNotifications": "",
|
||||
"Settings.Notifications.EnableOnlineNotifications.Description": "",
|
||||
"Settings.Notifications.IndividualPairsOnly": "",
|
||||
"Settings.Notifications.IndividualPairsOnly.Description": "",
|
||||
"Settings.Notifications.NamedPairsOnly": "",
|
||||
"Settings.Notifications.NamedPairsOnly.Description": "",
|
||||
"Compact.Version.UnsupportedTitle": "",
|
||||
"Compact.Version.Outdated": "",
|
||||
"Compact.Toggle.IndividualPairs": "",
|
||||
"Compact.Toggle.Syncshells": "",
|
||||
"Compact.AddUser.ModalTitle": "",
|
||||
"Compact.AddUser.Description": "",
|
||||
"Compact.AddUser.NoteHint": "",
|
||||
"Compact.AddUser.Save": "",
|
||||
"Compact.AddCharacter.Button": "",
|
||||
"Compact.AddCharacter.SecretKeyLabel": "",
|
||||
"Compact.AddCharacter.NoKeys": "",
|
||||
"Compact.AddPair.Hint": "",
|
||||
"Compact.AddPair.Tooltip": "",
|
||||
"Compact.AddPair.Tooltip.DefaultUser": "",
|
||||
"Compact.Filter.Hint": "",
|
||||
"Compact.Filter.ToggleTooltip": "",
|
||||
"Compact.Filter.ToggleTooltip.Resume": "",
|
||||
"Compact.Filter.ToggleTooltip.Pause": "",
|
||||
"Compact.Filter.CooldownTooltip": "",
|
||||
"Compact.Nearby.Title": "",
|
||||
"Compact.Nearby.Button": "",
|
||||
"Compact.Nearby.None": "",
|
||||
"Compact.Nearby.Tooltip.AlreadyPaired": "",
|
||||
"Compact.Nearby.Tooltip.RequestsDisabled": "",
|
||||
"Compact.Nearby.Tooltip.SendInvite": "",
|
||||
"Compact.Nearby.Tooltip.CannotInvite": "",
|
||||
"Compact.Nearby.Incoming": "",
|
||||
"Compact.Nearby.Incoming.Entry": "",
|
||||
"Compact.Nearby.Incoming.Accept": "",
|
||||
"Compact.Nearby.Incoming.Dismiss": "",
|
||||
"Compact.Header.SettingsTooltip": "",
|
||||
"Compact.Header.CopyUid": "",
|
||||
"Compact.ServerStatus.UsersOnline": "",
|
||||
"Compact.ServerStatus.Shard": "",
|
||||
"Compact.ServerStatus.NotConnected": "",
|
||||
"Compact.ServerStatus.EditProfile": "",
|
||||
"Compact.ServerStatus.Disconnect": "",
|
||||
"Compact.ServerStatus.Connect": "",
|
||||
"Compact.ServerError.Connecting": "",
|
||||
"Compact.ServerError.Reconnecting": "",
|
||||
"Compact.ServerError.Disconnected": "",
|
||||
"Compact.ServerError.Disconnecting": "",
|
||||
"Compact.ServerError.Unauthorized": "",
|
||||
"Compact.ServerError.Offline": "",
|
||||
"Compact.ServerError.VersionMismatch": "",
|
||||
"Compact.ServerError.RateLimited": "",
|
||||
"Compact.ServerError.NoSecretKey": "",
|
||||
"Compact.ServerError.MultiChara": "",
|
||||
"Compact.Transfers.CharacterAnalysis": "",
|
||||
"Compact.Transfers.CharacterDataHub": "",
|
||||
"Compact.UidText.Reconnecting": "",
|
||||
"Compact.UidText.Connecting": "",
|
||||
"Compact.UidText.Disconnected": "",
|
||||
"Compact.UidText.Disconnecting": "",
|
||||
"Compact.UidText.Unauthorized": "",
|
||||
"Compact.UidText.VersionMismatch": "",
|
||||
"Compact.UidText.Offline": "",
|
||||
"Compact.UidText.RateLimited": "",
|
||||
"Compact.UidText.NoSecretKey": "",
|
||||
"Compact.UidText.MultiChara": "",
|
||||
"UserPair.Status.Online": "",
|
||||
"UserPair.Status.Offline": "",
|
||||
"UserPair.Tooltip.NotAddedBack": "",
|
||||
"UserPair.Tooltip.Paused": "",
|
||||
"UserPair.Tooltip.Visible": "",
|
||||
"UserPair.Tooltip.Visible.LastPrefix": "",
|
||||
"UserPair.Tooltip.Visible.ModsInfo": "",
|
||||
"UserPair.Tooltip.Visible.FilesSize": "",
|
||||
"UserPair.Tooltip.Visible.Vram": "",
|
||||
"UserPair.Tooltip.Visible.Tris": "",
|
||||
"UserPair.Tooltip.Pause": "",
|
||||
"UserPair.Tooltip.Resume": "",
|
||||
"UserPair.Tooltip.Permission.Header": "",
|
||||
"UserPair.Tooltip.Permission.Sound": "",
|
||||
"UserPair.Tooltip.Permission.Animation": "",
|
||||
"UserPair.Tooltip.Permission.Vfx": "",
|
||||
"UserPair.Tooltip.Permission.Status": "",
|
||||
"UserPair.Tooltip.Permission.State.Disabled": "",
|
||||
"UserPair.Tooltip.Permission.State.Enabled": "",
|
||||
"UserPair.Tooltip.SharedData": "",
|
||||
"UserPair.Tooltip.SharedData.OpenHub": "",
|
||||
"UserPair.Menu.Target": "",
|
||||
"UserPair.Menu.OpenProfile": "",
|
||||
"UserPair.Menu.OpenProfile.Tooltip": "",
|
||||
"UserPair.Menu.OpenAnalysis": "",
|
||||
"UserPair.Menu.ReloadData": "",
|
||||
"UserPair.Menu.ReloadData.Tooltip": "",
|
||||
"UserPair.Menu.CyclePause": "",
|
||||
"UserPair.Menu.PairGroups": "",
|
||||
"UserPair.Menu.PairGroups.Tooltip": "",
|
||||
"UserPair.Menu.EnableSoundSync": "",
|
||||
"UserPair.Menu.DisableSoundSync": "",
|
||||
"UserPair.Menu.EnableAnimationSync": "",
|
||||
"UserPair.Menu.DisableAnimationSync": "",
|
||||
"UserPair.Menu.EnableVfxSync": "",
|
||||
"UserPair.Menu.DisableVfxSync": "",
|
||||
"UserPair.Menu.Unpair": "",
|
||||
"UserPair.Menu.Unpair.Tooltip": "",
|
||||
"Popup.Generic.Close": "",
|
||||
"Popup.BanUser.Description": "",
|
||||
"Popup.BanUser.ReasonHint": "",
|
||||
"Popup.BanUser.Button": "",
|
||||
"Popup.BanUser.ReasonNote": "",
|
||||
"Popup.Report.Title": "",
|
||||
"Popup.Report.Note": "",
|
||||
"Popup.Report.Warning": "",
|
||||
"Popup.Report.Scope": "",
|
||||
"Popup.Report.Button": "",
|
||||
"PairGroups.Popup.Title": "",
|
||||
"PairGroups.Popup.SelectPrompt": "",
|
||||
"PairGroups.Popup.CreatePrompt": "",
|
||||
"PairGroups.Popup.NewGroupHint": "",
|
||||
"PairGroups.SelectPairs.Title": "",
|
||||
"PairGroups.SelectPairs.SelectPrompt": "",
|
||||
"PairGroups.SelectPairs.FilterHint": "",
|
||||
"UidDisplay.Tooltip": "",
|
||||
"UidDisplay.EditNotes.Hint": "",
|
||||
"DataAnalysis.WindowTitle": "",
|
||||
"DataAnalysis.Bc7.ModalTitle": "",
|
||||
"DataAnalysis.Bc7.Status": "",
|
||||
"DataAnalysis.Bc7.CurrentFile": "",
|
||||
"DataAnalysis.Bc7.Cancel": "",
|
||||
"DataAnalysis.Description": "",
|
||||
"DataAnalysis.Analyzing": "",
|
||||
"DataAnalysis.Button.CancelAnalysis": "",
|
||||
"DataAnalysis.Analyze.MissingNotice": "",
|
||||
"DataAnalysis.Button.StartMissing": "",
|
||||
"DataAnalysis.Button.StartAll": "",
|
||||
"DataAnalysis.TotalFiles": "",
|
||||
"DataAnalysis.Tooltip.FileSummary": "",
|
||||
"DataAnalysis.TotalSizeActual": "",
|
||||
"DataAnalysis.TotalSizeDownload": "",
|
||||
"DataAnalysis.Tooltip.CalculateDownloadSize": "",
|
||||
"DataAnalysis.TotalTriangles": "",
|
||||
"DataAnalysis.FilesFor": "",
|
||||
"DataAnalysis.Object.SizeActual": "",
|
||||
"DataAnalysis.Object.SizeDownload": "",
|
||||
"DataAnalysis.Object.Vram": "",
|
||||
"DataAnalysis.Object.Triangles": "",
|
||||
"DataAnalysis.FileGroup.Count": "",
|
||||
"DataAnalysis.FileGroup.SizeActual": "",
|
||||
"DataAnalysis.FileGroup.SizeDownload": "",
|
||||
"DataAnalysis.Bc7.EnableMode": "",
|
||||
"DataAnalysis.Bc7.WarningTitle": "",
|
||||
"DataAnalysis.Bc7.WarningIrreversible": "",
|
||||
"DataAnalysis.Bc7.WarningDetails": "",
|
||||
"DataAnalysis.Bc7.StartConversion": "",
|
||||
"DataAnalysis.Table.Hash": "",
|
||||
"DataAnalysis.Table.Filepaths": "",
|
||||
"DataAnalysis.Table.Gamepaths": "",
|
||||
"DataAnalysis.Table.FileSize": "",
|
||||
"DataAnalysis.Table.DownloadSize": "",
|
||||
"DataAnalysis.Table.Format": "",
|
||||
"DataAnalysis.Table.ConvertToBc7": "",
|
||||
"DataAnalysis.Table.Triangles": "",
|
||||
"DataAnalysis.SelectedFile": "",
|
||||
"DataAnalysis.LocalFilePath": "",
|
||||
"DataAnalysis.MoreCount": "",
|
||||
"DataAnalysis.GamePath": "",
|
||||
"DownloadUi.WindowTitle": "",
|
||||
"DownloadUi.UploadStatus": "",
|
||||
"DownloadUi.DownloadStatus": "",
|
||||
"DownloadUi.UploadingLabel": "",
|
||||
"EventViewer.WindowTitle": "",
|
||||
"EventViewer.Button.Unfreeze": "",
|
||||
"EventViewer.Button.Freeze": "",
|
||||
"EventViewer.Tooltip.NewEvents": "",
|
||||
"EventViewer.FilterLabel": "",
|
||||
"EventViewer.Button.OpenLog": "",
|
||||
"EventViewer.Column.Time": "",
|
||||
"EventViewer.Column.Source": "",
|
||||
"EventViewer.Column.Uid": "",
|
||||
"EventViewer.Column.Character": "",
|
||||
"EventViewer.Column.Event": "",
|
||||
"EventViewer.Severity.Informational": "",
|
||||
"EventViewer.Severity.Warning": "",
|
||||
"EventViewer.Severity.Error": "",
|
||||
"EventViewer.NoValue": "",
|
||||
"DtrEntry.EntryName": "",
|
||||
"DtrEntry.Tooltip.Connected": "",
|
||||
"DtrEntry.Tooltip.Disconnected": "",
|
||||
"PermissionWindow.Title": "",
|
||||
"PermissionWindow.Pause.Label": "",
|
||||
"PermissionWindow.Pause.HelpMain": "",
|
||||
"PermissionWindow.Pause.HelpNote": "",
|
||||
"PermissionWindow.OtherPaused.True": "",
|
||||
"PermissionWindow.OtherPaused.False": "",
|
||||
"PermissionWindow.Sounds.Label": "",
|
||||
"PermissionWindow.Sounds.HelpMain": "",
|
||||
"PermissionWindow.Sounds.HelpNote": "",
|
||||
"PermissionWindow.OtherSoundDisabled.True": "",
|
||||
"PermissionWindow.OtherSoundDisabled.False": "",
|
||||
"PermissionWindow.Animations.Label": "",
|
||||
"PermissionWindow.Animations.HelpMain": "",
|
||||
"PermissionWindow.Animations.HelpNote": "",
|
||||
"PermissionWindow.OtherAnimationDisabled.True": "",
|
||||
"PermissionWindow.OtherAnimationDisabled.False": "",
|
||||
"PermissionWindow.Vfx.Label": "",
|
||||
"PermissionWindow.Vfx.HelpMain": "",
|
||||
"PermissionWindow.Vfx.HelpNote": "",
|
||||
"PermissionWindow.OtherVfxDisabled.True": "",
|
||||
"PermissionWindow.OtherVfxDisabled.False": "",
|
||||
"PermissionWindow.Button.Save": "",
|
||||
"PermissionWindow.Tooltip.Save": "",
|
||||
"PermissionWindow.Button.Revert": "",
|
||||
"PermissionWindow.Tooltip.Revert": "",
|
||||
"PermissionWindow.Button.Reset": "",
|
||||
"PermissionWindow.Tooltip.Reset": "",
|
||||
"EditProfile.WindowTitle": "",
|
||||
"EditProfile.CurrentProfile": "",
|
||||
"EditProfile.Button.UploadPicture": "",
|
||||
"EditProfile.Dialog.PictureTitle": "",
|
||||
"EditProfile.Tooltip.UploadPicture": "",
|
||||
"EditProfile.Button.ClearPicture": "",
|
||||
"EditProfile.Tooltip.ClearPicture": "",
|
||||
"EditProfile.Error.PictureTooLarge": "",
|
||||
"EditProfile.Checkbox.Nsfw": "",
|
||||
"EditProfile.Help.Nsfw": "",
|
||||
"EditProfile.DescriptionCounter": "",
|
||||
"EditProfile.PreviewLabel": "",
|
||||
"EditProfile.Button.SaveDescription": "",
|
||||
"EditProfile.Tooltip.SaveDescription": "",
|
||||
"EditProfile.Button.ClearDescription": "",
|
||||
"EditProfile.Tooltip.ClearDescription": "",
|
||||
"Intro.Welcome.Title": "",
|
||||
"Intro.Welcome.Paragraph1": "",
|
||||
"Intro.Welcome.Paragraph2": "",
|
||||
"Intro.Welcome.Note": "",
|
||||
"Intro.Welcome.Next": "",
|
||||
"Intro.Agreement.Title": "",
|
||||
"Intro.Agreement.Callout": "",
|
||||
"Intro.Agreement.Timeout": "",
|
||||
"Intro.Agreement.Paragraph1": "",
|
||||
"Intro.Agreement.Paragraph2": "",
|
||||
"Intro.Agreement.Paragraph3": "",
|
||||
"Intro.Agreement.Paragraph4": "",
|
||||
"Intro.Agreement.Paragraph5": "",
|
||||
"Intro.Agreement.Paragraph6": "",
|
||||
"Intro.Agreement.Paragraph7": "",
|
||||
"Intro.Agreement.Paragraph8": "",
|
||||
"Intro.Agreement.Paragraph9": "",
|
||||
"Intro.Agreement.Paragraph10": "",
|
||||
"Intro.Agreement.Accept": "",
|
||||
"Intro.Storage.Title": "",
|
||||
"Intro.Storage.Description": "",
|
||||
"Intro.Storage.ScanNote": "",
|
||||
"Intro.Storage.Warning.FileCache": "",
|
||||
"Intro.Storage.Warning.ScanHang": "",
|
||||
"Intro.Storage.NoPenumbra": "",
|
||||
"Intro.Storage.StartScan": "",
|
||||
"Intro.Storage.UseCompactor": "",
|
||||
"Intro.Storage.CompactorDescription": "",
|
||||
"Intro.Registration.Title": "",
|
||||
"Intro.Registration.Description": "",
|
||||
"Intro.Registration.Support": "",
|
||||
"Intro.Registration.NewAccountInfo": "",
|
||||
"Intro.Registration.RegisterButton": "",
|
||||
"Intro.Registration.SendingRequest": "",
|
||||
"Intro.Registration.Success": "",
|
||||
"Intro.Registration.UnknownError": "",
|
||||
"Intro.Registration.SecretKeyLabel": "",
|
||||
"Intro.Registration.SecretKeyLabelRegistered": "",
|
||||
"Intro.Registration.SecretKeyInstructions": "",
|
||||
"Intro.Registration.SecretKeyLength": "",
|
||||
"Intro.Registration.SecretKeyCharacters": "",
|
||||
"Intro.Registration.SaveAndConnect": "",
|
||||
"Intro.Registration.SavedKeyRegistered": "",
|
||||
"Intro.Registration.SavedKeySetup": "",
|
||||
"Intro.ConnectionStatus.Connected": "",
|
||||
"AutoDetect.Disabled": "",
|
||||
"AutoDetect.MaxDistance": "",
|
||||
"AutoDetect.Table.Name": "",
|
||||
"AutoDetect.Table.World": "",
|
||||
"AutoDetect.Table.Distance": "",
|
||||
"AutoDetect.Table.Status": "",
|
||||
"AutoDetect.Table.Action": "",
|
||||
"AutoDetect.World.Unknown": "",
|
||||
"AutoDetect.Distance.Unknown": "",
|
||||
"AutoDetect.Distance.Format": "",
|
||||
"AutoDetect.Status.Paired": "",
|
||||
"AutoDetect.Status.RequestsDisabled": "",
|
||||
"AutoDetect.Status.OnUmbra": "",
|
||||
"AutoDetect.Action.AlreadySynced": "",
|
||||
"AutoDetect.Action.RequestsDisabled": "",
|
||||
"AutoDetect.Action.SendRequest": "",
|
||||
"PairGroups.ResumeAll": "",
|
||||
"PairGroups.PauseAll": "",
|
||||
"PairGroups.Menu.Title": "",
|
||||
"PairGroups.Menu.AddPeople": "",
|
||||
"PairGroups.Menu.AddPeople.Tooltip": "",
|
||||
"PairGroups.Menu.Delete": "",
|
||||
"PairGroups.Menu.Delete.Tooltip": "",
|
||||
"PairGroups.Tag.Unpaired": "",
|
||||
"PairGroups.Tag.Offline": "",
|
||||
"PairGroups.Tag.Online": "",
|
||||
"PairGroups.Tag.Contacts": "",
|
||||
"PairGroups.Tag.Visible": "",
|
||||
"PairGroups.Header.WithCounts": "",
|
||||
"PairGroups.Header.Special": "",
|
||||
"PairGroups.Tooltip.Title": "",
|
||||
"PairGroups.Tooltip.Visible": "",
|
||||
"PairGroups.Tooltip.Online": "",
|
||||
"PairGroups.Tooltip.Total": "",
|
||||
"GroupPanel.Join.InputHint": "",
|
||||
"GroupPanel.Join.PasswordPopup": "",
|
||||
"GroupPanel.Create.PopupTitle": "",
|
||||
"GroupPanel.Create.Tooltip": "",
|
||||
"GroupPanel.Create.TooMany": "",
|
||||
"GroupPanel.Join.Tooltip": "",
|
||||
"GroupPanel.Join.TooMany": "",
|
||||
"GroupPanel.Join.Warning": "",
|
||||
"GroupPanel.Join.EnterPassword": "",
|
||||
"GroupPanel.Join.PasswordHint": "",
|
||||
"GroupPanel.Join.Error": "",
|
||||
"GroupPanel.Join.Button": "",
|
||||
"GroupPanel.Create.ChooseType": "",
|
||||
"GroupPanel.Create.Permanent": "",
|
||||
"GroupPanel.Create.Temporary": "",
|
||||
"GroupPanel.Create.AliasPrompt": "",
|
||||
"GroupPanel.Create.AliasHint": "",
|
||||
"GroupPanel.Create.TempMaxDuration": "",
|
||||
"GroupPanel.Create.TempExpires": "",
|
||||
"GroupPanel.Create.Instruction": "",
|
||||
"GroupPanel.Create.Button": "",
|
||||
"GroupPanel.Create.Error.NameInUse": "",
|
||||
"GroupPanel.Create.Result.Name": "",
|
||||
"GroupPanel.Create.Result.Id": "",
|
||||
"GroupPanel.Create.Result.Password": "",
|
||||
"GroupPanel.Create.Result.ChangeLater": "",
|
||||
"GroupPanel.Create.Result.TempExpires": "",
|
||||
"GroupPanel.Create.Error.Generic": "",
|
||||
"GroupPanel.CommentHint": "",
|
||||
"GroupPanel.CommentTooltip": "",
|
||||
"GroupPanel.Banlist.Title": "",
|
||||
"GroupPanel.Banlist.Refresh": "",
|
||||
"GroupPanel.Banlist.Column.Uid": "",
|
||||
"GroupPanel.Banlist.Column.Alias": "",
|
||||
"GroupPanel.Banlist.Column.By": "",
|
||||
"GroupPanel.Banlist.Column.Date": "",
|
||||
"GroupPanel.Banlist.Column.Reason": "",
|
||||
"GroupPanel.Banlist.Column.Actions": "",
|
||||
"GroupPanel.Banlist.Unban": "",
|
||||
"GroupPanel.Password.Title": "",
|
||||
"GroupPanel.Password.Description": "",
|
||||
"GroupPanel.Password.Warning": "",
|
||||
"GroupPanel.Password.Hint": "",
|
||||
"GroupPanel.Password.Button": "",
|
||||
"GroupPanel.Password.Error.TooShort": "",
|
||||
"GroupPanel.Invites.Title": "",
|
||||
"GroupPanel.Invites.Description": "",
|
||||
"GroupPanel.Invites.CreateButton": "",
|
||||
"GroupPanel.Invites.Result": "",
|
||||
"GroupPanel.Invites.Copy": "",
|
||||
"GroupPanel.List.Visible": "",
|
||||
"GroupPanel.List.Online": "",
|
||||
"GroupPanel.List.Offline": "",
|
||||
"GroupPanel.List.OfflineOmitted": "",
|
||||
"GroupPanel.Permissions.Header": "",
|
||||
"GroupPanel.Permissions.InvitesDisabled": "",
|
||||
"GroupPanel.Permissions.SoundDisabledOwner": "",
|
||||
"GroupPanel.Permissions.AnimationDisabledOwner": "",
|
||||
"GroupPanel.Permissions.VfxDisabledOwner": "",
|
||||
"GroupPanel.Permissions.OwnHeader": "",
|
||||
"GroupPanel.Permissions.SoundDisabledSelf": "",
|
||||
"GroupPanel.Permissions.AnimationDisabledSelf": "",
|
||||
"GroupPanel.Permissions.VfxDisabledSelf": "",
|
||||
"GroupPanel.Permissions.NotePriority": "",
|
||||
"GroupPanel.PauseToggle.Tooltip": "",
|
||||
"GroupPanel.PauseToggle.Resume": "",
|
||||
"GroupPanel.PauseToggle.Pause": "",
|
||||
"GroupPanel.Popup.Leave": "",
|
||||
"GroupPanel.Popup.LeaveTooltip": "",
|
||||
"GroupPanel.Popup.LeaveWarning": "",
|
||||
"GroupPanel.Popup.CopyId": "",
|
||||
"GroupPanel.Popup.CopyIdTooltip": "",
|
||||
"GroupPanel.Popup.CopyNotes": "",
|
||||
"GroupPanel.Popup.CopyNotesTooltip": "",
|
||||
"GroupPanel.Popup.EnableSound": "",
|
||||
"GroupPanel.Popup.DisableSound": "",
|
||||
"GroupPanel.Popup.SoundTooltip": "",
|
||||
"GroupPanel.Popup.EnableAnimations": "",
|
||||
"GroupPanel.Popup.DisableAnimations": "",
|
||||
"GroupPanel.Popup.AnimTooltip": "",
|
||||
"GroupPanel.Popup.EnableVfx": "",
|
||||
"GroupPanel.Popup.DisableVfx": "",
|
||||
"GroupPanel.Popup.VfxTooltip": "",
|
||||
"GroupPanel.Syncshell.OwnerTooltip": "",
|
||||
"GroupPanel.Syncshell.ModeratorTooltip": "",
|
||||
"GroupPanel.Syncshell.MemberCount": "",
|
||||
"GroupPanel.Syncshell.MemberCountTooltip": "",
|
||||
"GroupPanel.Syncshell.NameTooltip": "",
|
||||
"GroupPanel.Syncshell.TempTag": "",
|
||||
"GroupPanel.Syncshell.TempExpires": "",
|
||||
"GroupPanel.Syncshell.TempTooltip": "",
|
||||
"GroupPanel.Create.Duration.SingleDay": "",
|
||||
"GroupPanel.Create.Duration.Days": "",
|
||||
"GroupPanel.Create.Duration.Hours": "",
|
||||
"GroupPanel.Invites.AmountLabel": "",
|
||||
"GroupPanel.Popup.OpenAdmin": ""
|
||||
}
|
||||
@@ -1,15 +1,72 @@
|
||||
using MareSynchronos.WebAPI;
|
||||
using System.Text.Json;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace MareSynchronos.MareConfiguration;
|
||||
|
||||
public class ConfigurationMigrator(ILogger<ConfigurationMigrator> logger) : IHostedService
|
||||
public class ConfigurationMigrator(ILogger<ConfigurationMigrator> logger, MareConfigService mareConfig) : IHostedService
|
||||
{
|
||||
private readonly ILogger<ConfigurationMigrator> _logger = logger;
|
||||
private readonly MareConfigService _mareConfig = mareConfig;
|
||||
|
||||
public void Migrate()
|
||||
{
|
||||
try
|
||||
{
|
||||
var path = _mareConfig.ConfigurationPath;
|
||||
if (!File.Exists(path)) return;
|
||||
|
||||
using var doc = JsonDocument.Parse(File.ReadAllText(path));
|
||||
var root = doc.RootElement;
|
||||
|
||||
bool changed = false;
|
||||
|
||||
if (root.TryGetProperty("EnableAutoSyncDiscovery", out var enableAutoSync))
|
||||
{
|
||||
var val = enableAutoSync.GetBoolean();
|
||||
if (_mareConfig.Current.EnableAutoDetectDiscovery != val)
|
||||
{
|
||||
_mareConfig.Current.EnableAutoDetectDiscovery = val;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
if (root.TryGetProperty("AllowAutoSyncPairRequests", out var allowAutoSync))
|
||||
{
|
||||
var val = allowAutoSync.GetBoolean();
|
||||
if (_mareConfig.Current.AllowAutoDetectPairRequests != val)
|
||||
{
|
||||
_mareConfig.Current.AllowAutoDetectPairRequests = val;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
if (root.TryGetProperty("AutoSyncMaxDistanceMeters", out var maxDistSync) && maxDistSync.TryGetInt32(out var md))
|
||||
{
|
||||
if (_mareConfig.Current.AutoDetectMaxDistanceMeters != md)
|
||||
{
|
||||
_mareConfig.Current.AutoDetectMaxDistanceMeters = md;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
if (root.TryGetProperty("AutoSyncMuteMinutes", out var muteSync) && muteSync.TryGetInt32(out var mm))
|
||||
{
|
||||
if (_mareConfig.Current.AutoDetectMuteMinutes != mm)
|
||||
{
|
||||
_mareConfig.Current.AutoDetectMuteMinutes = mm;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
|
||||
if (changed)
|
||||
{
|
||||
_logger.LogInformation("Migrated config: AutoSync -> AutoDetect fields");
|
||||
_mareConfig.Save();
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Configuration migration failed");
|
||||
}
|
||||
}
|
||||
|
||||
public Task StartAsync(CancellationToken cancellationToken)
|
||||
|
||||
@@ -59,6 +59,10 @@ public class MareConfig : IMareConfiguration
|
||||
public bool ShowUploading { get; set; } = true;
|
||||
public bool ShowUploadingBigText { get; set; } = true;
|
||||
public bool ShowVisibleUsersSeparately { get; set; } = true;
|
||||
public bool EnableAutoDetectDiscovery { get; set; } = false;
|
||||
public bool AllowAutoDetectPairRequests { get; set; } = false;
|
||||
public int AutoDetectMaxDistanceMeters { get; set; } = 40;
|
||||
public int AutoDetectMuteMinutes { get; set; } = 5;
|
||||
public int TimeSpanBetweenScansInSeconds { get; set; } = 30;
|
||||
public int TransferBarsHeight { get; set; } = 12;
|
||||
public bool TransferBarsShowText { get; set; } = true;
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
<PropertyGroup>
|
||||
<AssemblyName>UmbraSync</AssemblyName>
|
||||
<RootNamespace>UmbraSync</RootNamespace>
|
||||
<Version>0.0.6</Version>
|
||||
<Version>0.1.8.0</Version>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
@@ -53,11 +53,21 @@
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\UmbraAPI\MareSynchronosAPI\MareSynchronos.API.csproj" />
|
||||
<ProjectReference Include="..\MareAPI\MareSynchronosAPI\MareSynchronos.API.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Include="..\.editorconfig" Link=".editorconfig" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Include="Localization\\*.json" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Content Include="Localization\\*.json">
|
||||
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
||||
</Content>
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
||||
@@ -1,33 +1,60 @@
|
||||
using MareSynchronos.API.Data;
|
||||
using MareSynchronos.API.Dto.Group;
|
||||
using MareSynchronos.FileCache;
|
||||
using MareSynchronos.Interop.Ipc;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.PlayerData.Handlers;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace MareSynchronos.PlayerData.Factories;
|
||||
|
||||
public class PairFactory
|
||||
public class PairHandlerFactory
|
||||
{
|
||||
private readonly PairHandlerFactory _cachedPlayerFactory;
|
||||
private readonly MareConfigService _configService;
|
||||
private readonly DalamudUtilService _dalamudUtilService;
|
||||
private readonly FileCacheManager _fileCacheManager;
|
||||
private readonly FileDownloadManagerFactory _fileDownloadManagerFactory;
|
||||
private readonly GameObjectHandlerFactory _gameObjectHandlerFactory;
|
||||
private readonly IHostApplicationLifetime _hostApplicationLifetime;
|
||||
private readonly IpcManager _ipcManager;
|
||||
private readonly ILoggerFactory _loggerFactory;
|
||||
private readonly MareMediator _mareMediator;
|
||||
private readonly MareConfigService _mareConfig;
|
||||
private readonly ServerConfigurationManager _serverConfigurationManager;
|
||||
private readonly PlayerPerformanceService _playerPerformanceService;
|
||||
private readonly ServerConfigurationManager _serverConfigManager;
|
||||
private readonly PluginWarningNotificationService _pluginWarningNotificationManager;
|
||||
private readonly PairAnalyzerFactory _pairAnalyzerFactory;
|
||||
private readonly VisibilityService _visibilityService;
|
||||
|
||||
public PairFactory(ILoggerFactory loggerFactory, PairHandlerFactory cachedPlayerFactory,
|
||||
MareMediator mareMediator, MareConfigService mareConfig, ServerConfigurationManager serverConfigurationManager)
|
||||
public PairHandlerFactory(ILoggerFactory loggerFactory, GameObjectHandlerFactory gameObjectHandlerFactory, IpcManager ipcManager,
|
||||
FileDownloadManagerFactory fileDownloadManagerFactory, DalamudUtilService dalamudUtilService,
|
||||
PluginWarningNotificationService pluginWarningNotificationManager, IHostApplicationLifetime hostApplicationLifetime,
|
||||
FileCacheManager fileCacheManager, MareMediator mareMediator, PlayerPerformanceService playerPerformanceService,
|
||||
ServerConfigurationManager serverConfigManager, PairAnalyzerFactory pairAnalyzerFactory,
|
||||
MareConfigService configService, VisibilityService visibilityService)
|
||||
{
|
||||
_loggerFactory = loggerFactory;
|
||||
_cachedPlayerFactory = cachedPlayerFactory;
|
||||
_gameObjectHandlerFactory = gameObjectHandlerFactory;
|
||||
_ipcManager = ipcManager;
|
||||
_fileDownloadManagerFactory = fileDownloadManagerFactory;
|
||||
_dalamudUtilService = dalamudUtilService;
|
||||
_pluginWarningNotificationManager = pluginWarningNotificationManager;
|
||||
_hostApplicationLifetime = hostApplicationLifetime;
|
||||
_fileCacheManager = fileCacheManager;
|
||||
_mareMediator = mareMediator;
|
||||
_mareConfig = mareConfig;
|
||||
_serverConfigurationManager = serverConfigurationManager;
|
||||
_playerPerformanceService = playerPerformanceService;
|
||||
_serverConfigManager = serverConfigManager;
|
||||
_pairAnalyzerFactory = pairAnalyzerFactory;
|
||||
_configService = configService;
|
||||
_visibilityService = visibilityService;
|
||||
}
|
||||
|
||||
public Pair Create(UserData userData)
|
||||
public PairHandler Create(Pair pair)
|
||||
{
|
||||
return new Pair(_loggerFactory.CreateLogger<Pair>(), userData, _cachedPlayerFactory, _mareMediator, _mareConfig, _serverConfigurationManager);
|
||||
return new PairHandler(_loggerFactory.CreateLogger<PairHandler>(), pair, _pairAnalyzerFactory.Create(pair), _gameObjectHandlerFactory,
|
||||
_ipcManager, _fileDownloadManagerFactory.Create(), _pluginWarningNotificationManager, _dalamudUtilService, _hostApplicationLifetime,
|
||||
_fileCacheManager, _mareMediator, _playerPerformanceService, _serverConfigManager, _configService, _visibilityService);
|
||||
}
|
||||
}
|
||||
@@ -6,6 +6,7 @@ using Dalamud.Plugin.Services;
|
||||
using MareSynchronos.FileCache;
|
||||
using MareSynchronos.Interop;
|
||||
using MareSynchronos.Interop.Ipc;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.MareConfiguration.Configurations;
|
||||
using MareSynchronos.PlayerData.Factories;
|
||||
@@ -29,7 +30,7 @@ using MareSynchronos.Services.CharaData;
|
||||
|
||||
using MareSynchronos;
|
||||
|
||||
namespace Snowcloak;
|
||||
namespace Umbra;
|
||||
|
||||
public sealed class Plugin : IDalamudPlugin
|
||||
{
|
||||
@@ -39,8 +40,6 @@ public sealed class Plugin : IDalamudPlugin
|
||||
public static Plugin Self;
|
||||
#pragma warning restore CA2211, CS8618, MA0069, S1104, S2223
|
||||
public Action<IFramework>? RealOnFrameworkUpdate { get; set; }
|
||||
|
||||
// Proxy function in the SnowcloakSync namespace to avoid confusion in /xlstats
|
||||
public void OnFrameworkUpdate(IFramework framework)
|
||||
{
|
||||
RealOnFrameworkUpdate?.Invoke(framework);
|
||||
@@ -93,11 +92,17 @@ public sealed class Plugin : IDalamudPlugin
|
||||
collection.AddSingleton<MareMediator>();
|
||||
collection.AddSingleton<FileCacheManager>();
|
||||
collection.AddSingleton<ServerConfigurationManager>();
|
||||
collection.AddSingleton<LocalizationService>();
|
||||
collection.AddSingleton<ApiController>();
|
||||
collection.AddSingleton<PerformanceCollectorService>();
|
||||
collection.AddSingleton<HubFactory>();
|
||||
collection.AddSingleton<FileUploadManager>();
|
||||
collection.AddSingleton<FileTransferOrchestrator>();
|
||||
collection.AddSingleton<MareSynchronos.Services.AutoDetect.DiscoveryConfigProvider>();
|
||||
collection.AddSingleton<MareSynchronos.WebAPI.AutoDetect.DiscoveryApiClient>();
|
||||
collection.AddSingleton<MareSynchronos.Services.AutoDetect.AutoDetectRequestService>();
|
||||
collection.AddSingleton<MareSynchronos.Services.AutoDetect.NearbyDiscoveryService>();
|
||||
collection.AddSingleton<MareSynchronos.Services.AutoDetect.NearbyPendingService>();
|
||||
collection.AddSingleton<MarePlugin>();
|
||||
collection.AddSingleton<MareProfileManager>();
|
||||
collection.AddSingleton<GameObjectHandlerFactory>();
|
||||
@@ -142,6 +147,7 @@ public sealed class Plugin : IDalamudPlugin
|
||||
collection.AddSingleton<IpcCallerMare>();
|
||||
collection.AddSingleton<IpcManager>();
|
||||
collection.AddSingleton<NotificationService>();
|
||||
collection.AddSingleton<TemporarySyncshellNotificationService>();
|
||||
|
||||
collection.AddSingleton((s) => new MareConfigService(pluginInterface.ConfigDirectory.FullName));
|
||||
collection.AddSingleton((s) => new ServerConfigService(pluginInterface.ConfigDirectory.FullName));
|
||||
@@ -177,6 +183,7 @@ public sealed class Plugin : IDalamudPlugin
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, CompactUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, IntroUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, DownloadUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, AutoDetectUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, PopoutProfileUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, DataAnalysisUi>();
|
||||
collection.AddScoped<WindowMediatorSubscriberBase, EventViewerUI>();
|
||||
@@ -199,6 +206,7 @@ public sealed class Plugin : IDalamudPlugin
|
||||
collection.AddHostedService(p => p.GetRequiredService<ConfigurationSaveService>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<MareMediator>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<NotificationService>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<TemporarySyncshellNotificationService>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<FileCacheManager>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<ConfigurationMigrator>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<DalamudUtilService>());
|
||||
@@ -207,6 +215,7 @@ public sealed class Plugin : IDalamudPlugin
|
||||
collection.AddHostedService(p => p.GetRequiredService<EventAggregator>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<MarePlugin>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<IpcProvider>());
|
||||
collection.AddHostedService(p => p.GetRequiredService<MareSynchronos.Services.AutoDetect.NearbyDiscoveryService>());
|
||||
})
|
||||
.Build();
|
||||
|
||||
@@ -227,4 +236,4 @@ public sealed class Plugin : IDalamudPlugin
|
||||
_host.StopAsync().GetAwaiter().GetResult();
|
||||
_host.Dispose();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,83 @@
|
||||
using Microsoft.Extensions.Logging;
|
||||
using MareSynchronos.WebAPI.AutoDetect;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.Services;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using NotificationType = MareSynchronos.MareConfiguration.Models.NotificationType;
|
||||
|
||||
namespace MareSynchronos.Services.AutoDetect;
|
||||
|
||||
public class AutoDetectRequestService
|
||||
{
|
||||
private readonly ILogger<AutoDetectRequestService> _logger;
|
||||
private readonly DiscoveryConfigProvider _configProvider;
|
||||
private readonly DiscoveryApiClient _client;
|
||||
private readonly MareConfigService _configService;
|
||||
private readonly DalamudUtilService _dalamud;
|
||||
private readonly MareMediator _mediator;
|
||||
|
||||
public AutoDetectRequestService(ILogger<AutoDetectRequestService> logger, DiscoveryConfigProvider configProvider, DiscoveryApiClient client, MareConfigService configService, MareMediator mediator, DalamudUtilService dalamudUtilService)
|
||||
{
|
||||
_logger = logger;
|
||||
_configProvider = configProvider;
|
||||
_client = client;
|
||||
_configService = configService;
|
||||
_mediator = mediator;
|
||||
_dalamud = dalamudUtilService;
|
||||
}
|
||||
|
||||
public async Task<bool> SendRequestAsync(string token, CancellationToken ct = default)
|
||||
{
|
||||
if (!_configService.Current.AllowAutoDetectPairRequests)
|
||||
{
|
||||
_logger.LogDebug("Nearby request blocked: AllowAutoDetectPairRequests is disabled");
|
||||
_mediator.Publish(new NotificationMessage("Nearby request blocked", "Enable 'Allow pair requests' in Settings to send requests.", NotificationType.Info));
|
||||
return false;
|
||||
}
|
||||
var endpoint = _configProvider.RequestEndpoint;
|
||||
if (string.IsNullOrEmpty(endpoint))
|
||||
{
|
||||
_logger.LogDebug("No request endpoint configured");
|
||||
_mediator.Publish(new NotificationMessage("Nearby request failed", "Server does not expose request endpoint.", NotificationType.Error));
|
||||
return false;
|
||||
}
|
||||
string? displayName = null;
|
||||
try
|
||||
{
|
||||
var me = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
displayName = me?.Name.TextValue;
|
||||
}
|
||||
catch { }
|
||||
|
||||
_logger.LogInformation("Nearby: sending pair request via {endpoint}", endpoint);
|
||||
var ok = await _client.SendRequestAsync(endpoint!, token, displayName, ct).ConfigureAwait(false);
|
||||
if (ok)
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby request sent", "The other user will receive a request notification.", NotificationType.Info));
|
||||
}
|
||||
else
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby request failed", "The server rejected the request. Try again soon.", NotificationType.Warning));
|
||||
}
|
||||
return ok;
|
||||
}
|
||||
|
||||
public async Task<bool> SendAcceptNotifyAsync(string targetUid, CancellationToken ct = default)
|
||||
{
|
||||
var endpoint = _configProvider.AcceptEndpoint;
|
||||
if (string.IsNullOrEmpty(endpoint))
|
||||
{
|
||||
_logger.LogDebug("No accept endpoint configured");
|
||||
return false;
|
||||
}
|
||||
string? displayName = null;
|
||||
try
|
||||
{
|
||||
var me = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
displayName = me?.Name.TextValue;
|
||||
}
|
||||
catch { }
|
||||
_logger.LogInformation("Nearby: sending accept notify via {endpoint}", endpoint);
|
||||
return await _client.SendAcceptAsync(endpoint!, targetUid, displayName, ct).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
173
MareSynchronos/Services/AutoDetect/DiscoveryConfigProvider.cs
Normal file
173
MareSynchronos/Services/AutoDetect/DiscoveryConfigProvider.cs
Normal file
@@ -0,0 +1,173 @@
|
||||
using System.Text.Json;
|
||||
using System.Text.Json.Serialization;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.WebAPI.SignalR;
|
||||
using System.Net.Http;
|
||||
using System.Net.Http.Headers;
|
||||
using System.Reflection;
|
||||
|
||||
namespace MareSynchronos.Services.AutoDetect;
|
||||
|
||||
public class DiscoveryConfigProvider
|
||||
{
|
||||
private readonly ILogger<DiscoveryConfigProvider> _logger;
|
||||
private readonly ServerConfigurationManager _serverManager;
|
||||
private readonly TokenProvider _tokenProvider;
|
||||
|
||||
private WellKnownRoot? _config;
|
||||
private DateTimeOffset _lastLoad = DateTimeOffset.MinValue;
|
||||
|
||||
public DiscoveryConfigProvider(ILogger<DiscoveryConfigProvider> logger, ServerConfigurationManager serverManager, TokenProvider tokenProvider)
|
||||
{
|
||||
_logger = logger;
|
||||
_serverManager = serverManager;
|
||||
_tokenProvider = tokenProvider;
|
||||
}
|
||||
|
||||
public bool HasConfig => _config != null;
|
||||
public bool NearbyEnabled => _config?.NearbyDiscovery?.Enabled ?? false;
|
||||
public byte[]? Salt => _config?.NearbyDiscovery?.SaltBytes;
|
||||
public string? SaltB64 => _config?.NearbyDiscovery?.SaltB64;
|
||||
public DateTimeOffset? SaltExpiresAt => _config?.NearbyDiscovery?.SaltExpiresAt;
|
||||
public int RefreshSec => _config?.NearbyDiscovery?.RefreshSec ?? 300;
|
||||
public int MinQueryIntervalMs => _config?.NearbyDiscovery?.Policies?.MinQueryIntervalMs ?? 2000;
|
||||
public int MaxQueryBatch => _config?.NearbyDiscovery?.Policies?.MaxQueryBatch ?? 100;
|
||||
public string? PublishEndpoint => _config?.NearbyDiscovery?.Endpoints?.Publish;
|
||||
public string? QueryEndpoint => _config?.NearbyDiscovery?.Endpoints?.Query;
|
||||
public string? RequestEndpoint => _config?.NearbyDiscovery?.Endpoints?.Request;
|
||||
public string? AcceptEndpoint => _config?.NearbyDiscovery?.Endpoints?.Accept;
|
||||
|
||||
public bool TryLoadFromStapled()
|
||||
{
|
||||
try
|
||||
{
|
||||
var json = _tokenProvider.GetStapledWellKnown(_serverManager.CurrentApiUrl);
|
||||
if (string.IsNullOrEmpty(json)) return false;
|
||||
|
||||
var root = JsonSerializer.Deserialize<WellKnownRoot>(json!);
|
||||
if (root == null) return false;
|
||||
|
||||
root.NearbyDiscovery?.Hydrate();
|
||||
_config = root;
|
||||
_lastLoad = DateTimeOffset.UtcNow;
|
||||
_logger.LogDebug("Loaded Nearby well-known (stapled), enabled={enabled}, expires={exp}", NearbyEnabled, _config?.NearbyDiscovery?.SaltExpiresAt);
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Failed to parse stapled well-known");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task<bool> TryFetchFromServerAsync(CancellationToken ct = default)
|
||||
{
|
||||
try
|
||||
{
|
||||
var baseUrl = _serverManager.CurrentApiUrl
|
||||
.Replace("wss://", "https://", StringComparison.OrdinalIgnoreCase)
|
||||
.Replace("ws://", "http://", StringComparison.OrdinalIgnoreCase);
|
||||
// Try likely candidates based on nginx config
|
||||
string[] candidates =
|
||||
[
|
||||
"/.well-known/Umbra/client", // matches provided nginx
|
||||
"/.well-known/umbra", // lowercase variant
|
||||
];
|
||||
|
||||
using var http = new HttpClient();
|
||||
try
|
||||
{
|
||||
var ver = Assembly.GetExecutingAssembly().GetName().Version!;
|
||||
http.DefaultRequestHeaders.UserAgent.Add(new ProductInfoHeaderValue("MareSynchronos", $"{ver.Major}.{ver.Minor}.{ver.Build}"));
|
||||
}
|
||||
catch { }
|
||||
|
||||
foreach (var path in candidates)
|
||||
{
|
||||
try
|
||||
{
|
||||
var uri = new Uri(new Uri(baseUrl), path);
|
||||
var json = await http.GetStringAsync(uri, ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(json)) continue;
|
||||
|
||||
var root = JsonSerializer.Deserialize<WellKnownRoot>(json);
|
||||
if (root == null) continue;
|
||||
|
||||
root.NearbyDiscovery?.Hydrate();
|
||||
_config = root;
|
||||
_lastLoad = DateTimeOffset.UtcNow;
|
||||
_logger.LogInformation("Loaded Nearby well-known (http {path}), enabled={enabled}", path, NearbyEnabled);
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "Nearby well-known fetch failed for {path}", path);
|
||||
}
|
||||
}
|
||||
|
||||
_logger.LogInformation("Nearby well-known not found via HTTP candidates");
|
||||
return false;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Failed to fetch Nearby well-known via HTTP");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public bool IsExpired()
|
||||
{
|
||||
if (_config?.NearbyDiscovery?.SaltExpiresAt == null) return false;
|
||||
return DateTimeOffset.UtcNow > _config.NearbyDiscovery.SaltExpiresAt;
|
||||
}
|
||||
|
||||
// DTOs for well-known JSON
|
||||
private sealed class WellKnownRoot
|
||||
{
|
||||
[JsonPropertyName("features")] public Features? Features { get; set; }
|
||||
[JsonPropertyName("nearby_discovery")] public Nearby? NearbyDiscovery { get; set; }
|
||||
}
|
||||
|
||||
private sealed class Features
|
||||
{
|
||||
[JsonPropertyName("nearby_discovery")] public bool NearbyDiscovery { get; set; }
|
||||
}
|
||||
|
||||
private sealed class Nearby
|
||||
{
|
||||
[JsonPropertyName("enabled")] public bool Enabled { get; set; }
|
||||
[JsonPropertyName("hash_algo")] public string? HashAlgo { get; set; }
|
||||
[JsonPropertyName("salt_b64")] public string? SaltB64 { get; set; }
|
||||
[JsonPropertyName("salt_expires_at")] public string? SaltExpiresAtRaw { get; set; }
|
||||
[JsonPropertyName("refresh_sec")] public int RefreshSec { get; set; } = 300;
|
||||
[JsonPropertyName("endpoints")] public Endpoints? Endpoints { get; set; }
|
||||
[JsonPropertyName("policies")] public Policies? Policies { get; set; }
|
||||
|
||||
[JsonIgnore] public byte[]? SaltBytes { get; private set; }
|
||||
[JsonIgnore] public DateTimeOffset? SaltExpiresAt { get; private set; }
|
||||
|
||||
public void Hydrate()
|
||||
{
|
||||
try { SaltBytes = string.IsNullOrEmpty(SaltB64) ? null : Convert.FromBase64String(SaltB64!); } catch { SaltBytes = null; }
|
||||
if (DateTimeOffset.TryParse(SaltExpiresAtRaw, out var dto)) SaltExpiresAt = dto;
|
||||
}
|
||||
}
|
||||
|
||||
private sealed class Endpoints
|
||||
{
|
||||
[JsonPropertyName("publish")] public string? Publish { get; set; }
|
||||
[JsonPropertyName("query")] public string? Query { get; set; }
|
||||
[JsonPropertyName("request")] public string? Request { get; set; }
|
||||
[JsonPropertyName("accept")] public string? Accept { get; set; }
|
||||
}
|
||||
|
||||
private sealed class Policies
|
||||
{
|
||||
[JsonPropertyName("max_query_batch")] public int MaxQueryBatch { get; set; } = 100;
|
||||
[JsonPropertyName("min_query_interval_ms")] public int MinQueryIntervalMs { get; set; } = 2000;
|
||||
[JsonPropertyName("rate_limit_per_min")] public int RateLimitPerMin { get; set; } = 30;
|
||||
[JsonPropertyName("token_ttl_sec")] public int TokenTtlSec { get; set; } = 120;
|
||||
}
|
||||
}
|
||||
477
MareSynchronos/Services/AutoDetect/NearbyDiscoveryService.cs
Normal file
477
MareSynchronos/Services/AutoDetect/NearbyDiscoveryService.cs
Normal file
@@ -0,0 +1,477 @@
|
||||
using Microsoft.Extensions.Logging;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.WebAPI.AutoDetect;
|
||||
using Dalamud.Plugin.Services;
|
||||
using System.Numerics;
|
||||
using System.Linq;
|
||||
using MareSynchronos.Utils;
|
||||
|
||||
namespace MareSynchronos.Services.AutoDetect;
|
||||
|
||||
public class NearbyDiscoveryService : IHostedService, IMediatorSubscriber
|
||||
{
|
||||
private readonly ILogger<NearbyDiscoveryService> _logger;
|
||||
private readonly MareMediator _mediator;
|
||||
private readonly MareConfigService _config;
|
||||
private readonly DiscoveryConfigProvider _configProvider;
|
||||
private readonly DalamudUtilService _dalamud;
|
||||
private readonly IObjectTable _objectTable;
|
||||
private readonly DiscoveryApiClient _api;
|
||||
private CancellationTokenSource? _loopCts;
|
||||
private string? _lastPublishedSignature;
|
||||
private bool _loggedLocalOnly;
|
||||
private int _lastLocalCount = -1;
|
||||
private int _lastMatchCount = -1;
|
||||
private bool _loggedConfigReady;
|
||||
private string? _lastSnapshotSig;
|
||||
private volatile bool _isConnected;
|
||||
private bool _notifiedDisabled;
|
||||
private bool _notifiedEnabled;
|
||||
private bool _disableSent;
|
||||
private bool _lastAutoDetectState;
|
||||
private DateTime _lastHeartbeat = DateTime.MinValue;
|
||||
private static readonly TimeSpan HeartbeatInterval = TimeSpan.FromSeconds(75);
|
||||
|
||||
public NearbyDiscoveryService(ILogger<NearbyDiscoveryService> logger, MareMediator mediator,
|
||||
MareConfigService config, DiscoveryConfigProvider configProvider, DalamudUtilService dalamudUtilService,
|
||||
IObjectTable objectTable, DiscoveryApiClient api)
|
||||
{
|
||||
_logger = logger;
|
||||
_mediator = mediator;
|
||||
_config = config;
|
||||
_configProvider = configProvider;
|
||||
_dalamud = dalamudUtilService;
|
||||
_objectTable = objectTable;
|
||||
_api = api;
|
||||
}
|
||||
|
||||
public MareMediator Mediator => _mediator;
|
||||
|
||||
public Task StartAsync(CancellationToken cancellationToken)
|
||||
{
|
||||
_loopCts = new CancellationTokenSource();
|
||||
_mediator.Subscribe<ConnectedMessage>(this, _ => { _isConnected = true; _configProvider.TryLoadFromStapled(); });
|
||||
_mediator.Subscribe<DisconnectedMessage>(this, _ => { _isConnected = false; _lastPublishedSignature = null; });
|
||||
_mediator.Subscribe<AllowPairRequestsToggled>(this, OnAllowPairRequestsToggled);
|
||||
_ = Task.Run(() => Loop(_loopCts.Token));
|
||||
_lastAutoDetectState = _config.Current.EnableAutoDetectDiscovery;
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
private async void OnAllowPairRequestsToggled(AllowPairRequestsToggled msg)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (!_config.Current.EnableAutoDetectDiscovery) return;
|
||||
// Force a publish now so the server immediately reflects the new allow/deny state
|
||||
_lastPublishedSignature = null; // ensure next loop won't skip
|
||||
await PublishSelfOnceAsync(CancellationToken.None).ConfigureAwait(false);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "OnAllowPairRequestsToggled failed");
|
||||
}
|
||||
}
|
||||
|
||||
private async Task PublishSelfOnceAsync(CancellationToken ct)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (!_config.Current.EnableAutoDetectDiscovery || !_dalamud.IsLoggedIn || !_isConnected) return;
|
||||
|
||||
if (!_configProvider.HasConfig || _configProvider.IsExpired())
|
||||
{
|
||||
if (!_configProvider.TryLoadFromStapled())
|
||||
{
|
||||
await _configProvider.TryFetchFromServerAsync(ct).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
|
||||
var ep = _configProvider.PublishEndpoint;
|
||||
var saltBytes = _configProvider.Salt;
|
||||
if (string.IsNullOrEmpty(ep) || saltBytes is not { Length: > 0 }) return;
|
||||
|
||||
var saltHex = Convert.ToHexString(saltBytes);
|
||||
string? displayName = null;
|
||||
ushort meWorld = 0;
|
||||
try
|
||||
{
|
||||
var me = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
if (me != null)
|
||||
{
|
||||
displayName = me.Name.TextValue;
|
||||
if (me is Dalamud.Game.ClientState.Objects.SubKinds.IPlayerCharacter mePc)
|
||||
meWorld = (ushort)mePc.HomeWorld.RowId;
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
if (string.IsNullOrEmpty(displayName)) return;
|
||||
|
||||
var selfHash = (saltHex + displayName + meWorld.ToString()).GetHash256();
|
||||
var ok = await _api.PublishAsync(ep!, new[] { selfHash }, displayName, ct, _config.Current.AllowAutoDetectPairRequests).ConfigureAwait(false);
|
||||
_logger.LogInformation("Nearby publish (manual/immediate): {result}", ok ? "success" : "failed");
|
||||
if (ok)
|
||||
{
|
||||
_lastPublishedSignature = selfHash;
|
||||
_lastHeartbeat = DateTime.UtcNow;
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "Immediate publish failed");
|
||||
}
|
||||
}
|
||||
|
||||
public Task StopAsync(CancellationToken cancellationToken)
|
||||
{
|
||||
_mediator.UnsubscribeAll(this);
|
||||
try { _loopCts?.Cancel(); } catch { }
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
private async Task Loop(CancellationToken ct)
|
||||
{
|
||||
_configProvider.TryLoadFromStapled();
|
||||
|
||||
while (!ct.IsCancellationRequested)
|
||||
{
|
||||
try
|
||||
{
|
||||
bool currentState = _config.Current.EnableAutoDetectDiscovery;
|
||||
if (currentState != _lastAutoDetectState)
|
||||
{
|
||||
_lastAutoDetectState = currentState;
|
||||
if (currentState)
|
||||
{
|
||||
// Force immediate publish on toggle ON
|
||||
try
|
||||
{
|
||||
// Ensure well-known is present
|
||||
if (!_configProvider.HasConfig || _configProvider.IsExpired())
|
||||
{
|
||||
if (!_configProvider.TryLoadFromStapled())
|
||||
{
|
||||
await _configProvider.TryFetchFromServerAsync(ct).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
|
||||
var ep = _configProvider.PublishEndpoint;
|
||||
var saltBytes = _configProvider.Salt;
|
||||
if (!string.IsNullOrEmpty(ep) && saltBytes is { Length: > 0 })
|
||||
{
|
||||
var saltHex = Convert.ToHexString(saltBytes);
|
||||
string? displayName = null;
|
||||
ushort meWorld = 0;
|
||||
try
|
||||
{
|
||||
var me = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
if (me != null)
|
||||
{
|
||||
displayName = me.Name.TextValue;
|
||||
if (me is Dalamud.Game.ClientState.Objects.SubKinds.IPlayerCharacter mePc)
|
||||
meWorld = (ushort)mePc.HomeWorld.RowId;
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
if (!string.IsNullOrEmpty(displayName))
|
||||
{
|
||||
var selfHash = (saltHex + displayName + meWorld.ToString()).GetHash256();
|
||||
_lastPublishedSignature = null; // ensure future loop doesn't skip
|
||||
var okNow = await _api.PublishAsync(ep!, new[] { selfHash }, displayName, ct, _config.Current.AllowAutoDetectPairRequests).ConfigureAwait(false);
|
||||
_logger.LogInformation("Nearby immediate publish on toggle ON: {result}", okNow ? "success" : "failed");
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "Nearby immediate publish on toggle ON failed");
|
||||
}
|
||||
|
||||
if (!_notifiedEnabled)
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby Detection", "AutoDetect enabled : you are now visible.", default));
|
||||
_notifiedEnabled = true;
|
||||
_notifiedDisabled = false;
|
||||
_disableSent = false;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var ep = _configProvider.PublishEndpoint;
|
||||
if (!string.IsNullOrEmpty(ep) && !_disableSent)
|
||||
{
|
||||
var disableUrl = ep.Replace("/publish", "/disable");
|
||||
try { await _api.DisableAsync(disableUrl, ct).ConfigureAwait(false); _disableSent = true; } catch { }
|
||||
}
|
||||
if (!_notifiedDisabled)
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby Detection", "AutoDetect disabled : you are not visible.", default));
|
||||
_notifiedDisabled = true;
|
||||
_notifiedEnabled = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!_config.Current.EnableAutoDetectDiscovery || !_dalamud.IsLoggedIn || !_isConnected)
|
||||
{
|
||||
if (!_config.Current.EnableAutoDetectDiscovery && !string.IsNullOrEmpty(_configProvider.PublishEndpoint))
|
||||
{
|
||||
var disableUrl = _configProvider.PublishEndpoint.Replace("/publish", "/disable");
|
||||
try
|
||||
{
|
||||
if (!_disableSent)
|
||||
{
|
||||
await _api.DisableAsync(disableUrl, ct).ConfigureAwait(false);
|
||||
_disableSent = true;
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
if (!_notifiedDisabled)
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby Detection", "AutoDetect disabled : you are not visible.", default));
|
||||
_notifiedDisabled = true;
|
||||
_notifiedEnabled = false;
|
||||
}
|
||||
}
|
||||
await Task.Delay(1000, ct).ConfigureAwait(false);
|
||||
continue;
|
||||
}
|
||||
if (!_configProvider.HasConfig || _configProvider.IsExpired())
|
||||
{
|
||||
if (!_configProvider.TryLoadFromStapled())
|
||||
{
|
||||
await _configProvider.TryFetchFromServerAsync(ct).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
else if (!_loggedConfigReady && _configProvider.NearbyEnabled)
|
||||
{
|
||||
_loggedConfigReady = true;
|
||||
_logger.LogInformation("Nearby: well-known loaded and enabled; refresh={refresh}s, expires={exp}", _configProvider.RefreshSec, _configProvider.SaltExpiresAt);
|
||||
}
|
||||
|
||||
var entries = await GetLocalNearbyAsync().ConfigureAwait(false);
|
||||
// Log when local count changes (including 0) to indicate activity
|
||||
if (entries.Count != _lastLocalCount)
|
||||
{
|
||||
_lastLocalCount = entries.Count;
|
||||
_logger.LogTrace("Nearby: {count} players detected locally", _lastLocalCount);
|
||||
}
|
||||
|
||||
// Try query server if config and endpoints are present
|
||||
if (_configProvider.NearbyEnabled && !_configProvider.IsExpired() &&
|
||||
_configProvider.Salt is { Length: > 0 })
|
||||
{
|
||||
try
|
||||
{
|
||||
var saltHex = Convert.ToHexString(_configProvider.Salt!);
|
||||
// map hash->index for result matching
|
||||
Dictionary<string, int> hashToIndex = new(StringComparer.Ordinal);
|
||||
List<string> hashes = new(entries.Count);
|
||||
foreach (var (entry, idx) in entries.Select((e, i) => (e, i)))
|
||||
{
|
||||
var h = (saltHex + entry.Name + entry.WorldId.ToString()).GetHash256();
|
||||
hashToIndex[h] = idx;
|
||||
hashes.Add(h);
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
var snapSig = string.Join(',', hashes.OrderBy(s => s, StringComparer.Ordinal)).GetHash256();
|
||||
if (!string.Equals(snapSig, _lastSnapshotSig, StringComparison.Ordinal))
|
||||
{
|
||||
_lastSnapshotSig = snapSig;
|
||||
var sample = entries.Take(5).Select(e =>
|
||||
{
|
||||
var hh = (saltHex + e.Name + e.WorldId.ToString()).GetHash256();
|
||||
var shortH = hh.Length > 8 ? hh[..8] : hh;
|
||||
return $"{e.Name}({e.WorldId})->{shortH}";
|
||||
});
|
||||
var saltShort = saltHex.Length > 8 ? saltHex[..8] : saltHex;
|
||||
_logger.LogTrace("Nearby snapshot: {count} entries; salt={saltShort}…; samples=[{samples}]",
|
||||
entries.Count, saltShort, string.Join(", ", sample));
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
if (!string.IsNullOrEmpty(_configProvider.PublishEndpoint))
|
||||
{
|
||||
string? displayName = null;
|
||||
string? selfHash = null;
|
||||
try
|
||||
{
|
||||
var me = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
if (me != null)
|
||||
{
|
||||
displayName = me.Name.TextValue;
|
||||
ushort meWorld = 0;
|
||||
if (me is Dalamud.Game.ClientState.Objects.SubKinds.IPlayerCharacter mePc)
|
||||
meWorld = (ushort)mePc.HomeWorld.RowId;
|
||||
_logger.LogTrace("Nearby self ident: {name} ({world})", displayName, meWorld);
|
||||
selfHash = (saltHex + displayName + meWorld.ToString()).GetHash256();
|
||||
}
|
||||
}
|
||||
catch { /* ignore */ }
|
||||
|
||||
if (!string.IsNullOrEmpty(selfHash))
|
||||
{
|
||||
var sig = selfHash!;
|
||||
if (!string.Equals(sig, _lastPublishedSignature, StringComparison.Ordinal))
|
||||
{
|
||||
_lastPublishedSignature = sig;
|
||||
var shortSelf = selfHash!.Length > 8 ? selfHash[..8] : selfHash;
|
||||
_logger.LogDebug("Nearby publish: self presence updated (hash={hash})", shortSelf);
|
||||
var ok = await _api.PublishAsync(_configProvider.PublishEndpoint!, new[] { selfHash! }, displayName, ct, _config.Current.AllowAutoDetectPairRequests).ConfigureAwait(false);
|
||||
_logger.LogInformation("Nearby publish result: {result}", ok ? "success" : "failed");
|
||||
if (ok) _lastHeartbeat = DateTime.UtcNow;
|
||||
if (ok)
|
||||
{
|
||||
if (!_notifiedEnabled)
|
||||
{
|
||||
_mediator.Publish(new NotificationMessage("Nearby Detection", "AutoDetect enabled : you are now visible.", default));
|
||||
_notifiedEnabled = true;
|
||||
_notifiedDisabled = false;
|
||||
_disableSent = false; // allow future /disable when turning off again
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// No changes; perform heartbeat publish if interval elapsed
|
||||
if (DateTime.UtcNow - _lastHeartbeat >= HeartbeatInterval)
|
||||
{
|
||||
var okHb = await _api.PublishAsync(_configProvider.PublishEndpoint!, new[] { selfHash! }, displayName, ct, _config.Current.AllowAutoDetectPairRequests).ConfigureAwait(false);
|
||||
_logger.LogDebug("Nearby heartbeat publish: {result}", okHb ? "success" : "failed");
|
||||
if (okHb) _lastHeartbeat = DateTime.UtcNow;
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogDebug("Nearby publish skipped (no changes)");
|
||||
}
|
||||
}
|
||||
}
|
||||
// else: no self character available; skip publish silently
|
||||
}
|
||||
|
||||
// Query for matches if endpoint is available
|
||||
if (!string.IsNullOrEmpty(_configProvider.QueryEndpoint))
|
||||
{
|
||||
// chunked queries
|
||||
int batch = Math.Max(1, _configProvider.MaxQueryBatch);
|
||||
List<ServerMatch> allMatches = new();
|
||||
for (int i = 0; i < hashes.Count; i += batch)
|
||||
{
|
||||
var slice = hashes.Skip(i).Take(batch).ToArray();
|
||||
var res = await _api.QueryAsync(_configProvider.QueryEndpoint!, slice, ct).ConfigureAwait(false);
|
||||
if (res != null && res.Count > 0) allMatches.AddRange(res);
|
||||
}
|
||||
|
||||
if (allMatches.Count > 0)
|
||||
{
|
||||
foreach (var m in allMatches)
|
||||
{
|
||||
if (hashToIndex.TryGetValue(m.Hash, out var idx))
|
||||
{
|
||||
var e = entries[idx];
|
||||
entries[idx] = new NearbyEntry(e.Name, e.WorldId, e.Distance, true, m.Token, m.DisplayName, m.Uid);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (allMatches.Count > 0)
|
||||
{
|
||||
_logger.LogInformation("Nearby: server returned {count} matches", allMatches.Count);
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogTrace("Nearby: server returned {count} matches", allMatches.Count);
|
||||
}
|
||||
|
||||
// Log change in number of Umbra matches
|
||||
int matchCount = entries.Count(e => e.IsMatch);
|
||||
if (matchCount != _lastMatchCount)
|
||||
{
|
||||
_lastMatchCount = matchCount;
|
||||
if (matchCount > 0)
|
||||
{
|
||||
var matchSamples = entries.Where(e => e.IsMatch).Take(5)
|
||||
.Select(e => string.IsNullOrEmpty(e.DisplayName) ? e.Name : e.DisplayName!);
|
||||
_logger.LogInformation("Nearby: {count} Umbra users nearby [{samples}]",
|
||||
matchCount, string.Join(", ", matchSamples));
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogTrace("Nearby: {count} Umbra users nearby", matchCount);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "Nearby query failed; falling back to local list");
|
||||
if (ex.Message.Contains("DISCOVERY_SALT_EXPIRED", StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
_logger.LogInformation("Nearby: salt expired, refetching well-known");
|
||||
try { await _configProvider.TryFetchFromServerAsync(ct).ConfigureAwait(false); } catch { }
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (!_loggedLocalOnly)
|
||||
{
|
||||
_loggedLocalOnly = true;
|
||||
_logger.LogDebug("Nearby: well-known not available or disabled; running in local-only mode");
|
||||
}
|
||||
}
|
||||
_mediator.Publish(new DiscoveryListUpdated(entries));
|
||||
|
||||
var delayMs = Math.Max(1000, _configProvider.MinQueryIntervalMs);
|
||||
if (entries.Count == 0) delayMs = Math.Max(delayMs, 5000);
|
||||
await Task.Delay(delayMs, ct).ConfigureAwait(false);
|
||||
}
|
||||
catch (OperationCanceledException) { }
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogDebug(ex, "NearbyDiscoveryService loop error");
|
||||
await Task.Delay(2000, ct).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private async Task<List<NearbyEntry>> GetLocalNearbyAsync()
|
||||
{
|
||||
var list = new List<NearbyEntry>();
|
||||
try
|
||||
{
|
||||
var local = await _dalamud.RunOnFrameworkThread(() => _dalamud.GetPlayerCharacter()).ConfigureAwait(false);
|
||||
var localPos = local?.Position ?? Vector3.Zero;
|
||||
int maxDist = Math.Clamp(_config.Current.AutoDetectMaxDistanceMeters, 5, 100);
|
||||
|
||||
int limit = Math.Min(200, _objectTable.Length);
|
||||
for (int i = 0; i < limit; i++)
|
||||
{
|
||||
var obj = await _dalamud.RunOnFrameworkThread(() => _objectTable[i]).ConfigureAwait(false);
|
||||
if (obj == null || obj.ObjectKind != Dalamud.Game.ClientState.Objects.Enums.ObjectKind.Player) continue;
|
||||
if (local != null && obj.Address == local.Address) continue;
|
||||
|
||||
float dist = local == null ? float.NaN : Vector3.Distance(localPos, obj.Position);
|
||||
if (!float.IsNaN(dist) && dist > maxDist) continue;
|
||||
|
||||
string name = obj.Name.TextValue;
|
||||
ushort worldId = 0;
|
||||
if (obj is Dalamud.Game.ClientState.Objects.SubKinds.IPlayerCharacter pc)
|
||||
worldId = (ushort)pc.HomeWorld.RowId;
|
||||
|
||||
list.Add(new NearbyEntry(name, worldId, dist, false, null, null, null));
|
||||
}
|
||||
}
|
||||
catch
|
||||
{
|
||||
// ignore
|
||||
}
|
||||
return list;
|
||||
}
|
||||
}
|
||||
87
MareSynchronos/Services/AutoDetect/NearbyPendingService.cs
Normal file
87
MareSynchronos/Services/AutoDetect/NearbyPendingService.cs
Normal file
@@ -0,0 +1,87 @@
|
||||
using System.Collections.Concurrent;
|
||||
using System.Text.RegularExpressions;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.WebAPI;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace MareSynchronos.Services.AutoDetect;
|
||||
|
||||
public sealed class NearbyPendingService : IMediatorSubscriber
|
||||
{
|
||||
private readonly ILogger<NearbyPendingService> _logger;
|
||||
private readonly MareMediator _mediator;
|
||||
private readonly ApiController _api;
|
||||
private readonly AutoDetectRequestService _requestService;
|
||||
private readonly ConcurrentDictionary<string, string> _pending = new(StringComparer.Ordinal);
|
||||
private static readonly Regex ReqRegex = new(@"^Nearby Request: (.+) \[(?<uid>[A-Z0-9]+)\]$", RegexOptions.Compiled);
|
||||
private static readonly Regex AcceptRegex = new(@"^Nearby Accept: (.+) \[(?<uid>[A-Z0-9]+)\]$", RegexOptions.Compiled);
|
||||
|
||||
public NearbyPendingService(ILogger<NearbyPendingService> logger, MareMediator mediator, ApiController api, AutoDetectRequestService requestService)
|
||||
{
|
||||
_logger = logger;
|
||||
_mediator = mediator;
|
||||
_api = api;
|
||||
_requestService = requestService;
|
||||
_mediator.Subscribe<NotificationMessage>(this, OnNotification);
|
||||
}
|
||||
|
||||
public MareMediator Mediator => _mediator;
|
||||
|
||||
public IReadOnlyDictionary<string, string> Pending => _pending;
|
||||
|
||||
private void OnNotification(NotificationMessage msg)
|
||||
{
|
||||
// Watch info messages for Nearby request pattern
|
||||
if (msg.Type != MareSynchronos.MareConfiguration.Models.NotificationType.Info) return;
|
||||
var ma = AcceptRegex.Match(msg.Message);
|
||||
if (ma.Success)
|
||||
{
|
||||
var uidA = ma.Groups["uid"].Value;
|
||||
if (!string.IsNullOrEmpty(uidA))
|
||||
{
|
||||
_ = _api.UserAddPair(new MareSynchronos.API.Dto.User.UserDto(new MareSynchronos.API.Data.UserData(uidA)));
|
||||
_pending.TryRemove(uidA, out _);
|
||||
_logger.LogInformation("NearbyPending: auto-accepted pairing with {uid}", uidA);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
var m = ReqRegex.Match(msg.Message);
|
||||
if (!m.Success) return;
|
||||
var uid = m.Groups["uid"].Value;
|
||||
if (string.IsNullOrEmpty(uid)) return;
|
||||
// Try to extract name as everything before space and '['
|
||||
var name = msg.Message;
|
||||
try
|
||||
{
|
||||
var idx = msg.Message.IndexOf(':');
|
||||
if (idx >= 0) name = msg.Message[(idx + 1)..].Trim();
|
||||
var br = name.LastIndexOf('[');
|
||||
if (br > 0) name = name[..br].Trim();
|
||||
}
|
||||
catch { name = uid; }
|
||||
_pending[uid] = name;
|
||||
_logger.LogInformation("NearbyPending: received request from {uid} ({name})", uid, name);
|
||||
}
|
||||
|
||||
public void Remove(string uid)
|
||||
{
|
||||
_pending.TryRemove(uid, out _);
|
||||
}
|
||||
|
||||
public async Task<bool> AcceptAsync(string uid)
|
||||
{
|
||||
try
|
||||
{
|
||||
await _api.UserAddPair(new MareSynchronos.API.Dto.User.UserDto(new MareSynchronos.API.Data.UserData(uid))).ConfigureAwait(false);
|
||||
_pending.TryRemove(uid, out _);
|
||||
_ = _requestService.SendAcceptNotifyAsync(uid);
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "NearbyPending: accept failed for {uid}", uid);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -72,7 +72,7 @@ public class ChatService : DisposableMediatorSubscriberBase
|
||||
{
|
||||
var chatMsg = message.ChatMsg;
|
||||
var prefix = new SeStringBuilder();
|
||||
prefix.AddText("[SnowChat] ");
|
||||
prefix.AddText("[UmbraChat] ");
|
||||
_chatGui.Print(new XivChatEntry{
|
||||
MessageBytes = [..prefix.Build().Encode(), ..message.ChatMsg.PayloadContent],
|
||||
Name = chatMsg.SenderName,
|
||||
|
||||
@@ -14,10 +14,8 @@ namespace MareSynchronos.Services;
|
||||
|
||||
public sealed class CommandManagerService : IDisposable
|
||||
{
|
||||
private const string _commandName = "/sync";
|
||||
private const string _commandName2 = "/umbra";
|
||||
|
||||
private const string _ssCommandPrefix = "/ss";
|
||||
private const string _commandName = "/usync";
|
||||
private const string _ssCommandPrefix = "/ums";
|
||||
|
||||
private readonly ApiController _apiController;
|
||||
private readonly ICommandManager _commandManager;
|
||||
@@ -42,11 +40,7 @@ public sealed class CommandManagerService : IDisposable
|
||||
_mareConfigService = mareConfigService;
|
||||
_commandManager.AddHandler(_commandName, new CommandInfo(OnCommand)
|
||||
{
|
||||
HelpMessage = "Opens the Umbra UI"
|
||||
});
|
||||
_commandManager.AddHandler(_commandName2, new CommandInfo(OnCommand)
|
||||
{
|
||||
HelpMessage = "Opens the Umbra UI"
|
||||
HelpMessage = "Opens the UmbraSync UI"
|
||||
});
|
||||
|
||||
// Lazy registration of all possible /ss# commands which tbf is what the game does for linkshells anyway
|
||||
@@ -62,7 +56,7 @@ public sealed class CommandManagerService : IDisposable
|
||||
public void Dispose()
|
||||
{
|
||||
_commandManager.RemoveHandler(_commandName);
|
||||
_commandManager.RemoveHandler(_commandName2);
|
||||
|
||||
|
||||
for (int i = 1; i <= ChatService.CommandMaxNumber; ++i)
|
||||
_commandManager.RemoveHandler($"{_ssCommandPrefix}{i}");
|
||||
@@ -147,7 +141,6 @@ public sealed class CommandManagerService : IDisposable
|
||||
}
|
||||
else
|
||||
{
|
||||
// FIXME: Chat content seems to already be stripped of any special characters here?
|
||||
byte[] chatBytes = Encoding.UTF8.GetBytes(args);
|
||||
_chatService.SendChatShell(shellNumber, chatBytes);
|
||||
}
|
||||
|
||||
@@ -12,7 +12,7 @@ using System.Numerics;
|
||||
|
||||
namespace MareSynchronos.Services.Mediator;
|
||||
|
||||
#pragma warning disable MA0048 // File name must match type name
|
||||
#pragma warning disable MA0048
|
||||
#pragma warning disable S2094
|
||||
public record SwitchToIntroUiMessage : MessageBase;
|
||||
public record SwitchToMainUiMessage : MessageBase;
|
||||
@@ -108,6 +108,11 @@ public record GPoseLobbyReceiveCharaData(CharaDataDownloadDto CharaDataDownloadD
|
||||
public record GPoseLobbyReceivePoseData(UserData UserData, PoseData PoseData) : MessageBase;
|
||||
public record GPoseLobbyReceiveWorldData(UserData UserData, WorldData WorldData) : MessageBase;
|
||||
|
||||
public record NearbyEntry(string Name, ushort WorldId, float Distance, bool IsMatch, string? Token, string? DisplayName, string? Uid, bool AcceptPairRequests = true);
|
||||
public record DiscoveryListUpdated(List<NearbyEntry> Entries) : MessageBase;
|
||||
public record NearbyDetectionToggled(bool Enabled) : MessageBase;
|
||||
public record AllowPairRequestsToggled(bool Enabled) : MessageBase;
|
||||
|
||||
public record PluginChangeMessage(string InternalName, Version Version, bool IsLoaded) : KeyedMessage(InternalName);
|
||||
#pragma warning restore S2094
|
||||
#pragma warning restore MA0048 // File name must match type name
|
||||
#pragma warning restore MA0048
|
||||
|
||||
225
MareSynchronos/Services/TemporarySyncshellNotificationService.cs
Normal file
225
MareSynchronos/Services/TemporarySyncshellNotificationService.cs
Normal file
@@ -0,0 +1,225 @@
|
||||
using System.Globalization;
|
||||
using System.Threading;
|
||||
using MareSynchronos.API.Dto.Group;
|
||||
using MareSynchronos.MareConfiguration.Models;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.WebAPI;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace MareSynchronos.Services;
|
||||
|
||||
public sealed class TemporarySyncshellNotificationService : MediatorSubscriberBase, IHostedService
|
||||
{
|
||||
private static readonly int[] NotificationThresholdMinutes = [30, 15, 5, 1];
|
||||
private readonly ApiController _apiController;
|
||||
private readonly PairManager _pairManager;
|
||||
private readonly Lock _stateLock = new();
|
||||
private readonly Dictionary<string, TrackedGroup> _trackedGroups = new(StringComparer.Ordinal);
|
||||
private CancellationTokenSource? _loopCts;
|
||||
private Task? _loopTask;
|
||||
|
||||
public TemporarySyncshellNotificationService(ILogger<TemporarySyncshellNotificationService> logger, MareMediator mediator, PairManager pairManager, ApiController apiController)
|
||||
: base(logger, mediator)
|
||||
{
|
||||
_pairManager = pairManager;
|
||||
_apiController = apiController;
|
||||
}
|
||||
|
||||
public Task StartAsync(CancellationToken cancellationToken)
|
||||
{
|
||||
_loopCts = new CancellationTokenSource();
|
||||
Mediator.Subscribe<ConnectedMessage>(this, _ => ResetTrackedGroups());
|
||||
Mediator.Subscribe<DisconnectedMessage>(this, _ => ResetTrackedGroups());
|
||||
_loopTask = Task.Run(() => MonitorLoopAsync(_loopCts.Token), _loopCts.Token);
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
public async Task StopAsync(CancellationToken cancellationToken)
|
||||
{
|
||||
Mediator.UnsubscribeAll(this);
|
||||
if (_loopCts == null)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
_loopCts.Cancel();
|
||||
if (_loopTask != null)
|
||||
{
|
||||
await _loopTask.ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
catch (OperationCanceledException)
|
||||
{
|
||||
}
|
||||
finally
|
||||
{
|
||||
_loopTask = null;
|
||||
_loopCts.Dispose();
|
||||
_loopCts = null;
|
||||
}
|
||||
}
|
||||
|
||||
private async Task MonitorLoopAsync(CancellationToken ct)
|
||||
{
|
||||
var delay = TimeSpan.FromSeconds(30);
|
||||
while (!ct.IsCancellationRequested)
|
||||
{
|
||||
try
|
||||
{
|
||||
CheckGroups();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
Logger.LogDebug(ex, "Failed to check temporary syncshell expirations");
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
await Task.Delay(delay, ct).ConfigureAwait(false);
|
||||
}
|
||||
catch (OperationCanceledException)
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void CheckGroups()
|
||||
{
|
||||
var nowUtc = DateTime.UtcNow;
|
||||
var groupsSnapshot = _pairManager.Groups.Values.ToList();
|
||||
var notifications = new List<NotificationPayload>();
|
||||
var expiredGroups = new List<GroupFullInfoDto>();
|
||||
var seenTemporaryGids = new HashSet<string>(StringComparer.Ordinal);
|
||||
|
||||
using (var guard = _stateLock.EnterScope())
|
||||
{
|
||||
foreach (var group in groupsSnapshot)
|
||||
{
|
||||
if (!group.IsTemporary || group.ExpiresAt == null)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
if (string.IsNullOrEmpty(_apiController.UID) || !string.Equals(group.OwnerUID, _apiController.UID, StringComparison.Ordinal))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
var gid = group.Group.GID;
|
||||
seenTemporaryGids.Add(gid);
|
||||
var expiresAtUtc = NormalizeToUtc(group.ExpiresAt.Value);
|
||||
var remaining = expiresAtUtc - nowUtc;
|
||||
|
||||
if (!_trackedGroups.TryGetValue(gid, out var state))
|
||||
{
|
||||
state = new TrackedGroup(expiresAtUtc);
|
||||
_trackedGroups[gid] = state;
|
||||
}
|
||||
else if (state.ExpiresAtUtc != expiresAtUtc)
|
||||
{
|
||||
state.UpdateExpiresAt(expiresAtUtc);
|
||||
}
|
||||
|
||||
if (remaining <= TimeSpan.Zero)
|
||||
{
|
||||
_trackedGroups.Remove(gid);
|
||||
expiredGroups.Add(group);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!state.LastRemaining.HasValue)
|
||||
{
|
||||
state.UpdateRemaining(remaining);
|
||||
continue;
|
||||
}
|
||||
|
||||
var previousRemaining = state.LastRemaining.Value;
|
||||
|
||||
foreach (var thresholdMinutes in NotificationThresholdMinutes)
|
||||
{
|
||||
var threshold = TimeSpan.FromMinutes(thresholdMinutes);
|
||||
if (previousRemaining > threshold && remaining <= threshold)
|
||||
{
|
||||
notifications.Add(new NotificationPayload(group, thresholdMinutes, expiresAtUtc));
|
||||
}
|
||||
}
|
||||
|
||||
state.UpdateRemaining(remaining);
|
||||
}
|
||||
|
||||
var toRemove = _trackedGroups.Keys.Where(k => !seenTemporaryGids.Contains(k)).ToList();
|
||||
foreach (var gid in toRemove)
|
||||
{
|
||||
_trackedGroups.Remove(gid);
|
||||
}
|
||||
}
|
||||
|
||||
foreach (var expiredGroup in expiredGroups)
|
||||
{
|
||||
Logger.LogInformation("Temporary syncshell {gid} expired locally; removing", expiredGroup.Group.GID);
|
||||
_pairManager.RemoveGroup(expiredGroup.Group);
|
||||
}
|
||||
|
||||
foreach (var notification in notifications)
|
||||
{
|
||||
PublishNotification(notification.Group, notification.ThresholdMinutes, notification.ExpiresAtUtc);
|
||||
}
|
||||
}
|
||||
|
||||
private void PublishNotification(GroupFullInfoDto group, int thresholdMinutes, DateTime expiresAtUtc)
|
||||
{
|
||||
string displayName = string.IsNullOrWhiteSpace(group.GroupAlias) ? group.Group.GID : group.GroupAlias!;
|
||||
string threshold = thresholdMinutes == 1 ? "1 minute" : $"{thresholdMinutes} minutes";
|
||||
string expiresLocal = expiresAtUtc.ToLocalTime().ToString("t", CultureInfo.CurrentCulture);
|
||||
|
||||
string message = $"La Syncshell temporaire \"{displayName}\" sera supprimee dans {threshold} (a {expiresLocal}).";
|
||||
Mediator.Publish(new NotificationMessage("Syncshell temporaire", message, NotificationType.Warning, TimeSpan.FromSeconds(6)));
|
||||
}
|
||||
|
||||
private static DateTime NormalizeToUtc(DateTime expiresAt)
|
||||
{
|
||||
return expiresAt.Kind switch
|
||||
{
|
||||
DateTimeKind.Utc => expiresAt,
|
||||
DateTimeKind.Local => expiresAt.ToUniversalTime(),
|
||||
_ => DateTime.SpecifyKind(expiresAt, DateTimeKind.Utc)
|
||||
};
|
||||
}
|
||||
|
||||
private void ResetTrackedGroups()
|
||||
{
|
||||
using (var guard = _stateLock.EnterScope())
|
||||
{
|
||||
_trackedGroups.Clear();
|
||||
}
|
||||
}
|
||||
|
||||
private sealed class TrackedGroup
|
||||
{
|
||||
public TrackedGroup(DateTime expiresAtUtc)
|
||||
{
|
||||
ExpiresAtUtc = expiresAtUtc;
|
||||
}
|
||||
|
||||
public DateTime ExpiresAtUtc { get; private set; }
|
||||
public TimeSpan? LastRemaining { get; private set; }
|
||||
|
||||
public void UpdateExpiresAt(DateTime expiresAtUtc)
|
||||
{
|
||||
ExpiresAtUtc = expiresAtUtc;
|
||||
LastRemaining = null;
|
||||
}
|
||||
|
||||
public void UpdateRemaining(TimeSpan remaining)
|
||||
{
|
||||
LastRemaining = remaining;
|
||||
}
|
||||
}
|
||||
|
||||
private sealed record NotificationPayload(GroupFullInfoDto Group, int ThresholdMinutes, DateTime ExpiresAtUtc);
|
||||
}
|
||||
208
MareSynchronos/UI/AutoDetectUi.cs
Normal file
208
MareSynchronos/UI/AutoDetectUi.cs
Normal file
@@ -0,0 +1,208 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface.Colors;
|
||||
using Dalamud.Game.ClientState.Objects.SubKinds;
|
||||
using Dalamud.Interface.Utility;
|
||||
using Dalamud.Interface.Utility.Raii;
|
||||
using Dalamud.Plugin.Services;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.Services;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using System.Numerics;
|
||||
using System.Globalization;
|
||||
using System.Text;
|
||||
|
||||
namespace MareSynchronos.UI;
|
||||
|
||||
public class AutoDetectUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
private readonly MareConfigService _configService;
|
||||
private readonly DalamudUtilService _dalamud;
|
||||
private readonly IObjectTable _objectTable;
|
||||
private readonly Services.AutoDetect.AutoDetectRequestService _requestService;
|
||||
private readonly PairManager _pairManager;
|
||||
private List<Services.Mediator.NearbyEntry> _entries = new();
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
=> LocalizationService.Instance?.GetString(key, fallback, args)
|
||||
?? string.Format(CultureInfo.CurrentCulture, fallback, args);
|
||||
|
||||
public AutoDetectUi(ILogger<AutoDetectUi> logger, MareMediator mediator,
|
||||
MareConfigService configService, DalamudUtilService dalamudUtilService, IObjectTable objectTable,
|
||||
Services.AutoDetect.AutoDetectRequestService requestService, PairManager pairManager,
|
||||
PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "AutoDetect", performanceCollectorService)
|
||||
{
|
||||
_configService = configService;
|
||||
_dalamud = dalamudUtilService;
|
||||
_objectTable = objectTable;
|
||||
_requestService = requestService;
|
||||
_pairManager = pairManager;
|
||||
|
||||
Flags |= ImGuiWindowFlags.NoScrollbar;
|
||||
SizeConstraints = new WindowSizeConstraints()
|
||||
{
|
||||
MinimumSize = new Vector2(350, 220),
|
||||
MaximumSize = new Vector2(600, 600),
|
||||
};
|
||||
}
|
||||
|
||||
public override bool DrawConditions()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
using var idScope = ImRaii.PushId("autodetect-ui");
|
||||
|
||||
if (!_configService.Current.EnableAutoDetectDiscovery)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped(L("AutoDetect.Disabled", "Nearby detection is disabled. Enable it in Settings to start detecting nearby Umbra users."), ImGuiColors.DalamudYellow);
|
||||
ImGuiHelpers.ScaledDummy(6);
|
||||
}
|
||||
|
||||
int maxDist = Math.Clamp(_configService.Current.AutoDetectMaxDistanceMeters, 5, 100);
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(L("AutoDetect.MaxDistance", "Max distance (m)"));
|
||||
ImGui.SameLine();
|
||||
ImGui.SetNextItemWidth(120 * ImGuiHelpers.GlobalScale);
|
||||
if (ImGui.SliderInt("##autodetect-dist", ref maxDist, 5, 100))
|
||||
{
|
||||
_configService.Current.AutoDetectMaxDistanceMeters = maxDist;
|
||||
_configService.Save();
|
||||
}
|
||||
|
||||
ImGuiHelpers.ScaledDummy(6);
|
||||
|
||||
// Table header
|
||||
if (ImGui.BeginTable("autodetect-nearby", 5, ImGuiTableFlags.SizingStretchProp))
|
||||
{
|
||||
ImGui.TableSetupColumn(L("AutoDetect.Table.Name", "Name"));
|
||||
ImGui.TableSetupColumn(L("AutoDetect.Table.World", "World"));
|
||||
ImGui.TableSetupColumn(L("AutoDetect.Table.Distance", "Distance"));
|
||||
ImGui.TableSetupColumn(L("AutoDetect.Table.Status", "Status"));
|
||||
ImGui.TableSetupColumn(L("AutoDetect.Table.Action", "Action"));
|
||||
ImGui.TableHeadersRow();
|
||||
|
||||
var data = _entries.Count > 0 ? _entries.Where(e => e.IsMatch).ToList() : new List<Services.Mediator.NearbyEntry>();
|
||||
foreach (var e in data)
|
||||
{
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.TextUnformatted(e.Name);
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.TextUnformatted(e.WorldId == 0 ? L("AutoDetect.World.Unknown", "-") : (_dalamud.WorldData.Value.TryGetValue(e.WorldId, out var w) ? w : e.WorldId.ToString()));
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.TextUnformatted(float.IsNaN(e.Distance)
|
||||
? L("AutoDetect.Distance.Unknown", "-")
|
||||
: string.Format(CultureInfo.CurrentCulture, L("AutoDetect.Distance.Format", "{0:0.0} m"), e.Distance));
|
||||
ImGui.TableNextColumn();
|
||||
bool alreadyPaired = IsAlreadyPairedByUidOrAlias(e);
|
||||
string status = alreadyPaired
|
||||
? L("AutoDetect.Status.Paired", "Paired")
|
||||
: (string.IsNullOrEmpty(e.Token)
|
||||
? L("AutoDetect.Status.RequestsDisabled", "Requests disabled")
|
||||
: L("AutoDetect.Status.OnUmbra", "On Umbra"));
|
||||
ImGui.TextUnformatted(status);
|
||||
ImGui.TableNextColumn();
|
||||
using (ImRaii.Disabled(alreadyPaired || string.IsNullOrEmpty(e.Token)))
|
||||
{
|
||||
if (alreadyPaired)
|
||||
{
|
||||
ImGui.Button(L("AutoDetect.Action.AlreadySynced", "Already sync") + "##" + e.Name);
|
||||
}
|
||||
else if (string.IsNullOrEmpty(e.Token))
|
||||
{
|
||||
ImGui.Button(L("AutoDetect.Action.RequestsDisabled", "Requests disabled") + "##" + e.Name);
|
||||
}
|
||||
else if (ImGui.Button(L("AutoDetect.Action.SendRequest", "Send request") + "##" + e.Name))
|
||||
{
|
||||
_ = _requestService.SendRequestAsync(e.Token!);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.EndTable();
|
||||
}
|
||||
}
|
||||
|
||||
public override void OnOpen()
|
||||
{
|
||||
base.OnOpen();
|
||||
Mediator.Subscribe<Services.Mediator.DiscoveryListUpdated>(this, OnDiscoveryUpdated);
|
||||
}
|
||||
|
||||
public override void OnClose()
|
||||
{
|
||||
Mediator.Unsubscribe<Services.Mediator.DiscoveryListUpdated>(this);
|
||||
base.OnClose();
|
||||
}
|
||||
|
||||
private void OnDiscoveryUpdated(Services.Mediator.DiscoveryListUpdated msg)
|
||||
{
|
||||
_entries = msg.Entries;
|
||||
}
|
||||
|
||||
private List<Services.Mediator.NearbyEntry> BuildLocalSnapshot(int maxDist)
|
||||
{
|
||||
var list = new List<Services.Mediator.NearbyEntry>();
|
||||
var local = _dalamud.GetPlayerCharacter();
|
||||
var localPos = local?.Position ?? Vector3.Zero;
|
||||
for (int i = 0; i < 200; i += 2)
|
||||
{
|
||||
var obj = _objectTable[i];
|
||||
if (obj == null || obj.ObjectKind != Dalamud.Game.ClientState.Objects.Enums.ObjectKind.Player) continue;
|
||||
if (local != null && obj.Address == local.Address) continue;
|
||||
float dist = local == null ? float.NaN : Vector3.Distance(localPos, obj.Position);
|
||||
if (!float.IsNaN(dist) && dist > maxDist) continue;
|
||||
string name = obj.Name.ToString();
|
||||
ushort worldId = 0;
|
||||
if (obj is IPlayerCharacter pc) worldId = (ushort)pc.HomeWorld.RowId;
|
||||
list.Add(new Services.Mediator.NearbyEntry(name, worldId, dist, false, null, null, null));
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
private bool IsAlreadyPairedByUidOrAlias(Services.Mediator.NearbyEntry e)
|
||||
{
|
||||
try
|
||||
{
|
||||
// 1) Match by UID when available (authoritative)
|
||||
if (!string.IsNullOrEmpty(e.Uid))
|
||||
{
|
||||
foreach (var p in _pairManager.DirectPairs)
|
||||
{
|
||||
if (string.Equals(p.UserData.UID, e.Uid, StringComparison.Ordinal))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
var key = NormalizeKey(e.DisplayName ?? e.Name);
|
||||
if (string.IsNullOrEmpty(key)) return false;
|
||||
foreach (var p in _pairManager.DirectPairs)
|
||||
{
|
||||
if (NormalizeKey(p.UserData.AliasOrUID) == key) return true;
|
||||
if (!string.IsNullOrEmpty(p.UserData.Alias) && NormalizeKey(p.UserData.Alias!) == key) return true;
|
||||
}
|
||||
}
|
||||
catch
|
||||
{
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private static string NormalizeKey(string? input)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(input)) return string.Empty;
|
||||
var formD = input.Normalize(NormalizationForm.FormD);
|
||||
var sb = new StringBuilder(formD.Length);
|
||||
foreach (var ch in formD)
|
||||
{
|
||||
var cat = CharUnicodeInfo.GetUnicodeCategory(ch);
|
||||
if (cat != UnicodeCategory.NonSpacingMark)
|
||||
sb.Append(char.ToLowerInvariant(ch));
|
||||
}
|
||||
return sb.ToString();
|
||||
}
|
||||
}
|
||||
@@ -12,6 +12,7 @@ using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.Services.AutoDetect;
|
||||
using MareSynchronos.UI.Components;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using MareSynchronos.WebAPI;
|
||||
@@ -24,6 +25,7 @@ using System.Diagnostics;
|
||||
using System.Globalization;
|
||||
using System.Numerics;
|
||||
using System.Reflection;
|
||||
using System.Linq;
|
||||
|
||||
namespace MareSynchronos.UI;
|
||||
|
||||
@@ -43,6 +45,8 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
private readonly ServerConfigurationManager _serverManager;
|
||||
private readonly Stopwatch _timeout = new();
|
||||
private readonly CharaDataManager _charaDataManager;
|
||||
private readonly NearbyPendingService _nearbyPending;
|
||||
private readonly AutoDetectRequestService _autoDetectRequestService;
|
||||
private readonly UidDisplayHandler _uidDisplayHandler;
|
||||
private readonly UiSharedService _uiSharedService;
|
||||
private bool _buttonState;
|
||||
@@ -56,11 +60,17 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
private bool _showModalForUserAddition;
|
||||
private bool _showSyncShells;
|
||||
private bool _wasOpen;
|
||||
private bool _nearbyOpen = true;
|
||||
private List<Services.Mediator.NearbyEntry> _nearbyEntries = new();
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiSharedService.Localize(key, fallback, args);
|
||||
|
||||
public CompactUi(ILogger<CompactUi> logger, UiSharedService uiShared, MareConfigService configService, ApiController apiController, PairManager pairManager, ChatService chatService,
|
||||
ServerConfigurationManager serverManager, MareMediator mediator, FileUploadManager fileTransferManager, UidDisplayHandler uidDisplayHandler, CharaDataManager charaDataManager,
|
||||
NearbyPendingService nearbyPendingService,
|
||||
AutoDetectRequestService autoDetectRequestService,
|
||||
PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "###UmbraSyncSyncMainUI", performanceCollectorService)
|
||||
: base(logger, mediator, "###UmbraSyncMainUI", performanceCollectorService)
|
||||
{
|
||||
_uiSharedService = uiShared;
|
||||
_configService = configService;
|
||||
@@ -70,21 +80,23 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_fileTransferManager = fileTransferManager;
|
||||
_uidDisplayHandler = uidDisplayHandler;
|
||||
_charaDataManager = charaDataManager;
|
||||
_nearbyPending = nearbyPendingService;
|
||||
_autoDetectRequestService = autoDetectRequestService;
|
||||
var tagHandler = new TagHandler(_serverManager);
|
||||
|
||||
_groupPanel = new(this, uiShared, _pairManager, chatService, uidDisplayHandler, _configService, _serverManager, _charaDataManager);
|
||||
_selectGroupForPairUi = new(tagHandler, uidDisplayHandler, _uiSharedService);
|
||||
_selectPairsForGroupUi = new(tagHandler, uidDisplayHandler);
|
||||
_selectPairsForGroupUi = new(tagHandler, uidDisplayHandler, _uiSharedService);
|
||||
_pairGroupsUi = new(configService, tagHandler, uidDisplayHandler, apiController, _selectPairsForGroupUi, _uiSharedService);
|
||||
|
||||
#if DEBUG
|
||||
string dev = "Dev Build";
|
||||
var ver = Assembly.GetExecutingAssembly().GetName().Version!;
|
||||
WindowName = $"UmbraSync {dev} ({ver.Major}.{ver.Minor}.{ver.Build})###UmbraSyncSyncMainUIDev";
|
||||
WindowName = $"UmbraSync {dev} ({ver.Major}.{ver.Minor}.{ver.Build})###UmbraSyncMainUIDev";
|
||||
Toggle();
|
||||
#else
|
||||
var ver = Assembly.GetExecutingAssembly().GetName().Version!;
|
||||
WindowName = "UmbraSync " + ver.Major + "." + ver.Minor + "." + ver.Build + "###UmbraSyncSyncMainUI";
|
||||
WindowName = "UmbraSync " + ver.Major + "." + ver.Minor + "." + ver.Build + "###UmbracSyncMainUI";
|
||||
#endif
|
||||
Mediator.Subscribe<SwitchToMainUiMessage>(this, (_) => IsOpen = true);
|
||||
Mediator.Subscribe<SwitchToIntroUiMessage>(this, (_) => IsOpen = false);
|
||||
@@ -92,6 +104,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
Mediator.Subscribe<CutsceneEndMessage>(this, (_) => UiSharedService_GposeEnd());
|
||||
Mediator.Subscribe<DownloadStartedMessage>(this, (msg) => _currentDownloads[msg.DownloadId] = msg.DownloadStatus);
|
||||
Mediator.Subscribe<DownloadFinishedMessage>(this, (msg) => _currentDownloads.TryRemove(msg.DownloadId, out _));
|
||||
Mediator.Subscribe<DiscoveryListUpdated>(this, (msg) => _nearbyEntries = msg.Entries);
|
||||
|
||||
Flags |= ImGuiWindowFlags.NoDocking;
|
||||
|
||||
@@ -104,13 +117,13 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
UiSharedService.AccentColor = new Vector4(0.2f, 0.6f, 1f, 1f); // custom blue
|
||||
UiSharedService.AccentColor = new Vector4(0.63f, 0.25f, 1f, 1f);
|
||||
ImGui.SetCursorPosY(ImGui.GetCursorPosY() - ImGui.GetStyle().WindowPadding.Y - 1f * ImGuiHelpers.GlobalScale + ImGui.GetStyle().ItemSpacing.Y);
|
||||
WindowContentWidth = UiSharedService.GetWindowContentRegionWidth();
|
||||
if (!_apiController.IsCurrentVersion)
|
||||
{
|
||||
var ver = _apiController.CurrentClientVersion;
|
||||
var unsupported = "UNSUPPORTED VERSION";
|
||||
var unsupported = L("Compact.Version.UnsupportedTitle", "UNSUPPORTED VERSION");
|
||||
using (_uiSharedService.UidFont.Push())
|
||||
{
|
||||
var uidTextSize = ImGui.CalcTextSize(unsupported);
|
||||
@@ -118,8 +131,10 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextColored(ImGuiColors.DalamudRed, unsupported);
|
||||
}
|
||||
UiSharedService.ColorTextWrapped($"Your UmbraSync installation is out of date, the current version is {ver.Major}.{ver.Minor}.{ver.Build}. " +
|
||||
$"It is highly recommended to keep UmbraSync up to date. Open /xlplugins and update the plugin.", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("Compact.Version.Outdated",
|
||||
"Your UmbraSync installation is out of date, the current version is {0}.{1}.{2}. It is highly recommended to keep UmbraSync up to date. Open /xlplugins and update the plugin.",
|
||||
ver.Major, ver.Minor, ver.Build),
|
||||
ImGuiColors.DalamudRed);
|
||||
}
|
||||
|
||||
using (ImRaii.PushId("header")) DrawUIDHeader();
|
||||
@@ -144,7 +159,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
ImGui.PopStyleColor();
|
||||
}
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("Individual pairs");
|
||||
UiSharedService.AttachToolTip(L("Compact.Toggle.IndividualPairs", "Individual pairs"));
|
||||
|
||||
ImGui.SameLine();
|
||||
|
||||
@@ -163,7 +178,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
}
|
||||
ImGui.PopFont();
|
||||
|
||||
UiSharedService.AttachToolTip("Syncshells");
|
||||
UiSharedService.AttachToolTip(L("Compact.Toggle.Syncshells", "Syncshells"));
|
||||
|
||||
ImGui.Separator();
|
||||
if (!hasShownSyncShells)
|
||||
@@ -185,12 +200,14 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
_lastAddedUser = _pairManager.LastAddedUser;
|
||||
_pairManager.LastAddedUser = null;
|
||||
ImGui.OpenPopup("Set Notes for New User");
|
||||
var setNotesTitle = L("Compact.AddUser.ModalTitle", "Set Notes for New User");
|
||||
ImGui.OpenPopup(setNotesTitle);
|
||||
_showModalForUserAddition = true;
|
||||
_lastAddedUserComment = string.Empty;
|
||||
}
|
||||
|
||||
if (ImGui.BeginPopupModal("Set Notes for New User", ref _showModalForUserAddition, UiSharedService.PopupWindowFlags))
|
||||
var setNotesModalTitle = L("Compact.AddUser.ModalTitle", "Set Notes for New User");
|
||||
if (ImGui.BeginPopupModal(setNotesModalTitle, ref _showModalForUserAddition, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
if (_lastAddedUser == null)
|
||||
{
|
||||
@@ -198,9 +215,9 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.TextWrapped($"You have successfully added {_lastAddedUser.UserData.AliasOrUID}. Set a local note for the user in the field below:");
|
||||
ImGui.InputTextWithHint("##noteforuser", $"Note for {_lastAddedUser.UserData.AliasOrUID}", ref _lastAddedUserComment, 100);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Save, "Save Note"))
|
||||
UiSharedService.TextWrapped(L("Compact.AddUser.Description", "You have successfully added {0}. Set a local note for the user in the field below:", _lastAddedUser.UserData.AliasOrUID));
|
||||
ImGui.InputTextWithHint("##noteforuser", L("Compact.AddUser.NoteHint", "Note for {0}", _lastAddedUser.UserData.AliasOrUID), ref _lastAddedUserComment, 100);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Save, L("Compact.AddUser.Save", "Save Note")))
|
||||
{
|
||||
_serverManager.SetNoteForUid(_lastAddedUser.UserData.UID, _lastAddedUserComment);
|
||||
_lastAddedUser = null;
|
||||
@@ -235,7 +252,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
if (keys.Any())
|
||||
{
|
||||
if (_secretKeyIdx == -1) _secretKeyIdx = keys.First().Key;
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Plus, "Add current character with secret key"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Plus, L("Compact.AddCharacter.Button", "Add current character with secret key")))
|
||||
{
|
||||
_serverManager.CurrentServer!.Authentications.Add(new MareConfiguration.Models.Authentication()
|
||||
{
|
||||
@@ -249,11 +266,11 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_ = _apiController.CreateConnections();
|
||||
}
|
||||
|
||||
_uiSharedService.DrawCombo("Secret Key##addCharacterSecretKey", keys, (f) => f.Value.FriendlyName, (f) => _secretKeyIdx = f.Key);
|
||||
_uiSharedService.DrawCombo(L("Compact.AddCharacter.SecretKeyLabel", "Secret Key") + "##addCharacterSecretKey", keys, (f) => f.Value.FriendlyName, (f) => _secretKeyIdx = f.Key);
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("No secret keys are configured for the current server.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorTextWrapped(L("Compact.AddCharacter.NoKeys", "No secret keys are configured for the current server."), ImGuiColors.DalamudYellow);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -261,7 +278,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
var buttonSize = _uiSharedService.GetIconButtonSize(FontAwesomeIcon.Plus);
|
||||
ImGui.SetNextItemWidth(UiSharedService.GetWindowContentRegionWidth() - ImGui.GetWindowContentRegionMin().X - buttonSize.X);
|
||||
ImGui.InputTextWithHint("##otheruid", "Other players UID/Alias", ref _pairToAdd, 20);
|
||||
ImGui.InputTextWithHint("##otheruid", L("Compact.AddPair.Hint", "Other player's UID/Alias"), ref _pairToAdd, 20);
|
||||
ImGui.SameLine(ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth() - buttonSize.X);
|
||||
var canAdd = !_pairManager.DirectPairs.Any(p => string.Equals(p.UserData.UID, _pairToAdd, StringComparison.Ordinal) || string.Equals(p.UserData.Alias, _pairToAdd, StringComparison.Ordinal));
|
||||
using (ImRaii.Disabled(!canAdd))
|
||||
@@ -271,7 +288,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_ = _apiController.UserAddPair(new(new(_pairToAdd)));
|
||||
_pairToAdd = string.Empty;
|
||||
}
|
||||
UiSharedService.AttachToolTip("Pair with " + (_pairToAdd.IsNullOrEmpty() ? "other user" : _pairToAdd));
|
||||
UiSharedService.AttachToolTip(L("Compact.AddPair.Tooltip", "Pair with {0}", _pairToAdd.IsNullOrEmpty() ? L("Compact.AddPair.Tooltip.DefaultUser", "other user") : _pairToAdd));
|
||||
}
|
||||
|
||||
ImGuiHelpers.ScaledDummy(2);
|
||||
@@ -289,7 +306,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
: 0;
|
||||
|
||||
ImGui.SetNextItemWidth(WindowContentWidth - spacing);
|
||||
ImGui.InputTextWithHint("##filter", "Filter for UID/notes", ref _characterOrCommentFilter, 255);
|
||||
ImGui.InputTextWithHint("##filter", L("Compact.Filter.Hint", "Filter for UID/notes"), ref _characterOrCommentFilter, 255);
|
||||
|
||||
if (userCount == 0) return;
|
||||
|
||||
@@ -338,9 +355,12 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_buttonState = !_buttonState;
|
||||
}
|
||||
if (!_timeout.IsRunning)
|
||||
UiSharedService.AttachToolTip($"Hold Control to {(button == FontAwesomeIcon.Play ? "resume" : "pause")} pairing with {users.Count} out of {userCount} displayed users.");
|
||||
UiSharedService.AttachToolTip(L("Compact.Filter.ToggleTooltip", "Hold Control to {0} pairing with {1} out of {2} displayed users.",
|
||||
button == FontAwesomeIcon.Play ? L("Compact.Filter.ToggleTooltip.Resume", "resume") : L("Compact.Filter.ToggleTooltip.Pause", "pause"),
|
||||
users.Count, userCount));
|
||||
else
|
||||
UiSharedService.AttachToolTip($"Next execution is available at {(5000 - _timeout.ElapsedMilliseconds) / 1000} seconds");
|
||||
UiSharedService.AttachToolTip(L("Compact.Filter.CooldownTooltip", "Next execution is available at {0} seconds",
|
||||
(5000 - _timeout.ElapsedMilliseconds) / 1000));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -367,6 +387,122 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
|
||||
_pairGroupsUi.Draw(visibleUsers, onlineUsers, offlineUsers);
|
||||
|
||||
// Always show a Nearby group when detection is enabled, even if empty
|
||||
if (_configService.Current.EnableAutoDetectDiscovery)
|
||||
{
|
||||
using (ImRaii.PushId("group-Nearby"))
|
||||
{
|
||||
var icon = _nearbyOpen ? FontAwesomeIcon.CaretSquareDown : FontAwesomeIcon.CaretSquareRight;
|
||||
_uiSharedService.IconText(icon);
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left)) _nearbyOpen = !_nearbyOpen;
|
||||
ImGui.SameLine();
|
||||
var onUmbra = _nearbyEntries?.Count(e => e.IsMatch) ?? 0;
|
||||
ImGui.TextUnformatted(L("Compact.Nearby.Title", "Nearby ({0})", onUmbra));
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left)) _nearbyOpen = !_nearbyOpen;
|
||||
var nearbyButtonLabel = L("Compact.Nearby.Button", "Nearby");
|
||||
var btnWidth = _uiSharedService.GetIconTextButtonSize(FontAwesomeIcon.UserPlus, nearbyButtonLabel);
|
||||
var headerRight = ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth();
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPosX(headerRight - btnWidth);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.UserPlus, nearbyButtonLabel, btnWidth))
|
||||
{
|
||||
Mediator.Publish(new UiToggleMessage(typeof(AutoDetectUi)));
|
||||
}
|
||||
|
||||
if (_nearbyOpen)
|
||||
{
|
||||
ImGui.Indent();
|
||||
var nearby = _nearbyEntries == null
|
||||
? new List<Services.Mediator.NearbyEntry>()
|
||||
: _nearbyEntries.Where(e => e.IsMatch)
|
||||
.OrderBy(e => e.Distance)
|
||||
.ToList();
|
||||
if (nearby.Count == 0)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped(L("Compact.Nearby.None", "No nearby players detected."), ImGuiColors.DalamudGrey3);
|
||||
}
|
||||
else
|
||||
{
|
||||
foreach (var e in nearby)
|
||||
{
|
||||
var name = e.DisplayName ?? e.Name;
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(name);
|
||||
var right = ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth();
|
||||
ImGui.SameLine();
|
||||
|
||||
bool isPaired = false;
|
||||
try
|
||||
{
|
||||
isPaired = _pairManager.DirectPairs.Any(p => string.Equals(p.UserData.UID, e.Uid, StringComparison.Ordinal));
|
||||
}
|
||||
catch
|
||||
{
|
||||
var key = (e.DisplayName ?? e.Name) ?? string.Empty;
|
||||
isPaired = _pairManager.DirectPairs.Any(p => string.Equals(p.UserData.AliasOrUID, key, StringComparison.OrdinalIgnoreCase));
|
||||
}
|
||||
|
||||
var statusButtonSize = _uiSharedService.GetIconButtonSize(FontAwesomeIcon.UserPlus);
|
||||
ImGui.SetCursorPosX(right - statusButtonSize.X);
|
||||
|
||||
if (isPaired)
|
||||
{
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Check, ImGuiColors.ParsedGreen);
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Tooltip.AlreadyPaired", "Already paired on Umbra"));
|
||||
}
|
||||
else if (!e.AcceptPairRequests)
|
||||
{
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Ban, ImGuiColors.DalamudGrey3);
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Tooltip.RequestsDisabled", "Pair requests are disabled for this player"));
|
||||
}
|
||||
else if (!string.IsNullOrEmpty(e.Token))
|
||||
{
|
||||
if (_uiSharedService.IconButton(FontAwesomeIcon.UserPlus))
|
||||
{
|
||||
_ = _autoDetectRequestService.SendRequestAsync(e.Token!);
|
||||
}
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Tooltip.SendInvite", "Send Umbra invitation"));
|
||||
}
|
||||
else
|
||||
{
|
||||
_uiSharedService.IconText(FontAwesomeIcon.QuestionCircle, ImGuiColors.DalamudGrey3);
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Tooltip.CannotInvite", "Unable to invite this player"));
|
||||
}
|
||||
}
|
||||
}
|
||||
try
|
||||
{
|
||||
var inbox = _nearbyPending;
|
||||
if (inbox != null && inbox.Pending.Count > 0)
|
||||
{
|
||||
ImGuiHelpers.ScaledDummy(6);
|
||||
_uiSharedService.BigText(L("Compact.Nearby.Incoming", "Incoming requests"));
|
||||
foreach (var kv in inbox.Pending)
|
||||
{
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(L("Compact.Nearby.Incoming.Entry", "{0} [{1}]", kv.Value, kv.Key));
|
||||
ImGui.SameLine();
|
||||
if (_uiSharedService.IconButton(FontAwesomeIcon.Check))
|
||||
{
|
||||
_ = inbox.AcceptAsync(kv.Key);
|
||||
}
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Incoming.Accept", "Accept and add as pair"));
|
||||
ImGui.SameLine();
|
||||
if (_uiSharedService.IconButton(FontAwesomeIcon.Times))
|
||||
{
|
||||
inbox.Remove(kv.Key);
|
||||
}
|
||||
UiSharedService.AttachToolTip(L("Compact.Nearby.Incoming.Dismiss", "Dismiss request"));
|
||||
}
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
ImGui.Unindent();
|
||||
ImGui.Separator();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ImGui.EndChild();
|
||||
}
|
||||
|
||||
@@ -375,8 +511,11 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
var buttonSize = _uiSharedService.GetIconButtonSize(FontAwesomeIcon.Link);
|
||||
var userCount = _apiController.OnlineUsers.ToString(CultureInfo.InvariantCulture);
|
||||
var userSize = ImGui.CalcTextSize(userCount);
|
||||
var textSize = ImGui.CalcTextSize("Users Online");
|
||||
string shardConnection = string.Equals(_apiController.ServerInfo.ShardName, "Main", StringComparison.OrdinalIgnoreCase) ? string.Empty : $"Shard: {_apiController.ServerInfo.ShardName}";
|
||||
var usersOnlineText = L("Compact.ServerStatus.UsersOnline", "Users Online");
|
||||
var textSize = ImGui.CalcTextSize(usersOnlineText);
|
||||
string shardConnection = string.Equals(_apiController.ServerInfo.ShardName, "Main", StringComparison.OrdinalIgnoreCase)
|
||||
? string.Empty
|
||||
: L("Compact.ServerStatus.Shard", "Shard: {0}", _apiController.ServerInfo.ShardName);
|
||||
var shardTextSize = ImGui.CalcTextSize(shardConnection);
|
||||
var printShard = !string.IsNullOrEmpty(_apiController.ServerInfo.ShardName) && shardConnection != string.Empty;
|
||||
|
||||
@@ -384,15 +523,15 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
ImGui.SetCursorPosX((ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth()) / 2 - (userSize.X + textSize.X) / 2 - ImGui.GetStyle().ItemSpacing.X / 2);
|
||||
if (!printShard) ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextColored(ImGuiColors.ParsedBlue, userCount);
|
||||
ImGui.TextColored(UiSharedService.AccentColor, userCount);
|
||||
ImGui.SameLine();
|
||||
if (!printShard) ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted("Users Online");
|
||||
ImGui.TextUnformatted(usersOnlineText);
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextColored(ImGuiColors.DalamudRed, "Not connected to any server");
|
||||
ImGui.TextColored(ImGuiColors.DalamudRed, L("Compact.ServerStatus.NotConnected", "Not connected to any server"));
|
||||
}
|
||||
|
||||
if (printShard)
|
||||
@@ -407,8 +546,9 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
ImGui.SetCursorPosY(ImGui.GetCursorPosY() - ((userSize.Y + textSize.Y) / 2 + shardTextSize.Y) / 2 - ImGui.GetStyle().ItemSpacing.Y + buttonSize.Y / 2);
|
||||
}
|
||||
var color = UiSharedService.GetBoolColor(!_serverManager.CurrentServer!.FullPause);
|
||||
var connectedIcon = !_serverManager.CurrentServer.FullPause ? FontAwesomeIcon.Link : FontAwesomeIcon.Unlink;
|
||||
var isLinked = !_serverManager.CurrentServer!.FullPause;
|
||||
var color = isLinked ? new Vector4(0.63f, 0.25f, 1f, 1f) : UiSharedService.GetBoolColor(isLinked);
|
||||
var connectedIcon = isLinked ? FontAwesomeIcon.Link : FontAwesomeIcon.Unlink;
|
||||
|
||||
if (_apiController.ServerState is ServerState.Connected)
|
||||
{
|
||||
@@ -417,7 +557,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
Mediator.Publish(new UiToggleMessage(typeof(EditProfileUi)));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Edit your Profile");
|
||||
UiSharedService.AttachToolTip(L("Compact.ServerStatus.EditProfile", "Edit your Profile"));
|
||||
}
|
||||
|
||||
ImGui.SameLine(ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth() - buttonSize.X);
|
||||
@@ -436,7 +576,9 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_ = _apiController.CreateConnections();
|
||||
}
|
||||
ImGui.PopStyleColor();
|
||||
UiSharedService.AttachToolTip(!_serverManager.CurrentServer.FullPause ? "Disconnect from " + _serverManager.CurrentServer.ServerName : "Connect to " + _serverManager.CurrentServer.ServerName);
|
||||
UiSharedService.AttachToolTip(!_serverManager.CurrentServer.FullPause
|
||||
? L("Compact.ServerStatus.Disconnect", "Disconnect from {0}", _serverManager.CurrentServer.ServerName)
|
||||
: L("Compact.ServerStatus.Connect", "Connect to {0}", _serverManager.CurrentServer.ServerName));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -483,22 +625,19 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
ImGui.SameLine(WindowContentWidth - textSize.X);
|
||||
ImGui.TextUnformatted(downloadText);
|
||||
}
|
||||
|
||||
var bottomButtonWidth = (WindowContentWidth - ImGui.GetStyle().ItemSpacing.X) / 2;
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PersonCircleQuestion, "Character Analysis", bottomButtonWidth))
|
||||
var spacing = ImGui.GetStyle().ItemSpacing.X;
|
||||
var bottomButtonWidth = (WindowContentWidth - spacing) / 2f;
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PersonCircleQuestion, L("Compact.Transfers.CharacterAnalysis", "Character Analysis"), bottomButtonWidth))
|
||||
{
|
||||
Mediator.Publish(new UiToggleMessage(typeof(DataAnalysisUi)));
|
||||
}
|
||||
|
||||
ImGui.SameLine();
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Running, "Character Data Hub", bottomButtonWidth))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Running, L("Compact.Transfers.CharacterDataHub", "Character Data Hub"), bottomButtonWidth))
|
||||
{
|
||||
Mediator.Publish(new UiToggleMessage(typeof(CharaDataHubUi)));
|
||||
}
|
||||
|
||||
ImGui.SameLine();
|
||||
ImGuiHelpers.ScaledDummy(2);
|
||||
}
|
||||
|
||||
private void DrawUIDHeader()
|
||||
@@ -522,9 +661,9 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
Mediator.Publish(new OpenSettingsUiMessage());
|
||||
}
|
||||
UiSharedService.AttachToolTip("Open the UmbraSync Settings");
|
||||
UiSharedService.AttachToolTip(L("Compact.Header.SettingsTooltip", "Open the UmbraSync settings"));
|
||||
|
||||
ImGui.SameLine(); //Important to draw the uidText consistently
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPos(originalPos);
|
||||
|
||||
if (_apiController.ServerState is ServerState.Connected)
|
||||
@@ -535,7 +674,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
ImGui.SetClipboardText(_apiController.DisplayName);
|
||||
}
|
||||
UiSharedService.AttachToolTip("Copy your UID to clipboard");
|
||||
UiSharedService.AttachToolTip(L("Compact.Header.CopyUid", "Copy your UID to clipboard"));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
ImGui.SetWindowFontScale(1f);
|
||||
@@ -570,18 +709,19 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
return _apiController.ServerState switch
|
||||
{
|
||||
ServerState.Connecting => "Attempting to connect to the server.",
|
||||
ServerState.Reconnecting => "Connection to server interrupted, attempting to reconnect to the server.",
|
||||
ServerState.Disconnected => "You are currently disconnected from the sync server.",
|
||||
ServerState.Disconnecting => "Disconnecting from the server",
|
||||
ServerState.Unauthorized => "Server Response: " + _apiController.AuthFailureMessage,
|
||||
ServerState.Offline => "Your selected sync server is currently offline.",
|
||||
ServerState.Connecting => L("Compact.ServerError.Connecting", "Attempting to connect to the server."),
|
||||
ServerState.Reconnecting => L("Compact.ServerError.Reconnecting", "Connection to server interrupted, attempting to reconnect to the server."),
|
||||
ServerState.Disconnected => L("Compact.ServerError.Disconnected", "You are currently disconnected from the sync server."),
|
||||
ServerState.Disconnecting => L("Compact.ServerError.Disconnecting", "Disconnecting from the server"),
|
||||
ServerState.Unauthorized => L("Compact.ServerError.Unauthorized", "Server Response: {0}", _apiController.AuthFailureMessage),
|
||||
ServerState.Offline => L("Compact.ServerError.Offline", "Your selected sync server is currently offline."),
|
||||
ServerState.VersionMisMatch =>
|
||||
"Your plugin or the server you are connecting to is out of date. Please update your plugin now. If you already did so, contact the server provider to update their server to the latest version.",
|
||||
ServerState.RateLimited => "You are rate limited for (re)connecting too often. Disconnect, wait 10 minutes and try again.",
|
||||
L("Compact.ServerError.VersionMismatch",
|
||||
"Your plugin or the server you are connecting to is out of date. Please update your plugin now. If you already did so, contact the server provider to update their server to the latest version."),
|
||||
ServerState.RateLimited => L("Compact.ServerError.RateLimited", "You are rate limited for (re)connecting too often. Disconnect, wait 10 minutes and try again."),
|
||||
ServerState.Connected => string.Empty,
|
||||
ServerState.NoSecretKey => "You have no secret key set for this current character. Use the button below or open the settings and set a secret key for the current character. You can reuse the same secret key for multiple characters.",
|
||||
ServerState.MultiChara => "Your Character Configuration has multiple characters configured with same name and world. You will not be able to connect until you fix this issue. Remove the duplicates from the configuration in Settings -> Service Settings -> Character Management and reconnect manually after.",
|
||||
ServerState.NoSecretKey => L("Compact.ServerError.NoSecretKey", "You have no secret key set for this current character. Use the button below or open the settings and set a secret key for the current character. You can reuse the same secret key for multiple characters."),
|
||||
ServerState.MultiChara => L("Compact.ServerError.MultiChara", "Your Character Configuration has multiple characters configured with same name and world. You will not be able to connect until you fix this issue. Remove the duplicates from the configuration in Settings -> Service Settings -> Character Management and reconnect manually after."),
|
||||
_ => string.Empty
|
||||
};
|
||||
}
|
||||
@@ -592,7 +732,7 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
ServerState.Connecting => ImGuiColors.DalamudYellow,
|
||||
ServerState.Reconnecting => ImGuiColors.DalamudRed,
|
||||
ServerState.Connected => new Vector4(0.2f, 0.6f, 1f, 1f), // custom blue
|
||||
ServerState.Connected => new Vector4(0.63f, 0.25f, 1f, 1f), // custom violet
|
||||
ServerState.Disconnected => ImGuiColors.DalamudYellow,
|
||||
ServerState.Disconnecting => ImGuiColors.DalamudYellow,
|
||||
ServerState.Unauthorized => ImGuiColors.DalamudRed,
|
||||
@@ -609,16 +749,16 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
return _apiController.ServerState switch
|
||||
{
|
||||
ServerState.Reconnecting => "Reconnecting",
|
||||
ServerState.Connecting => "Connecting",
|
||||
ServerState.Disconnected => "Disconnected",
|
||||
ServerState.Disconnecting => "Disconnecting",
|
||||
ServerState.Unauthorized => "Unauthorized",
|
||||
ServerState.VersionMisMatch => "Version mismatch",
|
||||
ServerState.Offline => "Unavailable",
|
||||
ServerState.RateLimited => "Rate Limited",
|
||||
ServerState.NoSecretKey => "No Secret Key",
|
||||
ServerState.MultiChara => "Duplicate Characters",
|
||||
ServerState.Reconnecting => L("Compact.UidText.Reconnecting", "Reconnecting"),
|
||||
ServerState.Connecting => L("Compact.UidText.Connecting", "Connecting"),
|
||||
ServerState.Disconnected => L("Compact.UidText.Disconnected", "Disconnected"),
|
||||
ServerState.Disconnecting => L("Compact.UidText.Disconnecting", "Disconnecting"),
|
||||
ServerState.Unauthorized => L("Compact.UidText.Unauthorized", "Unauthorized"),
|
||||
ServerState.VersionMisMatch => L("Compact.UidText.VersionMismatch", "Version mismatch"),
|
||||
ServerState.Offline => L("Compact.UidText.Offline", "Unavailable"),
|
||||
ServerState.RateLimited => L("Compact.UidText.RateLimited", "Rate Limited"),
|
||||
ServerState.NoSecretKey => L("Compact.UidText.NoSecretKey", "No Secret Key"),
|
||||
ServerState.MultiChara => L("Compact.UidText.MultiChara", "Duplicate Characters"),
|
||||
ServerState.Connected => _apiController.DisplayName,
|
||||
_ => string.Empty
|
||||
};
|
||||
@@ -634,4 +774,4 @@ public class CompactUi : WindowMediatorSubscriberBase
|
||||
_wasOpen = IsOpen;
|
||||
IsOpen = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,4 +1,6 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using System;
|
||||
using System.Numerics;
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Colors;
|
||||
using Dalamud.Interface.Utility;
|
||||
@@ -11,6 +13,7 @@ using MareSynchronos.Services;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using MareSynchronos.WebAPI;
|
||||
using MareSynchronos.Localization;
|
||||
|
||||
namespace MareSynchronos.UI.Components;
|
||||
|
||||
@@ -21,6 +24,13 @@ public class DrawGroupPair : DrawPairBase
|
||||
private readonly GroupFullInfoDto _group;
|
||||
private readonly CharaDataManager _charaDataManager;
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(System.Globalization.CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public DrawGroupPair(string id, Pair entry, ApiController apiController,
|
||||
MareMediator mareMediator, GroupFullInfoDto group, GroupPairFullInfoDto fullInfoDto,
|
||||
UidDisplayHandler handler, UiSharedService uiSharedService, CharaDataManager charaDataManager)
|
||||
@@ -38,40 +48,50 @@ public class DrawGroupPair : DrawPairBase
|
||||
var entryIsMod = _fullInfoDto.GroupPairStatusInfo.IsModerator();
|
||||
var entryIsOwner = string.Equals(_pair.UserData.UID, _group.OwnerUID, StringComparison.Ordinal);
|
||||
var entryIsPinned = _fullInfoDto.GroupPairStatusInfo.IsPinned();
|
||||
var presenceIcon = _pair.IsVisible ? FontAwesomeIcon.Eye : (_pair.IsOnline ? FontAwesomeIcon.Link : FontAwesomeIcon.Unlink);
|
||||
var presenceColor = (_pair.IsOnline || _pair.IsVisible) ? ImGuiColors.ParsedGreen : ImGuiColors.DalamudRed;
|
||||
var presenceText = entryUID + " is offline";
|
||||
var presenceIcon = _pair.IsVisible ? FontAwesomeIcon.Eye : FontAwesomeIcon.CloudMoon;
|
||||
var presenceColor = (_pair.IsOnline || _pair.IsVisible) ? new Vector4(0.63f, 0.25f, 1f, 1f) : ImGuiColors.DalamudGrey;
|
||||
var presenceText = L("GroupPair.Presence.Offline", "{0} is offline", entryUID);
|
||||
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
bool drewPrefixIcon = false;
|
||||
|
||||
if (_pair.IsPaused)
|
||||
{
|
||||
presenceIcon = FontAwesomeIcon.Question;
|
||||
presenceColor = ImGuiColors.DalamudGrey;
|
||||
presenceText = entryUID + " online status is unknown (paused)";
|
||||
presenceText = L("GroupPair.Presence.Paused", "{0} online status is unknown (paused)", entryUID);
|
||||
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.PauseCircle.ToIconString(), ImGuiColors.DalamudYellow);
|
||||
ImGui.PopFont();
|
||||
|
||||
UiSharedService.AttachToolTip("Pairing status with " + entryUID + " is paused");
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.Paused", "Pairing status with {0} is paused", entryUID));
|
||||
drewPrefixIcon = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.Check.ToIconString(), ImGuiColors.ParsedGreen);
|
||||
ImGui.PopFont();
|
||||
|
||||
UiSharedService.AttachToolTip("You are paired with " + entryUID);
|
||||
bool individuallyPaired = _pair.UserPair != null;
|
||||
var violet = new Vector4(0.63f, 0.25f, 1f, 1f);
|
||||
if (individuallyPaired && (_pair.IsOnline || _pair.IsVisible))
|
||||
{
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.Moon.ToIconString(), violet);
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.IndividuallyPaired", "You are individually paired with {0}", entryUID));
|
||||
drewPrefixIcon = true;
|
||||
}
|
||||
}
|
||||
if (drewPrefixIcon)
|
||||
ImGui.SameLine();
|
||||
|
||||
if (_pair.IsOnline && !_pair.IsVisible) presenceText = entryUID + " is online";
|
||||
else if (_pair.IsOnline && _pair.IsVisible) presenceText = entryUID + " is visible: " + _pair.PlayerName + Environment.NewLine + "Click to target this player";
|
||||
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(presenceIcon.ToIconString(), presenceColor);
|
||||
ImGui.PopFont();
|
||||
|
||||
if (_pair.IsOnline && !_pair.IsVisible)
|
||||
presenceText = L("GroupPair.Presence.Online", "{0} is online", entryUID);
|
||||
else if (_pair.IsOnline && _pair.IsVisible)
|
||||
presenceText = L("GroupPair.Presence.Visible", "{0} is visible: {1}\nClick to target this player", entryUID, _pair.PlayerName);
|
||||
|
||||
if (_pair.IsVisible)
|
||||
{
|
||||
if (ImGui.IsItemClicked())
|
||||
@@ -81,19 +101,23 @@ public class DrawGroupPair : DrawPairBase
|
||||
if (_pair.LastAppliedDataBytes >= 0)
|
||||
{
|
||||
presenceText += UiSharedService.TooltipSeparator;
|
||||
presenceText += ((!_pair.IsVisible) ? "(Last) " : string.Empty) + "Mods Info" + Environment.NewLine;
|
||||
presenceText += "Files Size: " + UiSharedService.ByteToString(_pair.LastAppliedDataBytes, true);
|
||||
presenceText += (!_pair.IsVisible ? L("GroupPair.Presence.LastPrefix", "(Last) ") : string.Empty)
|
||||
+ L("GroupPair.Presence.ModsInfo", "Mods Info") + Environment.NewLine;
|
||||
presenceText += L("GroupPair.Presence.FilesSize", "Files Size: {0}", UiSharedService.ByteToString(_pair.LastAppliedDataBytes, true));
|
||||
if (_pair.LastAppliedApproximateVRAMBytes >= 0)
|
||||
{
|
||||
presenceText += Environment.NewLine + "Approx. VRAM Usage: " + UiSharedService.ByteToString(_pair.LastAppliedApproximateVRAMBytes, true);
|
||||
presenceText += Environment.NewLine + L("GroupPair.Presence.Vram", "Approx. VRAM Usage: {0}", UiSharedService.ByteToString(_pair.LastAppliedApproximateVRAMBytes, true));
|
||||
}
|
||||
if (_pair.LastAppliedDataTris >= 0)
|
||||
{
|
||||
presenceText += Environment.NewLine + "Triangle Count (excl. Vanilla): "
|
||||
+ (_pair.LastAppliedDataTris > 1000 ? (_pair.LastAppliedDataTris / 1000d).ToString("0.0'k'") : _pair.LastAppliedDataTris);
|
||||
var trisValue = _pair.LastAppliedDataTris > 1000
|
||||
? (_pair.LastAppliedDataTris / 1000d).ToString("0.0'k'", System.Globalization.CultureInfo.CurrentCulture)
|
||||
: _pair.LastAppliedDataTris.ToString(System.Globalization.CultureInfo.CurrentCulture);
|
||||
presenceText += Environment.NewLine + L("GroupPair.Presence.Tris", "Triangle Count (excl. Vanilla): {0}", trisValue);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
UiSharedService.AttachToolTip(presenceText);
|
||||
|
||||
if (entryIsOwner)
|
||||
@@ -103,7 +127,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.Crown.ToIconString());
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("User is owner of this Syncshell");
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.Owner", "User is owner of this Syncshell"));
|
||||
}
|
||||
else if (entryIsMod)
|
||||
{
|
||||
@@ -112,7 +136,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.UserShield.ToIconString());
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("User is moderator of this Syncshell");
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.Moderator", "User is moderator of this Syncshell"));
|
||||
}
|
||||
else if (entryIsPinned)
|
||||
{
|
||||
@@ -121,7 +145,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.Thumbtack.ToIconString());
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("User is pinned in this Syncshell");
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.Pinned", "User is pinned in this Syncshell"));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -166,9 +190,10 @@ public class DrawGroupPair : DrawPairBase
|
||||
{
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Running);
|
||||
|
||||
UiSharedService.AttachToolTip($"This user has shared {sharedData!.Count} Character Data Sets with you." + UiSharedService.TooltipSeparator
|
||||
+ "Click to open the Character Data Hub and show the entries.");
|
||||
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Tooltip.SharedData", "This user has shared {0} Character Data Sets with you.", sharedData!.Count)
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("GroupPair.Tooltip.SharedData.OpenHub", "Click to open the Character Data Hub and show the entries."));
|
||||
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left))
|
||||
{
|
||||
_mediator.Publish(new OpenCharaDataHubWithFilterMessage(_pair.UserData));
|
||||
@@ -176,7 +201,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
ImGui.SameLine();
|
||||
}
|
||||
|
||||
if (individualAnimDisabled || individualSoundsDisabled)
|
||||
if (individualAnimDisabled || individualSoundsDisabled || individualVFXDisabled)
|
||||
{
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushStyleColor(ImGuiCol.Text, ImGuiColors.DalamudYellow);
|
||||
@@ -186,46 +211,52 @@ public class DrawGroupPair : DrawPairBase
|
||||
{
|
||||
ImGui.BeginTooltip();
|
||||
|
||||
ImGui.TextUnformatted("Individual User permissions");
|
||||
ImGui.TextUnformatted(L("GroupPair.Tooltip.IndividualHeader", "Individual User permissions"));
|
||||
|
||||
if (individualSoundsDisabled)
|
||||
{
|
||||
var userSoundsText = "Sound sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userSoundsText = L("GroupPair.Tooltip.SoundWith", "Sound sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.VolumeOff);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userSoundsText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableSounds() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableSounds() ? "Disabled" : "Enabled"));
|
||||
ImGui.TextUnformatted(L("GroupPair.Tooltip.YouThey", "You: {0}, They: {1}",
|
||||
_pair.UserPair!.OwnPermissions.IsDisableSounds() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled"),
|
||||
_pair.UserPair!.OtherPermissions.IsDisableSounds() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled")));
|
||||
}
|
||||
|
||||
if (individualAnimDisabled)
|
||||
{
|
||||
var userAnimText = "Animation sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userAnimText = L("GroupPair.Tooltip.AnimWith", "Animation sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Stop);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userAnimText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableAnimations() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableAnimations() ? "Disabled" : "Enabled"));
|
||||
ImGui.TextUnformatted(L("GroupPair.Tooltip.YouThey", "You: {0}, They: {1}",
|
||||
_pair.UserPair!.OwnPermissions.IsDisableAnimations() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled"),
|
||||
_pair.UserPair!.OtherPermissions.IsDisableAnimations() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled")));
|
||||
}
|
||||
|
||||
if (individualVFXDisabled)
|
||||
{
|
||||
var userVFXText = "VFX sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userVFXText = L("GroupPair.Tooltip.VfxWith", "VFX sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Circle);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userVFXText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableVFX() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableVFX() ? "Disabled" : "Enabled"));
|
||||
ImGui.TextUnformatted(L("GroupPair.Tooltip.YouThey", "You: {0}, They: {1}",
|
||||
_pair.UserPair!.OwnPermissions.IsDisableVFX() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled"),
|
||||
_pair.UserPair!.OtherPermissions.IsDisableVFX() ? L("GroupPair.Toggle.Disabled", "Disabled") : L("GroupPair.Toggle.Enabled", "Enabled")));
|
||||
}
|
||||
|
||||
ImGui.EndTooltip();
|
||||
}
|
||||
ImGui.SameLine();
|
||||
}
|
||||
else if ((animDisabled || soundsDisabled))
|
||||
else if ((animDisabled || soundsDisabled || vfxDisabled))
|
||||
{
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
_uiSharedService.IconText(permIcon);
|
||||
@@ -233,11 +264,11 @@ public class DrawGroupPair : DrawPairBase
|
||||
{
|
||||
ImGui.BeginTooltip();
|
||||
|
||||
ImGui.TextUnformatted("Syncshell User permissions");
|
||||
ImGui.TextUnformatted(L("GroupPair.Tooltip.SyncshellHeader", "Syncshell User permissions"));
|
||||
|
||||
if (soundsDisabled)
|
||||
{
|
||||
var userSoundsText = "Sound sync disabled by " + _pair.UserData.AliasOrUID;
|
||||
var userSoundsText = L("GroupPair.Tooltip.SoundBy", "Sound sync disabled by {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.VolumeOff);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userSoundsText);
|
||||
@@ -245,7 +276,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
|
||||
if (animDisabled)
|
||||
{
|
||||
var userAnimText = "Animation sync disabled by " + _pair.UserData.AliasOrUID;
|
||||
var userAnimText = L("GroupPair.Tooltip.AnimBy", "Animation sync disabled by {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Stop);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userAnimText);
|
||||
@@ -253,7 +284,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
|
||||
if (vfxDisabled)
|
||||
{
|
||||
var userVFXText = "VFX sync disabled by " + _pair.UserData.AliasOrUID;
|
||||
var userVFXText = L("GroupPair.Tooltip.VfxBy", "VFX sync disabled by {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Circle);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userVFXText);
|
||||
@@ -272,7 +303,7 @@ public class DrawGroupPair : DrawPairBase
|
||||
{
|
||||
_ = _apiController.UserAddPair(new UserDto(new(_pair.UserData.UID)));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Pair with " + entryUID + " individually");
|
||||
UiSharedService.AttachToolTip(L("GroupPair.Popup.PairIndividually", "Pair with {0} individually", entryUID));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
|
||||
@@ -373,4 +404,4 @@ public class DrawGroupPair : DrawPairBase
|
||||
|
||||
return pos - spacing;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,21 +5,32 @@ using Dalamud.Interface.Utility;
|
||||
using Dalamud.Interface.Utility.Raii;
|
||||
using MareSynchronos.API.Data.Extensions;
|
||||
using MareSynchronos.API.Dto.User;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using MareSynchronos.WebAPI;
|
||||
using System.Numerics;
|
||||
using System.Globalization;
|
||||
using System;
|
||||
|
||||
namespace MareSynchronos.UI.Components;
|
||||
|
||||
public class DrawUserPair : DrawPairBase
|
||||
{
|
||||
private static readonly Vector4 Violet = new(0.63f, 0.25f, 1f, 1f);
|
||||
protected readonly MareMediator _mediator;
|
||||
private readonly SelectGroupForPairUi _selectGroupForPairUi;
|
||||
private readonly CharaDataManager _charaDataManager;
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public DrawUserPair(string id, Pair entry, UidDisplayHandler displayHandler, ApiController apiController,
|
||||
MareMediator mareMediator, SelectGroupForPairUi selectGroupForPairUi,
|
||||
UiSharedService uiSharedService, CharaDataManager charaDataManager)
|
||||
@@ -38,58 +49,62 @@ public class DrawUserPair : DrawPairBase
|
||||
|
||||
protected override void DrawLeftSide(float textPosY, float originalY)
|
||||
{
|
||||
FontAwesomeIcon connectionIcon;
|
||||
Vector4 connectionColor;
|
||||
string connectionText;
|
||||
if (!(_pair.UserPair!.OwnPermissions.IsPaired() && _pair.UserPair!.OtherPermissions.IsPaired()))
|
||||
{
|
||||
connectionIcon = FontAwesomeIcon.ArrowUp;
|
||||
connectionText = _pair.UserData.AliasOrUID + " has not added you back";
|
||||
connectionColor = ImGuiColors.DalamudRed;
|
||||
}
|
||||
else if (_pair.UserPair!.OwnPermissions.IsPaused() || _pair.UserPair!.OtherPermissions.IsPaused())
|
||||
{
|
||||
connectionIcon = FontAwesomeIcon.PauseCircle;
|
||||
connectionText = "Pairing status with " + _pair.UserData.AliasOrUID + " is paused";
|
||||
connectionColor = ImGuiColors.DalamudYellow;
|
||||
}
|
||||
else
|
||||
{
|
||||
connectionIcon = FontAwesomeIcon.Check;
|
||||
connectionText = "You are paired with " + _pair.UserData.AliasOrUID;
|
||||
connectionColor = ImGuiColors.ParsedGreen;
|
||||
}
|
||||
var online = _pair.IsOnline;
|
||||
var offlineGrey = ImGuiColors.DalamudGrey3;
|
||||
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(connectionIcon.ToIconString(), connectionColor);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.Moon.ToIconString(), online ? Violet : offlineGrey);
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip(connectionText);
|
||||
UiSharedService.AttachToolTip(online
|
||||
? L("UserPair.Status.Online", "User is online")
|
||||
: L("UserPair.Status.Offline", "User is offline"));
|
||||
if (!(_pair.UserPair!.OwnPermissions.IsPaired() && _pair.UserPair!.OtherPermissions.IsPaired()))
|
||||
{
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.ArrowUp.ToIconString(), ImGuiColors.DalamudRed);
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip(L("UserPair.Tooltip.NotAddedBack", "{0} has not added you back", _pair.UserData.AliasOrUID));
|
||||
}
|
||||
else if (_pair.UserPair!.OwnPermissions.IsPaused() || _pair.UserPair!.OtherPermissions.IsPaused())
|
||||
{
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.PauseCircle.ToIconString(), ImGuiColors.DalamudYellow);
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip(L("UserPair.Tooltip.Paused", "Pairing with {0} is paused", _pair.UserData.AliasOrUID));
|
||||
}
|
||||
if (_pair is { IsOnline: true, IsVisible: true })
|
||||
{
|
||||
ImGui.SameLine();
|
||||
ImGui.SetCursorPosY(textPosY);
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.Eye.ToIconString(), ImGuiColors.ParsedGreen);
|
||||
UiSharedService.ColorText(FontAwesomeIcon.Eye.ToIconString(), Violet);
|
||||
if (ImGui.IsItemClicked())
|
||||
{
|
||||
_mediator.Publish(new TargetPairMessage(_pair));
|
||||
}
|
||||
ImGui.PopFont();
|
||||
var visibleTooltip = _pair.UserData.AliasOrUID + " is visible: " + _pair.PlayerName! + Environment.NewLine + "Click to target this player";
|
||||
var visibleTooltip = L("UserPair.Tooltip.Visible", "{0} is visible: {1}\nClick to target this player", _pair.UserData.AliasOrUID, _pair.PlayerName!);
|
||||
if (_pair.LastAppliedDataBytes >= 0)
|
||||
{
|
||||
visibleTooltip += UiSharedService.TooltipSeparator;
|
||||
visibleTooltip += ((!_pair.IsVisible) ? "(Last) " : string.Empty) + "Mods Info" + Environment.NewLine;
|
||||
visibleTooltip += "Files Size: " + UiSharedService.ByteToString(_pair.LastAppliedDataBytes, true);
|
||||
visibleTooltip += (!_pair.IsVisible ? L("UserPair.Tooltip.Visible.LastPrefix", "(Last) ") : string.Empty)
|
||||
+ L("UserPair.Tooltip.Visible.ModsInfo", "Mods Info") + Environment.NewLine;
|
||||
visibleTooltip += L("UserPair.Tooltip.Visible.FilesSize", "Files Size: {0}", UiSharedService.ByteToString(_pair.LastAppliedDataBytes, true));
|
||||
if (_pair.LastAppliedApproximateVRAMBytes >= 0)
|
||||
{
|
||||
visibleTooltip += Environment.NewLine + "Approx. VRAM Usage: " + UiSharedService.ByteToString(_pair.LastAppliedApproximateVRAMBytes, true);
|
||||
visibleTooltip += Environment.NewLine + L("UserPair.Tooltip.Visible.Vram", "Approx. VRAM Usage: {0}", UiSharedService.ByteToString(_pair.LastAppliedApproximateVRAMBytes, true));
|
||||
}
|
||||
if (_pair.LastAppliedDataTris >= 0)
|
||||
{
|
||||
visibleTooltip += Environment.NewLine + "Triangle Count (excl. Vanilla): "
|
||||
+ (_pair.LastAppliedDataTris > 1000 ? (_pair.LastAppliedDataTris / 1000d).ToString("0.0'k'") : _pair.LastAppliedDataTris);
|
||||
var trisValue = _pair.LastAppliedDataTris > 1000
|
||||
? (_pair.LastAppliedDataTris / 1000d).ToString("0.0'k'", CultureInfo.CurrentCulture)
|
||||
: _pair.LastAppliedDataTris.ToString(CultureInfo.CurrentCulture);
|
||||
visibleTooltip += Environment.NewLine + L("UserPair.Tooltip.Visible.Tris", "Triangle Count (excl. Vanilla): {0}", trisValue);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -134,15 +149,13 @@ public class DrawUserPair : DrawPairBase
|
||||
_ = _apiController.UserSetPairPermissions(new(_pair.UserData, perm));
|
||||
}
|
||||
UiSharedService.AttachToolTip(!_pair.UserPair!.OwnPermissions.IsPaused()
|
||||
? "Pause pairing with " + entryUID
|
||||
: "Resume pairing with " + entryUID);
|
||||
? L("UserPair.Tooltip.Pause", "Pause pairing with {0}", entryUID)
|
||||
: L("UserPair.Tooltip.Resume", "Resume pairing with {0}", entryUID));
|
||||
|
||||
|
||||
var individualSoundsDisabled = (_pair.UserPair?.OwnPermissions.IsDisableSounds() ?? false) || (_pair.UserPair?.OtherPermissions.IsDisableSounds() ?? false);
|
||||
var individualAnimDisabled = (_pair.UserPair?.OwnPermissions.IsDisableAnimations() ?? false) || (_pair.UserPair?.OtherPermissions.IsDisableAnimations() ?? false);
|
||||
var individualVFXDisabled = (_pair.UserPair?.OwnPermissions.IsDisableVFX() ?? false) || (_pair.UserPair?.OtherPermissions.IsDisableVFX() ?? false);
|
||||
|
||||
// Icon for individually applied permissions
|
||||
if (individualSoundsDisabled || individualAnimDisabled || individualVFXDisabled)
|
||||
{
|
||||
var icon = FontAwesomeIcon.ExclamationTriangle;
|
||||
@@ -158,47 +171,63 @@ public class DrawUserPair : DrawPairBase
|
||||
{
|
||||
ImGui.BeginTooltip();
|
||||
|
||||
ImGui.TextUnformatted("Individual User permissions");
|
||||
ImGui.TextUnformatted(L("UserPair.Tooltip.Permission.Header", "Individual user permissions"));
|
||||
|
||||
if (individualSoundsDisabled)
|
||||
{
|
||||
var userSoundsText = "Sound sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userSoundsText = L("UserPair.Tooltip.Permission.Sound", "Sound sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.VolumeOff);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userSoundsText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableSounds() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableSounds() ? "Disabled" : "Enabled"));
|
||||
var youStatus = _pair.UserPair!.OwnPermissions.IsDisableSounds()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
var theyStatus = _pair.UserPair!.OtherPermissions.IsDisableSounds()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
ImGui.TextUnformatted(L("UserPair.Tooltip.Permission.Status", "You: {0}, They: {1}", youStatus, theyStatus));
|
||||
}
|
||||
|
||||
if (individualAnimDisabled)
|
||||
{
|
||||
var userAnimText = "Animation sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userAnimText = L("UserPair.Tooltip.Permission.Animation", "Animation sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Stop);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userAnimText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableAnimations() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableAnimations() ? "Disabled" : "Enabled"));
|
||||
var youStatus = _pair.UserPair!.OwnPermissions.IsDisableAnimations()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
var theyStatus = _pair.UserPair!.OtherPermissions.IsDisableAnimations()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
ImGui.TextUnformatted(L("UserPair.Tooltip.Permission.Status", "You: {0}, They: {1}", youStatus, theyStatus));
|
||||
}
|
||||
|
||||
if (individualVFXDisabled)
|
||||
{
|
||||
var userVFXText = "VFX sync disabled with " + _pair.UserData.AliasOrUID;
|
||||
var userVFXText = L("UserPair.Tooltip.Permission.Vfx", "VFX sync disabled with {0}", _pair.UserData.AliasOrUID);
|
||||
_uiSharedService.IconText(FontAwesomeIcon.Circle);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userVFXText);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted("You: " + (_pair.UserPair!.OwnPermissions.IsDisableVFX() ? "Disabled" : "Enabled") + ", They: " + (_pair.UserPair!.OtherPermissions.IsDisableVFX() ? "Disabled" : "Enabled"));
|
||||
var youStatus = _pair.UserPair!.OwnPermissions.IsDisableVFX()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
var theyStatus = _pair.UserPair!.OtherPermissions.IsDisableVFX()
|
||||
? L("UserPair.Tooltip.Permission.State.Disabled", "Disabled")
|
||||
: L("UserPair.Tooltip.Permission.State.Enabled", "Enabled");
|
||||
ImGui.TextUnformatted(L("UserPair.Tooltip.Permission.Status", "You: {0}, They: {1}", youStatus, theyStatus));
|
||||
}
|
||||
|
||||
ImGui.EndTooltip();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Icon for shared character data
|
||||
if (_charaDataManager.SharedWithYouData.TryGetValue(_pair.UserData, out var sharedData))
|
||||
{
|
||||
var icon = FontAwesomeIcon.Running;
|
||||
@@ -207,8 +236,9 @@ public class DrawUserPair : DrawPairBase
|
||||
ImGui.SameLine(rightSidePos);
|
||||
_uiSharedService.IconText(icon);
|
||||
|
||||
UiSharedService.AttachToolTip($"This user has shared {sharedData.Count} Character Data Sets with you." + UiSharedService.TooltipSeparator
|
||||
+ "Click to open the Character Data Hub and show the entries.");
|
||||
UiSharedService.AttachToolTip(L("UserPair.Tooltip.SharedData", "This user has shared {0} Character Data Sets with you.", sharedData.Count)
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("UserPair.Tooltip.SharedData.OpenHub", "Click to open the Character Data Hub and show the entries."));
|
||||
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left))
|
||||
{
|
||||
@@ -223,7 +253,7 @@ public class DrawUserPair : DrawPairBase
|
||||
{
|
||||
if (entry.IsVisible)
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Eye, "Target player"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Eye, L("UserPair.Menu.Target", "Target player")))
|
||||
{
|
||||
_mediator.Publish(new TargetPairMessage(entry));
|
||||
ImGui.CloseCurrentPopup();
|
||||
@@ -231,44 +261,46 @@ public class DrawUserPair : DrawPairBase
|
||||
}
|
||||
if (!entry.IsPaused)
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.User, "Open Profile"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.User, L("UserPair.Menu.OpenProfile", "Open Profile")))
|
||||
{
|
||||
_displayHandler.OpenProfile(entry);
|
||||
ImGui.CloseCurrentPopup();
|
||||
}
|
||||
UiSharedService.AttachToolTip("Opens the profile for this user in a new window");
|
||||
UiSharedService.AttachToolTip(L("UserPair.Menu.OpenProfile.Tooltip", "Opens the profile for this user in a new window"));
|
||||
}
|
||||
if (entry.IsVisible)
|
||||
{
|
||||
#if DEBUG
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PersonCircleQuestion, "Open Analysis"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PersonCircleQuestion, L("UserPair.Menu.OpenAnalysis", "Open Analysis")))
|
||||
{
|
||||
_displayHandler.OpenAnalysis(_pair);
|
||||
ImGui.CloseCurrentPopup();
|
||||
}
|
||||
#endif
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Sync, "Reload last data"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Sync, L("UserPair.Menu.ReloadData", "Reload last data")))
|
||||
{
|
||||
entry.ApplyLastReceivedData(forced: true);
|
||||
ImGui.CloseCurrentPopup();
|
||||
}
|
||||
UiSharedService.AttachToolTip("This reapplies the last received character data to this character");
|
||||
UiSharedService.AttachToolTip(L("UserPair.Menu.ReloadData.Tooltip", "This reapplies the last received character data to this character"));
|
||||
}
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, "Cycle pause state"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, L("UserPair.Menu.CyclePause", "Cycle pause state")))
|
||||
{
|
||||
_ = _apiController.CyclePause(entry.UserData);
|
||||
ImGui.CloseCurrentPopup();
|
||||
}
|
||||
var entryUID = entry.UserData.AliasOrUID;
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Folder, "Pair Groups"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Folder, L("UserPair.Menu.PairGroups", "Pair Groups")))
|
||||
{
|
||||
_selectGroupForPairUi.Open(entry);
|
||||
}
|
||||
UiSharedService.AttachToolTip("Choose pair groups for " + entryUID);
|
||||
UiSharedService.AttachToolTip(L("UserPair.Menu.PairGroups.Tooltip", "Choose pair groups for {0}", entryUID));
|
||||
|
||||
var isDisableSounds = entry.UserPair!.OwnPermissions.IsDisableSounds();
|
||||
string disableSoundsText = isDisableSounds ? "Enable sound sync" : "Disable sound sync";
|
||||
string disableSoundsText = isDisableSounds
|
||||
? L("UserPair.Menu.EnableSoundSync", "Enable sound sync")
|
||||
: L("UserPair.Menu.DisableSoundSync", "Disable sound sync");
|
||||
var disableSoundsIcon = isDisableSounds ? FontAwesomeIcon.VolumeUp : FontAwesomeIcon.VolumeMute;
|
||||
if (_uiSharedService.IconTextButton(disableSoundsIcon, disableSoundsText))
|
||||
{
|
||||
@@ -278,7 +310,9 @@ public class DrawUserPair : DrawPairBase
|
||||
}
|
||||
|
||||
var isDisableAnims = entry.UserPair!.OwnPermissions.IsDisableAnimations();
|
||||
string disableAnimsText = isDisableAnims ? "Enable animation sync" : "Disable animation sync";
|
||||
string disableAnimsText = isDisableAnims
|
||||
? L("UserPair.Menu.EnableAnimationSync", "Enable animation sync")
|
||||
: L("UserPair.Menu.DisableAnimationSync", "Disable animation sync");
|
||||
var disableAnimsIcon = isDisableAnims ? FontAwesomeIcon.Running : FontAwesomeIcon.Stop;
|
||||
if (_uiSharedService.IconTextButton(disableAnimsIcon, disableAnimsText))
|
||||
{
|
||||
@@ -288,7 +322,9 @@ public class DrawUserPair : DrawPairBase
|
||||
}
|
||||
|
||||
var isDisableVFX = entry.UserPair!.OwnPermissions.IsDisableVFX();
|
||||
string disableVFXText = isDisableVFX ? "Enable VFX sync" : "Disable VFX sync";
|
||||
string disableVFXText = isDisableVFX
|
||||
? L("UserPair.Menu.EnableVfxSync", "Enable VFX sync")
|
||||
: L("UserPair.Menu.DisableVfxSync", "Disable VFX sync");
|
||||
var disableVFXIcon = isDisableVFX ? FontAwesomeIcon.Sun : FontAwesomeIcon.Circle;
|
||||
if (_uiSharedService.IconTextButton(disableVFXIcon, disableVFXText))
|
||||
{
|
||||
@@ -297,10 +333,10 @@ public class DrawUserPair : DrawPairBase
|
||||
_ = _apiController.UserSetPairPermissions(new UserPermissionsDto(entry.UserData, permissions));
|
||||
}
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, "Unpair Permanently") && UiSharedService.CtrlPressed())
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, L("UserPair.Menu.Unpair", "Unpair Permanently")) && UiSharedService.CtrlPressed())
|
||||
{
|
||||
_ = _apiController.UserRemovePair(new(entry.UserData));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Hold CTRL and click to unpair permanently from " + entryUID);
|
||||
UiSharedService.AttachToolTip(L("UserPair.Menu.Unpair.Tooltip", "Hold CTRL and click to unpair permanently from {0}", entryUID));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Colors;
|
||||
using Dalamud.Interface.Utility;
|
||||
@@ -17,6 +17,8 @@ using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.UI.Components;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using MareSynchronos.WebAPI;
|
||||
using MareSynchronos.Localization;
|
||||
using System;
|
||||
using System.Globalization;
|
||||
using System.Numerics;
|
||||
|
||||
@@ -40,6 +42,7 @@ internal sealed class GroupPanel
|
||||
private string _editGroupComment = string.Empty;
|
||||
private string _editGroupEntry = string.Empty;
|
||||
private bool _errorGroupCreate = false;
|
||||
private string _errorGroupCreateMessage = string.Empty;
|
||||
private bool _errorGroupJoin;
|
||||
private bool _isPasswordValid;
|
||||
private GroupPasswordDto? _lastCreatedGroup = null;
|
||||
@@ -52,9 +55,31 @@ internal sealed class GroupPanel
|
||||
private bool _showModalChangePassword;
|
||||
private bool _showModalCreateGroup;
|
||||
private bool _showModalEnterPassword;
|
||||
private string _newSyncShellAlias = string.Empty;
|
||||
private bool _createIsTemporary = false;
|
||||
private int _tempSyncshellDurationHours = 24;
|
||||
private readonly int[] _temporaryDurationOptions = new[]
|
||||
{
|
||||
1,
|
||||
12,
|
||||
24,
|
||||
48,
|
||||
72,
|
||||
96,
|
||||
120,
|
||||
144,
|
||||
168
|
||||
};
|
||||
private string _syncShellPassword = string.Empty;
|
||||
private string _syncShellToJoin = string.Empty;
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public GroupPanel(CompactUi mainUi, UiSharedService uiShared, PairManager pairManager, ChatService chatServivce,
|
||||
UidDisplayHandler uidDisplayHandler, MareConfigService mareConfig, ServerConfigurationManager serverConfigurationManager,
|
||||
CharaDataManager charaDataManager)
|
||||
@@ -82,13 +107,15 @@ internal sealed class GroupPanel
|
||||
{
|
||||
var buttonSize = _uiShared.GetIconButtonSize(FontAwesomeIcon.Plus);
|
||||
ImGui.SetNextItemWidth(UiSharedService.GetWindowContentRegionWidth() - ImGui.GetWindowContentRegionMin().X - buttonSize.X);
|
||||
ImGui.InputTextWithHint("##syncshellid", "Syncshell GID/Alias (leave empty to create)", ref _syncShellToJoin, 20);
|
||||
ImGui.InputTextWithHint("##syncshellid", L("GroupPanel.Join.InputHint", "Syncshell GID/Alias (leave empty to create)"), ref _syncShellToJoin, 50);
|
||||
ImGui.SameLine(ImGui.GetWindowContentRegionMin().X + UiSharedService.GetWindowContentRegionWidth() - buttonSize.X);
|
||||
|
||||
bool userCanJoinMoreGroups = _pairManager.GroupPairs.Count < ApiController.ServerInfo.MaxGroupsJoinedByUser;
|
||||
bool userCanCreateMoreGroups = _pairManager.GroupPairs.Count(u => string.Equals(u.Key.Owner.UID, ApiController.UID, StringComparison.Ordinal)) < ApiController.ServerInfo.MaxGroupsCreatedByUser;
|
||||
bool alreadyInGroup = _pairManager.GroupPairs.Select(p => p.Key).Any(p => string.Equals(p.Group.Alias, _syncShellToJoin, StringComparison.Ordinal)
|
||||
|| string.Equals(p.Group.GID, _syncShellToJoin, StringComparison.Ordinal));
|
||||
var enterPasswordPopupTitle = L("GroupPanel.Join.PasswordPopup", "Enter Syncshell Password");
|
||||
var createPopupTitle = L("GroupPanel.Create.PopupTitle", "Create Syncshell");
|
||||
|
||||
if (alreadyInGroup) ImGui.BeginDisabled();
|
||||
if (_uiShared.IconButton(FontAwesomeIcon.Plus))
|
||||
@@ -99,7 +126,7 @@ internal sealed class GroupPanel
|
||||
{
|
||||
_errorGroupJoin = false;
|
||||
_showModalEnterPassword = true;
|
||||
ImGui.OpenPopup("Enter Syncshell Password");
|
||||
ImGui.OpenPopup(enterPasswordPopupTitle);
|
||||
}
|
||||
}
|
||||
else
|
||||
@@ -108,31 +135,42 @@ internal sealed class GroupPanel
|
||||
{
|
||||
_lastCreatedGroup = null;
|
||||
_errorGroupCreate = false;
|
||||
_newSyncShellAlias = string.Empty;
|
||||
_createIsTemporary = false;
|
||||
_tempSyncshellDurationHours = 24;
|
||||
_errorGroupCreateMessage = string.Empty;
|
||||
_showModalCreateGroup = true;
|
||||
ImGui.OpenPopup("Create Syncshell");
|
||||
ImGui.OpenPopup(createPopupTitle);
|
||||
}
|
||||
}
|
||||
}
|
||||
UiSharedService.AttachToolTip(_syncShellToJoin.IsNullOrEmpty()
|
||||
? (userCanCreateMoreGroups ? "Create Syncshell" : $"You cannot create more than {ApiController.ServerInfo.MaxGroupsCreatedByUser} Syncshells")
|
||||
: (userCanJoinMoreGroups ? "Join Syncshell" + _syncShellToJoin : $"You cannot join more than {ApiController.ServerInfo.MaxGroupsJoinedByUser} Syncshells"));
|
||||
? (userCanCreateMoreGroups
|
||||
? L("GroupPanel.Create.Tooltip", "Create Syncshell")
|
||||
: L("GroupPanel.Create.TooMany", "You cannot create more than {0} Syncshells", ApiController.ServerInfo.MaxGroupsCreatedByUser))
|
||||
: (userCanJoinMoreGroups
|
||||
? L("GroupPanel.Join.Tooltip", "Join Syncshell {0}", _syncShellToJoin)
|
||||
: L("GroupPanel.Join.TooMany", "You cannot join more than {0} Syncshells", ApiController.ServerInfo.MaxGroupsJoinedByUser)));
|
||||
|
||||
if (alreadyInGroup) ImGui.EndDisabled();
|
||||
|
||||
if (ImGui.BeginPopupModal("Enter Syncshell Password", ref _showModalEnterPassword, UiSharedService.PopupWindowFlags))
|
||||
if (ImGui.BeginPopupModal(enterPasswordPopupTitle, ref _showModalEnterPassword, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
UiSharedService.TextWrapped("Before joining any Syncshells please be aware that you will be automatically paired with everyone in the Syncshell.");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Join.Warning", "Before joining any Syncshells please be aware that you will be automatically paired with everyone in the Syncshell."));
|
||||
ImGui.Separator();
|
||||
UiSharedService.TextWrapped("Enter the password for Syncshell " + _syncShellToJoin + ":");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Join.EnterPassword", "Enter the password for Syncshell {0}:", _syncShellToJoin));
|
||||
ImGui.SetNextItemWidth(-1);
|
||||
ImGui.InputTextWithHint("##password", _syncShellToJoin + " Password", ref _syncShellPassword, 255, ImGuiInputTextFlags.Password);
|
||||
ImGui.InputTextWithHint("##password", L("GroupPanel.Join.PasswordHint", "{0} Password", _syncShellToJoin), ref _syncShellPassword, 255, ImGuiInputTextFlags.Password);
|
||||
if (_errorGroupJoin)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped($"An error occured during joining of this Syncshell: you either have joined the maximum amount of Syncshells ({ApiController.ServerInfo.MaxGroupsJoinedByUser}), " +
|
||||
$"it does not exist, the password you entered is wrong, you already joined the Syncshell, the Syncshell is full ({ApiController.ServerInfo.MaxGroupUserCount} users) or the Syncshell has closed invites.",
|
||||
UiSharedService.ColorTextWrapped(
|
||||
L("GroupPanel.Join.Error",
|
||||
"An error occured during joining of this Syncshell: you either have joined the maximum amount of Syncshells ({0}), it does not exist, the password you entered is wrong, you already joined the Syncshell, the Syncshell is full ({1} users) or the Syncshell has closed invites.",
|
||||
ApiController.ServerInfo.MaxGroupsJoinedByUser,
|
||||
ApiController.ServerInfo.MaxGroupUserCount),
|
||||
new Vector4(1, 0, 0, 1));
|
||||
}
|
||||
if (ImGui.Button("Join " + _syncShellToJoin))
|
||||
if (ImGui.Button(L("GroupPanel.Join.Button", "Join {0}", _syncShellToJoin)))
|
||||
{
|
||||
var shell = _syncShellToJoin;
|
||||
var pw = _syncShellPassword;
|
||||
@@ -148,20 +186,100 @@ internal sealed class GroupPanel
|
||||
ImGui.EndPopup();
|
||||
}
|
||||
|
||||
if (ImGui.BeginPopupModal("Create Syncshell", ref _showModalCreateGroup, UiSharedService.PopupWindowFlags))
|
||||
if (ImGui.BeginPopupModal(createPopupTitle, ref _showModalCreateGroup, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
UiSharedService.TextWrapped("Press the button below to create a new Syncshell.");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.ChooseType", "Choisissez le type de Syncshell à créer."));
|
||||
bool showPermanent = !_createIsTemporary;
|
||||
if (ImGui.RadioButton(L("GroupPanel.Create.Permanent", "Permanente"), showPermanent))
|
||||
{
|
||||
_createIsTemporary = false;
|
||||
}
|
||||
ImGui.SameLine();
|
||||
if (ImGui.RadioButton(L("GroupPanel.Create.Temporary", "Temporaire"), _createIsTemporary))
|
||||
{
|
||||
_createIsTemporary = true;
|
||||
_newSyncShellAlias = string.Empty;
|
||||
}
|
||||
|
||||
if (!_createIsTemporary)
|
||||
{
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.AliasPrompt", "Donnez un nom à votre Syncshell (optionnel) puis créez-la."));
|
||||
ImGui.SetNextItemWidth(-1);
|
||||
ImGui.InputTextWithHint("##syncshellalias", L("GroupPanel.Create.AliasHint", "Nom du Syncshell"), ref _newSyncShellAlias, 50);
|
||||
}
|
||||
else
|
||||
{
|
||||
_newSyncShellAlias = string.Empty;
|
||||
}
|
||||
|
||||
if (_createIsTemporary)
|
||||
{
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.TempMaxDuration", "Durée maximale d'une Syncshell temporaire : 7 jours."));
|
||||
if (_tempSyncshellDurationHours > 168) _tempSyncshellDurationHours = 168;
|
||||
for (int i = 0; i < _temporaryDurationOptions.Length; i++)
|
||||
{
|
||||
var option = _temporaryDurationOptions[i];
|
||||
var isSelected = _tempSyncshellDurationHours == option;
|
||||
string label = option switch
|
||||
{
|
||||
>= 24 when option % 24 == 0 => option == 24
|
||||
? L("GroupPanel.Create.Duration.SingleDay", "24h")
|
||||
: L("GroupPanel.Create.Duration.Days", "{0}j", option / 24),
|
||||
_ => L("GroupPanel.Create.Duration.Hours", "{0}h", option)
|
||||
};
|
||||
|
||||
if (ImGui.RadioButton(label, isSelected))
|
||||
{
|
||||
_tempSyncshellDurationHours = option;
|
||||
}
|
||||
|
||||
if ((i + 1) % 3 == 0)
|
||||
{
|
||||
ImGui.NewLine();
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.SameLine();
|
||||
}
|
||||
}
|
||||
|
||||
var expiresLocal = DateTime.Now.AddHours(_tempSyncshellDurationHours);
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.TempExpires", "Expiration le {0:g} (heure locale).", expiresLocal));
|
||||
}
|
||||
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.Instruction", "Appuyez sur le bouton ci-dessous pour créer une nouvelle Syncshell."));
|
||||
ImGui.SetNextItemWidth(200 * ImGuiHelpers.GlobalScale);
|
||||
if (ImGui.Button("Create Syncshell"))
|
||||
if (ImGui.Button(L("GroupPanel.Create.Button", "Create Syncshell")))
|
||||
{
|
||||
try
|
||||
{
|
||||
_lastCreatedGroup = ApiController.GroupCreate().Result;
|
||||
if (_createIsTemporary)
|
||||
{
|
||||
var expiresAtUtc = DateTime.UtcNow.AddHours(_tempSyncshellDurationHours);
|
||||
_lastCreatedGroup = ApiController.GroupCreateTemporary(expiresAtUtc).Result;
|
||||
}
|
||||
else
|
||||
{
|
||||
var aliasInput = string.IsNullOrWhiteSpace(_newSyncShellAlias) ? null : _newSyncShellAlias.Trim();
|
||||
_lastCreatedGroup = ApiController.GroupCreate(aliasInput).Result;
|
||||
if (_lastCreatedGroup != null)
|
||||
{
|
||||
_newSyncShellAlias = string.Empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
catch
|
||||
catch (Exception ex)
|
||||
{
|
||||
_lastCreatedGroup = null;
|
||||
_errorGroupCreate = true;
|
||||
if (ex.Message.Contains("name is already in use", StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
_errorGroupCreateMessage = L("GroupPanel.Create.Error.NameInUse", "Le nom de la Syncshell est déjà utilisé.");
|
||||
}
|
||||
else
|
||||
{
|
||||
_errorGroupCreateMessage = ex.Message;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -169,21 +287,33 @@ internal sealed class GroupPanel
|
||||
{
|
||||
ImGui.Separator();
|
||||
_errorGroupCreate = false;
|
||||
ImGui.TextUnformatted("Syncshell ID: " + _lastCreatedGroup.Group.GID);
|
||||
_errorGroupCreateMessage = string.Empty;
|
||||
if (!string.IsNullOrWhiteSpace(_lastCreatedGroup.Group.Alias))
|
||||
{
|
||||
ImGui.TextUnformatted(L("GroupPanel.Create.Result.Name", "Syncshell Name: {0}", _lastCreatedGroup.Group.Alias));
|
||||
}
|
||||
ImGui.TextUnformatted(L("GroupPanel.Create.Result.Id", "Syncshell ID: {0}", _lastCreatedGroup.Group.GID));
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted("Syncshell Password: " + _lastCreatedGroup.Password);
|
||||
ImGui.TextUnformatted(L("GroupPanel.Create.Result.Password", "Syncshell Password: {0}", _lastCreatedGroup.Password));
|
||||
ImGui.SameLine();
|
||||
if (_uiShared.IconButton(FontAwesomeIcon.Copy))
|
||||
{
|
||||
ImGui.SetClipboardText(_lastCreatedGroup.Password);
|
||||
}
|
||||
UiSharedService.TextWrapped("You can change the Syncshell password later at any time.");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.Result.ChangeLater", "You can change the Syncshell password later at any time."));
|
||||
if (_lastCreatedGroup.IsTemporary && _lastCreatedGroup.ExpiresAt != null)
|
||||
{
|
||||
var expiresLocal = _lastCreatedGroup.ExpiresAt.Value.ToLocalTime();
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Create.Result.TempExpires", "Cette Syncshell expirera le {0:g} (heure locale).", expiresLocal));
|
||||
}
|
||||
}
|
||||
|
||||
if (_errorGroupCreate)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("You are already owner of the maximum amount of Syncshells (3) or joined the maximum amount of Syncshells (6). Relinquish ownership of your own Syncshells to someone else or leave existing Syncshells.",
|
||||
new Vector4(1, 0, 0, 1));
|
||||
var msg = string.IsNullOrWhiteSpace(_errorGroupCreateMessage)
|
||||
? L("GroupPanel.Create.Error.Generic", "You are already owner of the maximum amount of Syncshells (3) or joined the maximum amount of Syncshells (6). Relinquish ownership of your own Syncshells to someone else or leave existing Syncshells.")
|
||||
: _errorGroupCreateMessage;
|
||||
UiSharedService.ColorTextWrapped(msg, new Vector4(1, 0, 0, 1));
|
||||
}
|
||||
|
||||
UiSharedService.SetScaledWindowSize(350);
|
||||
@@ -220,7 +350,7 @@ internal sealed class GroupPanel
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.Crown.ToIconString());
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("You are the owner of Syncshell " + groupName);
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Syncshell.OwnerTooltip", "You are the owner of Syncshell {0}", groupName));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
else if (groupDto.GroupUserInfo.IsModerator())
|
||||
@@ -228,7 +358,7 @@ internal sealed class GroupPanel
|
||||
ImGui.PushFont(UiBuilder.IconFont);
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.UserShield.ToIconString());
|
||||
ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("You are a moderator of Syncshell " + groupName);
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Syncshell.ModeratorTooltip", "You are a moderator of Syncshell {0}", groupName));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
|
||||
@@ -243,18 +373,38 @@ internal sealed class GroupPanel
|
||||
if (!string.Equals(_editGroupEntry, groupDto.GID, StringComparison.Ordinal))
|
||||
{
|
||||
var shellConfig = _serverConfigurationManager.GetShellConfigForGid(groupDto.GID);
|
||||
if (!_mareConfig.Current.DisableSyncshellChat && shellConfig.Enabled)
|
||||
{
|
||||
ImGui.TextUnformatted($"[{shellNumber}]");
|
||||
UiSharedService.AttachToolTip("Chat command prefix: /ss" + shellNumber);
|
||||
}
|
||||
var totalMembers = pairsInGroup.Count + 1;
|
||||
var connectedMembers = pairsInGroup.Count(p => p.IsOnline) + 1;
|
||||
var maxCapacity = ApiController.ServerInfo.MaxGroupUserCount;
|
||||
ImGui.TextUnformatted(L("GroupPanel.Syncshell.MemberCount", "{0}/{1}", connectedMembers, totalMembers));
|
||||
UiSharedService.AttachToolTip(
|
||||
L("GroupPanel.Syncshell.MemberCountTooltip",
|
||||
"Membres connectés / membres totaux\nCapacité maximale : {0}\nSyncshell ID: {1}",
|
||||
maxCapacity,
|
||||
groupDto.Group.GID));
|
||||
if (textIsGid) ImGui.PushFont(UiBuilder.MonoFont);
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(groupName);
|
||||
if (textIsGid) ImGui.PopFont();
|
||||
UiSharedService.AttachToolTip("Left click to switch between GID display and comment" + Environment.NewLine +
|
||||
"Right click to change comment for " + groupName + Environment.NewLine + Environment.NewLine
|
||||
+ "Users: " + (pairsInGroup.Count + 1) + ", Owner: " + groupDto.OwnerAliasOrUID);
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Syncshell.NameTooltip",
|
||||
"Left click to switch between GID display and comment\nRight click to change comment for {0}\n\nUsers: {1}, Owner: {2}",
|
||||
groupName,
|
||||
pairsInGroup.Count + 1,
|
||||
groupDto.OwnerAliasOrUID));
|
||||
if (groupDto.IsTemporary)
|
||||
{
|
||||
ImGui.SameLine();
|
||||
UiSharedService.ColorText(L("GroupPanel.Syncshell.TempTag", "(Temp)"), ImGuiColors.DalamudOrange);
|
||||
if (groupDto.ExpiresAt != null)
|
||||
{
|
||||
var tempExpireLocal = groupDto.ExpiresAt.Value.ToLocalTime();
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Syncshell.TempExpires", "Expire le {0:g}", tempExpireLocal));
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Syncshell.TempTooltip", "Syncshell temporaire"));
|
||||
}
|
||||
}
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left))
|
||||
{
|
||||
var prevState = textIsGid;
|
||||
@@ -278,7 +428,7 @@ internal sealed class GroupPanel
|
||||
{
|
||||
var buttonSizes = _uiShared.GetIconButtonSize(FontAwesomeIcon.Bars).X + _uiShared.GetIconButtonSize(FontAwesomeIcon.LockOpen).X;
|
||||
ImGui.SetNextItemWidth(UiSharedService.GetWindowContentRegionWidth() - ImGui.GetCursorPosX() - buttonSizes - ImGui.GetStyle().ItemSpacing.X * 2);
|
||||
if (ImGui.InputTextWithHint("", "Comment/Notes", ref _editGroupComment, 255, ImGuiInputTextFlags.EnterReturnsTrue))
|
||||
if (ImGui.InputTextWithHint("", L("GroupPanel.CommentHint", "Comment/Notes"), ref _editGroupComment, 255, ImGuiInputTextFlags.EnterReturnsTrue))
|
||||
{
|
||||
_serverConfigurationManager.SetNoteForGid(groupDto.GID, _editGroupComment);
|
||||
_editGroupEntry = string.Empty;
|
||||
@@ -289,7 +439,7 @@ internal sealed class GroupPanel
|
||||
{
|
||||
_editGroupEntry = string.Empty;
|
||||
}
|
||||
UiSharedService.AttachToolTip("Hit ENTER to save\nRight click to cancel");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.CommentTooltip", "Hit ENTER to save\nRight click to cancel"));
|
||||
}
|
||||
|
||||
|
||||
@@ -298,26 +448,26 @@ internal sealed class GroupPanel
|
||||
if (_showModalBanList && !_modalBanListOpened)
|
||||
{
|
||||
_modalBanListOpened = true;
|
||||
ImGui.OpenPopup("Manage Banlist for " + groupDto.GID);
|
||||
ImGui.OpenPopup(L("GroupPanel.Banlist.Title", "Manage Banlist for {0}", groupDto.GID));
|
||||
}
|
||||
|
||||
if (!_showModalBanList) _modalBanListOpened = false;
|
||||
|
||||
if (ImGui.BeginPopupModal("Manage Banlist for " + groupDto.GID, ref _showModalBanList, UiSharedService.PopupWindowFlags))
|
||||
if (ImGui.BeginPopupModal(L("GroupPanel.Banlist.Title", "Manage Banlist for {0}", groupDto.GID), ref _showModalBanList, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Retweet, "Refresh Banlist from Server"))
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Retweet, L("GroupPanel.Banlist.Refresh", "Refresh Banlist from Server")))
|
||||
{
|
||||
_bannedUsers = ApiController.GroupGetBannedUsers(groupDto).Result;
|
||||
}
|
||||
|
||||
if (ImGui.BeginTable("bannedusertable" + groupDto.GID, 6, ImGuiTableFlags.RowBg | ImGuiTableFlags.SizingStretchProp | ImGuiTableFlags.ScrollY))
|
||||
{
|
||||
ImGui.TableSetupColumn("UID", ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn("Alias", ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn("By", ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn("Date", ImGuiTableColumnFlags.None, 2);
|
||||
ImGui.TableSetupColumn("Reason", ImGuiTableColumnFlags.None, 3);
|
||||
ImGui.TableSetupColumn("Actions", ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.Uid", "UID"), ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.Alias", "Alias"), ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.By", "By"), ImGuiTableColumnFlags.None, 1);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.Date", "Date"), ImGuiTableColumnFlags.None, 2);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.Reason", "Reason"), ImGuiTableColumnFlags.None, 3);
|
||||
ImGui.TableSetupColumn(L("GroupPanel.Banlist.Column.Actions", "Actions"), ImGuiTableColumnFlags.None, 1);
|
||||
|
||||
ImGui.TableHeadersRow();
|
||||
|
||||
@@ -334,7 +484,7 @@ internal sealed class GroupPanel
|
||||
ImGui.TableNextColumn();
|
||||
UiSharedService.TextWrapped(bannedUser.Reason);
|
||||
ImGui.TableNextColumn();
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Check, "Unban#" + bannedUser.UID))
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Check, L("GroupPanel.Banlist.Unban", "Unban") + "#" + bannedUser.UID))
|
||||
{
|
||||
_ = ApiController.GroupUnbanUser(bannedUser);
|
||||
_bannedUsers.RemoveAll(b => string.Equals(b.UID, bannedUser.UID, StringComparison.Ordinal));
|
||||
@@ -350,18 +500,18 @@ internal sealed class GroupPanel
|
||||
if (_showModalChangePassword && !_modalChangePwOpened)
|
||||
{
|
||||
_modalChangePwOpened = true;
|
||||
ImGui.OpenPopup("Change Syncshell Password");
|
||||
ImGui.OpenPopup(L("GroupPanel.Password.Title", "Change Syncshell Password"));
|
||||
}
|
||||
|
||||
if (!_showModalChangePassword) _modalChangePwOpened = false;
|
||||
|
||||
if (ImGui.BeginPopupModal("Change Syncshell Password", ref _showModalChangePassword, UiSharedService.PopupWindowFlags))
|
||||
if (ImGui.BeginPopupModal(L("GroupPanel.Password.Title", "Change Syncshell Password"), ref _showModalChangePassword, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
UiSharedService.TextWrapped("Enter the new Syncshell password for Syncshell " + name + " here.");
|
||||
UiSharedService.TextWrapped("This action is irreversible");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Password.Description", "Enter the new Syncshell password for Syncshell {0} here.", name));
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Password.Warning", "This action is irreversible"));
|
||||
ImGui.SetNextItemWidth(-1);
|
||||
ImGui.InputTextWithHint("##changepw", "New password for " + name, ref _newSyncShellPassword, 255);
|
||||
if (ImGui.Button("Change password"))
|
||||
ImGui.InputTextWithHint("##changepw", L("GroupPanel.Password.Hint", "New password for {0}", name), ref _newSyncShellPassword, 255);
|
||||
if (ImGui.Button(L("GroupPanel.Password.Button", "Change password")))
|
||||
{
|
||||
var pw = _newSyncShellPassword;
|
||||
_isPasswordValid = ApiController.GroupChangePassword(new(groupDto.Group, pw)).Result;
|
||||
@@ -371,7 +521,7 @@ internal sealed class GroupPanel
|
||||
|
||||
if (!_isPasswordValid)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("The selected password is too short. It must be at least 10 characters.", new Vector4(1, 0, 0, 1));
|
||||
UiSharedService.ColorTextWrapped(L("GroupPanel.Password.Error.TooShort", "The selected password is too short. It must be at least 10 characters."), new Vector4(1, 0, 0, 1));
|
||||
}
|
||||
|
||||
UiSharedService.SetScaledWindowSize(290);
|
||||
@@ -381,29 +531,28 @@ internal sealed class GroupPanel
|
||||
if (_showModalBulkOneTimeInvites && !_modalBulkOneTimeInvitesOpened)
|
||||
{
|
||||
_modalBulkOneTimeInvitesOpened = true;
|
||||
ImGui.OpenPopup("Create Bulk One-Time Invites");
|
||||
ImGui.OpenPopup(L("GroupPanel.Invites.Title", "Create Bulk One-Time Invites"));
|
||||
}
|
||||
|
||||
if (!_showModalBulkOneTimeInvites) _modalBulkOneTimeInvitesOpened = false;
|
||||
|
||||
if (ImGui.BeginPopupModal("Create Bulk One-Time Invites", ref _showModalBulkOneTimeInvites, UiSharedService.PopupWindowFlags))
|
||||
if (ImGui.BeginPopupModal(L("GroupPanel.Invites.Title", "Create Bulk One-Time Invites"), ref _showModalBulkOneTimeInvites, UiSharedService.PopupWindowFlags))
|
||||
{
|
||||
UiSharedService.TextWrapped("This allows you to create up to 100 one-time invites at once for the Syncshell " + name + "." + Environment.NewLine
|
||||
+ "The invites are valid for 24h after creation and will automatically expire.");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Invites.Description", "This allows you to create up to 100 one-time invites at once for the Syncshell {0}.\nThe invites are valid for 24h after creation and will automatically expire.", name));
|
||||
ImGui.Separator();
|
||||
if (_bulkOneTimeInvites.Count == 0)
|
||||
{
|
||||
ImGui.SetNextItemWidth(-1);
|
||||
ImGui.SliderInt("Amount##bulkinvites", ref _bulkInviteCount, 1, 100);
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.MailBulk, "Create invites"))
|
||||
ImGui.SliderInt(L("GroupPanel.Invites.AmountLabel", "Amount") + "##bulkinvites", ref _bulkInviteCount, 1, 100);
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.MailBulk, L("GroupPanel.Invites.CreateButton", "Create invites")))
|
||||
{
|
||||
_bulkOneTimeInvites = ApiController.GroupCreateTempInvite(groupDto, _bulkInviteCount).Result;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.TextWrapped("A total of " + _bulkOneTimeInvites.Count + " invites have been created.");
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Copy, "Copy invites to clipboard"))
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Invites.Result", "A total of {0} invites have been created.", _bulkOneTimeInvites.Count));
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Copy, L("GroupPanel.Invites.Copy", "Copy invites to clipboard")))
|
||||
{
|
||||
ImGui.SetClipboardText(string.Join(Environment.NewLine, _bulkOneTimeInvites));
|
||||
}
|
||||
@@ -450,25 +599,27 @@ internal sealed class GroupPanel
|
||||
|
||||
if (visibleUsers.Count > 0)
|
||||
{
|
||||
ImGui.TextUnformatted("Visible");
|
||||
ImGui.TextUnformatted(L("GroupPanel.List.Visible", "Visible"));
|
||||
ImGui.Separator();
|
||||
_uidDisplayHandler.RenderPairList(visibleUsers);
|
||||
}
|
||||
|
||||
if (onlineUsers.Count > 0)
|
||||
{
|
||||
ImGui.TextUnformatted("Online");
|
||||
ImGui.TextUnformatted(L("GroupPanel.List.Online", "Online"));
|
||||
ImGui.Separator();
|
||||
_uidDisplayHandler.RenderPairList(onlineUsers);
|
||||
}
|
||||
|
||||
if (offlineUsers.Count > 0)
|
||||
{
|
||||
ImGui.TextUnformatted("Offline/Unknown");
|
||||
ImGui.PushStyleColor(ImGuiCol.Text, ImGuiColors.DalamudGrey);
|
||||
ImGui.TextUnformatted(L("GroupPanel.List.Offline", "Offline/Unknown"));
|
||||
ImGui.PopStyleColor();
|
||||
ImGui.Separator();
|
||||
if (hideOfflineUsers)
|
||||
{
|
||||
UiSharedService.ColorText($" {offlineUsers.Count} offline users omitted from display.", ImGuiColors.DalamudGrey);
|
||||
UiSharedService.ColorText(L("GroupPanel.List.OfflineOmitted", " {0} offline users omitted from display.", offlineUsers.Count), ImGuiColors.DalamudGrey);
|
||||
}
|
||||
else
|
||||
{
|
||||
@@ -523,11 +674,11 @@ internal sealed class GroupPanel
|
||||
ImGui.BeginTooltip();
|
||||
if (!invitesEnabled || soundsDisabled || animDisabled || vfxDisabled)
|
||||
{
|
||||
ImGui.TextUnformatted("Syncshell permissions");
|
||||
ImGui.TextUnformatted(L("GroupPanel.Permissions.Header", "Syncshell permissions"));
|
||||
|
||||
if (!invitesEnabled)
|
||||
{
|
||||
var lockedText = "Syncshell is closed for joining";
|
||||
var lockedText = L("GroupPanel.Permissions.InvitesDisabled", "Syncshell is closed for joining");
|
||||
_uiShared.IconText(lockedIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(lockedText);
|
||||
@@ -535,7 +686,7 @@ internal sealed class GroupPanel
|
||||
|
||||
if (soundsDisabled)
|
||||
{
|
||||
var soundsText = "Sound sync disabled through owner";
|
||||
var soundsText = L("GroupPanel.Permissions.SoundDisabledOwner", "Sound sync disabled through owner");
|
||||
_uiShared.IconText(soundsIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(soundsText);
|
||||
@@ -543,7 +694,7 @@ internal sealed class GroupPanel
|
||||
|
||||
if (animDisabled)
|
||||
{
|
||||
var animText = "Animation sync disabled through owner";
|
||||
var animText = L("GroupPanel.Permissions.AnimationDisabledOwner", "Animation sync disabled through owner");
|
||||
_uiShared.IconText(animIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(animText);
|
||||
@@ -551,7 +702,7 @@ internal sealed class GroupPanel
|
||||
|
||||
if (vfxDisabled)
|
||||
{
|
||||
var vfxText = "VFX sync disabled through owner";
|
||||
var vfxText = L("GroupPanel.Permissions.VfxDisabledOwner", "VFX sync disabled through owner");
|
||||
_uiShared.IconText(vfxIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(vfxText);
|
||||
@@ -563,11 +714,11 @@ internal sealed class GroupPanel
|
||||
if (!invitesEnabled || soundsDisabled || animDisabled || vfxDisabled)
|
||||
ImGui.Separator();
|
||||
|
||||
ImGui.TextUnformatted("Your permissions");
|
||||
ImGui.TextUnformatted(L("GroupPanel.Permissions.OwnHeader", "Your permissions"));
|
||||
|
||||
if (userSoundsDisabled)
|
||||
{
|
||||
var userSoundsText = "Sound sync disabled through you";
|
||||
var userSoundsText = L("GroupPanel.Permissions.SoundDisabledSelf", "Sound sync disabled through you");
|
||||
_uiShared.IconText(userSoundsIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userSoundsText);
|
||||
@@ -575,7 +726,7 @@ internal sealed class GroupPanel
|
||||
|
||||
if (userAnimDisabled)
|
||||
{
|
||||
var userAnimText = "Animation sync disabled through you";
|
||||
var userAnimText = L("GroupPanel.Permissions.AnimationDisabledSelf", "Animation sync disabled through you");
|
||||
_uiShared.IconText(userAnimIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userAnimText);
|
||||
@@ -583,14 +734,14 @@ internal sealed class GroupPanel
|
||||
|
||||
if (userVFXDisabled)
|
||||
{
|
||||
var userVFXText = "VFX sync disabled through you";
|
||||
var userVFXText = L("GroupPanel.Permissions.VfxDisabledSelf", "VFX sync disabled through you");
|
||||
_uiShared.IconText(userVFXIcon);
|
||||
ImGui.SameLine(40 * ImGuiHelpers.GlobalScale);
|
||||
ImGui.TextUnformatted(userVFXText);
|
||||
}
|
||||
|
||||
if (!invitesEnabled || soundsDisabled || animDisabled || vfxDisabled)
|
||||
UiSharedService.TextWrapped("Note that syncshell permissions for disabling take precedence over your own set permissions");
|
||||
UiSharedService.TextWrapped(L("GroupPanel.Permissions.NotePriority", "Note that syncshell permissions for disabling take precedence over your own set permissions"));
|
||||
}
|
||||
ImGui.EndTooltip();
|
||||
}
|
||||
@@ -602,7 +753,8 @@ internal sealed class GroupPanel
|
||||
var userPerm = groupDto.GroupUserPermissions ^ GroupUserPermissions.Paused;
|
||||
_ = ApiController.GroupChangeIndividualPermissionState(new GroupPairUserPermissionDto(groupDto.Group, new UserData(ApiController.UID), userPerm));
|
||||
}
|
||||
UiSharedService.AttachToolTip((groupDto.GroupUserPermissions.IsPaused() ? "Resume" : "Pause") + " pairing with all users in this Syncshell");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.PauseToggle.Tooltip", "{0} pairing with all users in this Syncshell",
|
||||
groupDto.GroupUserPermissions.IsPaused() ? L("GroupPanel.PauseToggle.Resume", "Resume") : L("GroupPanel.PauseToggle.Pause", "Pause")));
|
||||
ImGui.SameLine();
|
||||
|
||||
if (_uiShared.IconButton(FontAwesomeIcon.Bars))
|
||||
@@ -612,28 +764,32 @@ internal sealed class GroupPanel
|
||||
|
||||
if (ImGui.BeginPopup("ShellPopup"))
|
||||
{
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.ArrowCircleLeft, "Leave Syncshell") && UiSharedService.CtrlPressed())
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.ArrowCircleLeft, L("GroupPanel.Popup.Leave", "Leave Syncshell")) && UiSharedService.CtrlPressed())
|
||||
{
|
||||
_ = ApiController.GroupLeave(groupDto);
|
||||
}
|
||||
UiSharedService.AttachToolTip("Hold CTRL and click to leave this Syncshell" + (!string.Equals(groupDto.OwnerUID, ApiController.UID, StringComparison.Ordinal) ? string.Empty : Environment.NewLine
|
||||
+ "WARNING: This action is irreversible" + Environment.NewLine + "Leaving an owned Syncshell will transfer the ownership to a random person in the Syncshell."));
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.LeaveTooltip", "Hold CTRL and click to leave this Syncshell{0}",
|
||||
!string.Equals(groupDto.OwnerUID, ApiController.UID, StringComparison.Ordinal)
|
||||
? string.Empty
|
||||
: Environment.NewLine + L("GroupPanel.Popup.LeaveWarning", "WARNING: This action is irreversible\nLeaving an owned Syncshell will transfer the ownership to a random person in the Syncshell.")));
|
||||
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Copy, "Copy ID"))
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Copy, L("GroupPanel.Popup.CopyId", "Copy ID")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
ImGui.SetClipboardText(groupDto.GroupAliasOrGID);
|
||||
}
|
||||
UiSharedService.AttachToolTip("Copy Syncshell ID to Clipboard");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.CopyIdTooltip", "Copy Syncshell ID to Clipboard"));
|
||||
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.StickyNote, "Copy Notes"))
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.StickyNote, L("GroupPanel.Popup.CopyNotes", "Copy Notes")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
ImGui.SetClipboardText(UiSharedService.GetNotes(groupPairs));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Copies all your notes for all users in this Syncshell to the clipboard." + Environment.NewLine + "They can be imported via Settings -> General -> Notes -> Import notes from clipboard");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.CopyNotesTooltip", "Copies all your notes for all users in this Syncshell to the clipboard.\nThey can be imported via Settings -> General -> Notes -> Import notes from clipboard"));
|
||||
|
||||
var soundsText = userSoundsDisabled ? "Enable sound sync" : "Disable sound sync";
|
||||
var soundsText = userSoundsDisabled
|
||||
? L("GroupPanel.Popup.EnableSound", "Enable sound sync")
|
||||
: L("GroupPanel.Popup.DisableSound", "Disable sound sync");
|
||||
if (_uiShared.IconTextButton(userSoundsIcon, soundsText))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
@@ -641,12 +797,11 @@ internal sealed class GroupPanel
|
||||
perm.SetDisableSounds(!perm.IsDisableSounds());
|
||||
_ = ApiController.GroupChangeIndividualPermissionState(new(groupDto.Group, new UserData(ApiController.UID), perm));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Sets your allowance for sound synchronization for users of this syncshell."
|
||||
+ Environment.NewLine + "Disabling the synchronization will stop applying sound modifications for users of this syncshell."
|
||||
+ Environment.NewLine + "Note: this setting can be forcefully overridden to 'disabled' through the syncshell owner."
|
||||
+ Environment.NewLine + "Note: this setting does not apply to individual pairs that are also in the syncshell.");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.SoundTooltip", "Sets your allowance for sound synchronization for users of this syncshell.\nDisabling the synchronization will stop applying sound modifications for users of this syncshell.\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\nNote: this setting does not apply to individual pairs that are also in the syncshell."));
|
||||
|
||||
var animText = userAnimDisabled ? "Enable animations sync" : "Disable animations sync";
|
||||
var animText = userAnimDisabled
|
||||
? L("GroupPanel.Popup.EnableAnimations", "Enable animations sync")
|
||||
: L("GroupPanel.Popup.DisableAnimations", "Disable animations sync");
|
||||
if (_uiShared.IconTextButton(userAnimIcon, animText))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
@@ -654,13 +809,11 @@ internal sealed class GroupPanel
|
||||
perm.SetDisableAnimations(!perm.IsDisableAnimations());
|
||||
_ = ApiController.GroupChangeIndividualPermissionState(new(groupDto.Group, new UserData(ApiController.UID), perm));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Sets your allowance for animations synchronization for users of this syncshell."
|
||||
+ Environment.NewLine + "Disabling the synchronization will stop applying animations modifications for users of this syncshell."
|
||||
+ Environment.NewLine + "Note: this setting might also affect sound synchronization"
|
||||
+ Environment.NewLine + "Note: this setting can be forcefully overridden to 'disabled' through the syncshell owner."
|
||||
+ Environment.NewLine + "Note: this setting does not apply to individual pairs that are also in the syncshell.");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.AnimTooltip", "Sets your allowance for animations synchronization for users of this syncshell.\nDisabling the synchronization will stop applying animations modifications for users of this syncshell.\nNote: this setting might also affect sound synchronization\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\nNote: this setting does not apply to individual pairs that are also in the syncshell."));
|
||||
|
||||
var vfxText = userVFXDisabled ? "Enable VFX sync" : "Disable VFX sync";
|
||||
var vfxText = userVFXDisabled
|
||||
? L("GroupPanel.Popup.EnableVfx", "Enable VFX sync")
|
||||
: L("GroupPanel.Popup.DisableVfx", "Disable VFX sync");
|
||||
if (_uiShared.IconTextButton(userVFXIcon, vfxText))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
@@ -668,16 +821,12 @@ internal sealed class GroupPanel
|
||||
perm.SetDisableVFX(!perm.IsDisableVFX());
|
||||
_ = ApiController.GroupChangeIndividualPermissionState(new(groupDto.Group, new UserData(ApiController.UID), perm));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Sets your allowance for VFX synchronization for users of this syncshell."
|
||||
+ Environment.NewLine + "Disabling the synchronization will stop applying VFX modifications for users of this syncshell."
|
||||
+ Environment.NewLine + "Note: this setting might also affect animation synchronization to some degree"
|
||||
+ Environment.NewLine + "Note: this setting can be forcefully overridden to 'disabled' through the syncshell owner."
|
||||
+ Environment.NewLine + "Note: this setting does not apply to individual pairs that are also in the syncshell.");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.Popup.VfxTooltip", "Sets your allowance for VFX synchronization for users of this syncshell.\nDisabling the synchronization will stop applying VFX modifications for users of this syncshell.\nNote: this setting might also affect animation synchronization to some degree\nNote: this setting can be forcefully overridden to 'disabled' through the syncshell owner.\nNote: this setting does not apply to individual pairs that are also in the syncshell."));
|
||||
|
||||
if (isOwner || groupDto.GroupUserInfo.IsModerator())
|
||||
{
|
||||
ImGui.Separator();
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Cog, "Open Admin Panel"))
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Cog, L("GroupPanel.Popup.OpenAdmin", "Open Admin Panel")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
_mainUi.Mediator.Publish(new OpenSyncshellAdminPanel(groupDto));
|
||||
@@ -700,4 +849,4 @@ internal sealed class GroupPanel
|
||||
}
|
||||
ImGui.EndChild();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,10 +1,12 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using System;
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Utility.Raii;
|
||||
using MareSynchronos.API.Data.Extensions;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using MareSynchronos.WebAPI;
|
||||
using MareSynchronos.Localization;
|
||||
|
||||
namespace MareSynchronos.UI.Components;
|
||||
|
||||
@@ -17,6 +19,13 @@ public class PairGroupsUi
|
||||
private readonly UidDisplayHandler _uidDisplayHandler;
|
||||
private readonly UiSharedService _uiSharedService;
|
||||
|
||||
private string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(System.Globalization.CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public PairGroupsUi(MareConfigService mareConfig, TagHandler tagHandler, UidDisplayHandler uidDisplayHandler, ApiController apiController,
|
||||
SelectPairForGroupUi selectGroupForPairUi, UiSharedService uiSharedService)
|
||||
{
|
||||
@@ -54,37 +63,32 @@ public class PairGroupsUi
|
||||
ImGui.SameLine(buttonPauseOffset);
|
||||
if (_uiSharedService.IconButton(pauseButton))
|
||||
{
|
||||
// If all of the currently visible pairs (after applying filters to the pairs)
|
||||
// are paused we display a resume button to resume all currently visible (after filters)
|
||||
// pairs. Otherwise, we just pause all the remaining pairs.
|
||||
if (allArePaused)
|
||||
{
|
||||
// If all are paused => resume all
|
||||
ResumeAllPairs(availablePairsInThisTag);
|
||||
}
|
||||
else
|
||||
{
|
||||
// otherwise pause all remaining
|
||||
PauseRemainingPairs(availablePairsInThisTag);
|
||||
}
|
||||
}
|
||||
if (allArePaused)
|
||||
{
|
||||
UiSharedService.AttachToolTip($"Resume pairing with all pairs in {tag}");
|
||||
UiSharedService.AttachToolTip(L("PairGroups.ResumeAll", "Resume pairing with all pairs in {0}", tag));
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.AttachToolTip($"Pause pairing with all pairs in {tag}");
|
||||
UiSharedService.AttachToolTip(L("PairGroups.PauseAll", "Pause pairing with all pairs in {0}", tag));
|
||||
}
|
||||
|
||||
var buttonDeleteOffset = windowX + windowWidth - flyoutMenuX;
|
||||
ImGui.SameLine(buttonDeleteOffset);
|
||||
if (_uiSharedService.IconButton(FontAwesomeIcon.Bars))
|
||||
{
|
||||
ImGui.OpenPopup("Group Flyout Menu");
|
||||
ImGui.OpenPopup(L("PairGroups.Menu.Title", "Group Flyout Menu"));
|
||||
}
|
||||
|
||||
if (ImGui.BeginPopup("Group Flyout Menu"))
|
||||
if (ImGui.BeginPopup(L("PairGroups.Menu.Title", "Group Flyout Menu")))
|
||||
{
|
||||
using (ImRaii.PushId($"buttons-{tag}")) DrawGroupMenu(tag);
|
||||
ImGui.EndPopup();
|
||||
@@ -120,7 +124,6 @@ public class PairGroupsUi
|
||||
}
|
||||
else
|
||||
{
|
||||
// Avoid uncomfortably close group names
|
||||
if (!_tagHandler.IsTagOpen(tag))
|
||||
{
|
||||
var size = ImGui.CalcTextSize("").Y + ImGui.GetStyle().FramePadding.Y * 2f;
|
||||
@@ -138,33 +141,35 @@ public class PairGroupsUi
|
||||
|
||||
private void DrawGroupMenu(string tag)
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Users, "Add people to " + tag))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Users, L("PairGroups.Menu.AddPeople", "Add people to {0}", tag)))
|
||||
{
|
||||
_selectGroupForPairUi.Open(tag);
|
||||
}
|
||||
UiSharedService.AttachToolTip($"Add more users to Group {tag}");
|
||||
UiSharedService.AttachToolTip(L("PairGroups.Menu.AddPeople.Tooltip", "Add more users to Group {0}", tag));
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, "Delete " + tag) && UiSharedService.CtrlPressed())
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, L("PairGroups.Menu.Delete", "Delete {0}", tag)) && UiSharedService.CtrlPressed())
|
||||
{
|
||||
_tagHandler.RemoveTag(tag);
|
||||
}
|
||||
UiSharedService.AttachToolTip($"Delete Group {tag} (Will not delete the pairs)" + Environment.NewLine + "Hold CTRL to delete");
|
||||
UiSharedService.AttachToolTip(L("PairGroups.Menu.Delete.Tooltip", "Delete Group {0} (Will not delete the pairs)\nHold CTRL to delete", tag));
|
||||
}
|
||||
|
||||
private void DrawName(string tag, bool isSpecialTag, int visible, int online, int? total)
|
||||
{
|
||||
string displayedName = tag switch
|
||||
{
|
||||
TagHandler.CustomUnpairedTag => "Unpaired",
|
||||
TagHandler.CustomOfflineTag => "Offline",
|
||||
TagHandler.CustomOnlineTag => _mareConfig.Current.ShowOfflineUsersSeparately ? "Online/Paused" : "Contacts",
|
||||
TagHandler.CustomVisibleTag => "Visible",
|
||||
TagHandler.CustomUnpairedTag => L("PairGroups.Tag.Unpaired", "Unpaired"),
|
||||
TagHandler.CustomOfflineTag => L("PairGroups.Tag.Offline", "Offline"),
|
||||
TagHandler.CustomOnlineTag => _mareConfig.Current.ShowOfflineUsersSeparately
|
||||
? L("PairGroups.Tag.Online", "Online")
|
||||
: L("PairGroups.Tag.Contacts", "Contacts"),
|
||||
TagHandler.CustomVisibleTag => L("PairGroups.Tag.Visible", "Visible"),
|
||||
_ => tag
|
||||
};
|
||||
|
||||
string resultFolderName = !isSpecialTag ? $"{displayedName} ({visible}/{online}/{total} Pairs)" : $"{displayedName} ({online} Pairs)";
|
||||
|
||||
// FontAwesomeIcon.CaretSquareDown : FontAwesomeIcon.CaretSquareRight
|
||||
string resultFolderName = !isSpecialTag
|
||||
? L("PairGroups.Header.WithCounts", "{0} ({1}/{2}/{3} Pairs)", displayedName, visible, online, total ?? 0)
|
||||
: L("PairGroups.Header.Special", "{0} ({1} Pairs)", displayedName, online);
|
||||
var icon = _tagHandler.IsTagOpen(tag) ? FontAwesomeIcon.CaretSquareDown : FontAwesomeIcon.CaretSquareRight;
|
||||
_uiSharedService.IconText(icon);
|
||||
if (ImGui.IsItemClicked(ImGuiMouseButton.Left))
|
||||
@@ -181,11 +186,11 @@ public class PairGroupsUi
|
||||
if (!isSpecialTag && ImGui.IsItemHovered())
|
||||
{
|
||||
ImGui.BeginTooltip();
|
||||
ImGui.TextUnformatted($"Group {tag}");
|
||||
ImGui.TextUnformatted(L("PairGroups.Tooltip.Title", "Group {0}", tag));
|
||||
ImGui.Separator();
|
||||
ImGui.TextUnformatted($"{visible} Pairs visible");
|
||||
ImGui.TextUnformatted($"{online} Pairs online/paused");
|
||||
ImGui.TextUnformatted($"{total} Pairs total");
|
||||
ImGui.TextUnformatted(L("PairGroups.Tooltip.Visible", "{0} Pairs visible", visible));
|
||||
ImGui.TextUnformatted(L("PairGroups.Tooltip.Online", "{0} Pairs online/paused", online));
|
||||
ImGui.TextUnformatted(L("PairGroups.Tooltip.Total", "{0} Pairs total", total ?? 0));
|
||||
ImGui.EndTooltip();
|
||||
}
|
||||
}
|
||||
@@ -255,4 +260,4 @@ public class PairGroupsUi
|
||||
bool open = !_tagHandler.IsTagOpen(tag);
|
||||
_tagHandler.SetTagOpen(tag, open);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -28,17 +28,17 @@ public class BanUserPopupHandler : IPopupHandler
|
||||
|
||||
public void DrawContent()
|
||||
{
|
||||
UiSharedService.TextWrapped("User " + (_reportedPair.UserData.AliasOrUID) + " will be banned and removed from this Syncshell.");
|
||||
ImGui.InputTextWithHint("##banreason", "Ban Reason", ref _banReason, 255);
|
||||
UiSharedService.TextWrapped(_uiSharedService.Localize("Popup.BanUser.Description", "User {0} will be banned and removed from this Syncshell.", _reportedPair.UserData.AliasOrUID));
|
||||
ImGui.InputTextWithHint("##banreason", _uiSharedService.Localize("Popup.BanUser.ReasonHint", "Ban Reason"), ref _banReason, 255);
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.UserSlash, "Ban User"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.UserSlash, _uiSharedService.Localize("Popup.BanUser.Button", "Ban User")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
var reason = _banReason;
|
||||
_ = _apiController.GroupBanUser(new GroupPairDto(_group.Group, _reportedPair.UserData), reason);
|
||||
_banReason = string.Empty;
|
||||
}
|
||||
UiSharedService.TextWrapped("The reason will be displayed in the banlist. The current server-side alias if present (Vanity ID) will automatically be attached to the reason.");
|
||||
UiSharedService.TextWrapped(_uiSharedService.Localize("Popup.BanUser.ReasonNote", "The reason will be displayed in the banlist. The current server-side alias if present (Vanity ID) will automatically be attached to the reason."));
|
||||
}
|
||||
|
||||
public void Open(OpenBanUserPopupMessage message)
|
||||
@@ -47,4 +47,4 @@ public class BanUserPopupHandler : IPopupHandler
|
||||
_group = message.GroupFullInfoDto;
|
||||
_banReason = string.Empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -72,10 +72,10 @@ public class PopupHandler : WindowMediatorSubscriberBase
|
||||
if (_currentHandler.ShowClose)
|
||||
{
|
||||
ImGui.Separator();
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Times, "Close"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Times, _uiSharedService.Localize("Popup.Generic.Close", "Close")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,23 +29,21 @@ internal class ReportPopupHandler : IPopupHandler
|
||||
public void DrawContent()
|
||||
{
|
||||
using (_uiSharedService.UidFont.Push())
|
||||
UiSharedService.TextWrapped("Report " + _reportedPair!.UserData.AliasOrUID + " Profile");
|
||||
UiSharedService.TextWrapped(_uiSharedService.Localize("Popup.Report.Title", "Report {0} Profile", _reportedPair!.UserData.AliasOrUID));
|
||||
|
||||
ImGui.InputTextMultiline("##reportReason", ref _reportReason, 500, new Vector2(500 - ImGui.GetStyle().ItemSpacing.X * 2, 200));
|
||||
UiSharedService.TextWrapped($"Note: Sending a report will disable the offending profile globally.{Environment.NewLine}" +
|
||||
$"The report will be sent to the team of your currently connected server.{Environment.NewLine}" +
|
||||
$"Depending on the severity of the offense the users profile or account can be permanently disabled or banned.");
|
||||
UiSharedService.ColorTextWrapped("Report spam and wrong reports will not be tolerated and can lead to permanent account suspension.", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped("This is not for reporting misbehavior but solely for the actual profile. " +
|
||||
"Reports that are not solely for the profile will be ignored.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.TextWrapped(_uiSharedService.Localize("Popup.Report.Note", "Note: Sending a report will disable the offending profile globally.\nThe report will be sent to the team of your currently connected server.\nDepending on the severity of the offense the users profile or account can be permanently disabled or banned."));
|
||||
UiSharedService.ColorTextWrapped(_uiSharedService.Localize("Popup.Report.Warning", "Report spam and wrong reports will not be tolerated and can lead to permanent account suspension."), ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(_uiSharedService.Localize("Popup.Report.Scope", "This is not for reporting misbehavior but solely for the actual profile. Reports that are not solely for the profile will be ignored."), ImGuiColors.DalamudYellow);
|
||||
|
||||
using (ImRaii.Disabled(string.IsNullOrEmpty(_reportReason)))
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.ExclamationTriangle, "Send Report"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.ExclamationTriangle, _uiSharedService.Localize("Popup.Report.Button", "Send Report")))
|
||||
{
|
||||
ImGui.CloseCurrentPopup();
|
||||
var reason = _reportReason;
|
||||
_ = _apiController.UserReportProfile(new(_reportedPair.UserData, reason));
|
||||
_reportReason = string.Empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -55,4 +53,4 @@ internal class ReportPopupHandler : IPopupHandler
|
||||
_reportedPair = msg.PairToReport;
|
||||
_reportReason = string.Empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,20 +15,8 @@ public class SelectGroupForPairUi
|
||||
private readonly UidDisplayHandler _uidDisplayHandler;
|
||||
private readonly UiSharedService _uiSharedService;
|
||||
|
||||
/// <summary>
|
||||
/// The group UI is always open for a specific pair. This defines which pair the UI is open for.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
private Pair? _pair;
|
||||
|
||||
/// <summary>
|
||||
/// Should the panel show, yes/no
|
||||
/// </summary>
|
||||
private bool _show;
|
||||
|
||||
/// <summary>
|
||||
/// For the add category option, this stores the currently typed in tag name
|
||||
/// </summary>
|
||||
private string _tagNameToAdd = "";
|
||||
|
||||
public SelectGroupForPairUi(TagHandler tagHandler, UidDisplayHandler uidDisplayHandler, UiSharedService uiSharedService)
|
||||
@@ -48,8 +36,7 @@ public class SelectGroupForPairUi
|
||||
}
|
||||
|
||||
var name = PairName(_pair);
|
||||
var popupName = $"Choose Groups for {name}";
|
||||
// Is the popup supposed to show but did not open yet? Open it
|
||||
var popupName = _uiSharedService.Localize("PairGroups.Popup.Title", "Choose Groups for {0}", name);
|
||||
if (_show)
|
||||
{
|
||||
ImGui.OpenPopup(popupName);
|
||||
@@ -62,7 +49,7 @@ public class SelectGroupForPairUi
|
||||
var childHeight = tags.Count != 0 ? tags.Count * 25 : 1;
|
||||
var childSize = new Vector2(0, childHeight > 100 ? 100 : childHeight) * ImGuiHelpers.GlobalScale;
|
||||
|
||||
ImGui.TextUnformatted($"Select the groups you want {name} to be in.");
|
||||
ImGui.TextUnformatted(_uiSharedService.Localize("PairGroups.Popup.SelectPrompt", "Select the groups you want {0} to be in.", name));
|
||||
if (ImGui.BeginChild(name + "##listGroups", childSize))
|
||||
{
|
||||
foreach (var tag in tags)
|
||||
@@ -73,13 +60,13 @@ public class SelectGroupForPairUi
|
||||
}
|
||||
|
||||
ImGui.Separator();
|
||||
ImGui.TextUnformatted($"Create a new group for {name}.");
|
||||
ImGui.TextUnformatted(_uiSharedService.Localize("PairGroups.Popup.CreatePrompt", "Create a new group for {0}.", name));
|
||||
if (_uiSharedService.IconButton(FontAwesomeIcon.Plus))
|
||||
{
|
||||
HandleAddTag();
|
||||
}
|
||||
ImGui.SameLine();
|
||||
ImGui.InputTextWithHint("##category_name", "New Group", ref _tagNameToAdd, 40);
|
||||
ImGui.InputTextWithHint("##category_name", _uiSharedService.Localize("PairGroups.Popup.NewGroupHint", "New Group"), ref _tagNameToAdd, 40);
|
||||
if (ImGui.IsKeyDown(ImGuiKey.Enter))
|
||||
{
|
||||
HandleAddTag();
|
||||
@@ -91,10 +78,6 @@ public class SelectGroupForPairUi
|
||||
public void Open(Pair pair)
|
||||
{
|
||||
_pair = pair;
|
||||
// Using "_show" here to de-couple the opening of the popup
|
||||
// The popup name is derived from the name the user currently sees, which is
|
||||
// based on the showUidForEntry dictionary.
|
||||
// We'd have to derive the name here to open it popup modal here, when the Open() is called
|
||||
_show = true;
|
||||
}
|
||||
|
||||
@@ -136,4 +119,4 @@ public class SelectGroupForPairUi
|
||||
{
|
||||
return _uidDisplayHandler.GetPlayerText(pair).text;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface.Utility;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.UI;
|
||||
using MareSynchronos.UI.Handlers;
|
||||
using System.Numerics;
|
||||
|
||||
@@ -15,11 +16,13 @@ public class SelectPairForGroupUi
|
||||
private HashSet<string> _peopleInGroup = new(StringComparer.Ordinal);
|
||||
private bool _show = false;
|
||||
private string _tag = string.Empty;
|
||||
private readonly UiSharedService _uiSharedService;
|
||||
|
||||
public SelectPairForGroupUi(TagHandler tagHandler, UidDisplayHandler uidDisplayHandler)
|
||||
public SelectPairForGroupUi(TagHandler tagHandler, UidDisplayHandler uidDisplayHandler, UiSharedService uiSharedService)
|
||||
{
|
||||
_tagHandler = tagHandler;
|
||||
_uidDisplayHandler = uidDisplayHandler;
|
||||
_uiSharedService = uiSharedService;
|
||||
}
|
||||
|
||||
public void Draw(List<Pair> pairs)
|
||||
@@ -28,7 +31,7 @@ public class SelectPairForGroupUi
|
||||
var minSize = new Vector2(300, workHeight < 400 ? workHeight : 400) * ImGuiHelpers.GlobalScale;
|
||||
var maxSize = new Vector2(300, 1000) * ImGuiHelpers.GlobalScale;
|
||||
|
||||
var popupName = $"Choose Users for Group {_tag}";
|
||||
var popupName = _uiSharedService.Localize("PairGroups.SelectPairs.Title", "Choose Users for Group {0}", _tag);
|
||||
|
||||
if (!_show)
|
||||
{
|
||||
@@ -46,9 +49,9 @@ public class SelectPairForGroupUi
|
||||
ImGui.SetNextWindowSizeConstraints(minSize, maxSize);
|
||||
if (ImGui.BeginPopupModal(popupName, ref _show, ImGuiWindowFlags.Popup | ImGuiWindowFlags.Modal))
|
||||
{
|
||||
ImGui.TextUnformatted($"Select users for group {_tag}");
|
||||
ImGui.TextUnformatted(_uiSharedService.Localize("PairGroups.SelectPairs.SelectPrompt", "Select users for group {0}", _tag));
|
||||
|
||||
ImGui.InputTextWithHint("##filter", "Filter", ref _filter, 255, ImGuiInputTextFlags.None);
|
||||
ImGui.InputTextWithHint("##filter", _uiSharedService.Localize("PairGroups.SelectPairs.FilterHint", "Filter"), ref _filter, 255, ImGuiInputTextFlags.None);
|
||||
foreach (var item in pairs
|
||||
.Where(p => string.IsNullOrEmpty(_filter) || PairName(p).Contains(_filter, StringComparison.OrdinalIgnoreCase))
|
||||
.OrderBy(p => PairName(p), StringComparer.OrdinalIgnoreCase)
|
||||
@@ -89,4 +92,4 @@ public class SelectPairForGroupUi
|
||||
{
|
||||
return _uidDisplayHandler.GetPlayerText(pair).text;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,6 +9,8 @@ using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Utils;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using System.Numerics;
|
||||
using MareSynchronos.Localization;
|
||||
using System.Globalization;
|
||||
|
||||
namespace MareSynchronos.UI;
|
||||
|
||||
@@ -33,11 +35,17 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
private ObjectKind _selectedObjectTab;
|
||||
private bool _showModal = false;
|
||||
|
||||
private string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return _uiSharedService.Localize(key, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public DataAnalysisUi(ILogger<DataAnalysisUi> logger, MareMediator mediator,
|
||||
CharacterAnalyzer characterAnalyzer, IpcManager ipcManager,
|
||||
PerformanceCollectorService performanceCollectorService,
|
||||
UiSharedService uiSharedService)
|
||||
: base(logger, mediator, "Character Data Analysis", performanceCollectorService)
|
||||
: base(logger, mediator, uiSharedService.Localize("DataAnalysis.WindowTitle", "Character Data Analysis"), performanceCollectorService)
|
||||
{
|
||||
_characterAnalyzer = characterAnalyzer;
|
||||
_ipcManager = ipcManager;
|
||||
@@ -65,14 +73,15 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
var modalTitle = L("DataAnalysis.Bc7.ModalTitle", "BC7 Conversion in Progress");
|
||||
if (_conversionTask != null && !_conversionTask.IsCompleted)
|
||||
{
|
||||
_showModal = true;
|
||||
if (ImGui.BeginPopupModal("BC7 Conversion in Progress"))
|
||||
if (ImGui.BeginPopupModal(modalTitle))
|
||||
{
|
||||
ImGui.TextUnformatted("BC7 Conversion in progress: " + _conversionCurrentFileProgress + "/" + _texturesToConvert.Count);
|
||||
UiSharedService.TextWrapped("Current file: " + _conversionCurrentFileName);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.StopCircle, "Cancel conversion"))
|
||||
ImGui.TextUnformatted(L("DataAnalysis.Bc7.Status", "BC7 Conversion in progress: {0}/{1}", _conversionCurrentFileProgress, _texturesToConvert.Count));
|
||||
UiSharedService.TextWrapped(L("DataAnalysis.Bc7.CurrentFile", "Current file: {0}", _conversionCurrentFileName));
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.StopCircle, L("DataAnalysis.Bc7.Cancel", "Cancel conversion")))
|
||||
{
|
||||
_conversionCancellationTokenSource.Cancel();
|
||||
}
|
||||
@@ -95,7 +104,7 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
|
||||
if (_showModal && !_modalOpen)
|
||||
{
|
||||
ImGui.OpenPopup("BC7 Conversion in Progress");
|
||||
ImGui.OpenPopup(modalTitle);
|
||||
_modalOpen = true;
|
||||
}
|
||||
|
||||
@@ -106,7 +115,7 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
_sortDirty = true;
|
||||
}
|
||||
|
||||
UiSharedService.TextWrapped("This window shows you all files and their sizes that are currently in use through your character and associated entities");
|
||||
UiSharedService.TextWrapped(L("DataAnalysis.Description", "This window shows you all files and their sizes that are currently in use through your character and associated entities"));
|
||||
|
||||
if (_cachedAnalysis == null || _cachedAnalysis.Count == 0) return;
|
||||
|
||||
@@ -114,9 +123,9 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
bool needAnalysis = _cachedAnalysis!.Any(c => c.Value.Any(f => !f.Value.IsComputed));
|
||||
if (isAnalyzing)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped($"Analyzing {_characterAnalyzer.CurrentFile}/{_characterAnalyzer.TotalFiles}",
|
||||
UiSharedService.ColorTextWrapped(L("DataAnalysis.Analyzing", "Analyzing {0}/{1}", _characterAnalyzer.CurrentFile, _characterAnalyzer.TotalFiles),
|
||||
ImGuiColors.DalamudYellow);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.StopCircle, "Cancel analysis"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.StopCircle, L("DataAnalysis.Button.CancelAnalysis", "Cancel analysis")))
|
||||
{
|
||||
_characterAnalyzer.CancelAnalyze();
|
||||
}
|
||||
@@ -125,16 +134,16 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
if (needAnalysis)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("Some entries in the analysis have file size not determined yet, press the button below to analyze your current data",
|
||||
UiSharedService.ColorTextWrapped(L("DataAnalysis.Analyze.MissingNotice", "Some entries in the analysis have file size not determined yet, press the button below to analyze your current data"),
|
||||
ImGuiColors.DalamudYellow);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, "Start analysis (missing entries)"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, L("DataAnalysis.Button.StartMissing", "Start analysis (missing entries)")))
|
||||
{
|
||||
_ = _characterAnalyzer.ComputeAnalysis(print: false);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, "Start analysis (recalculate all entries)"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, L("DataAnalysis.Button.StartAll", "Start analysis (recalculate all entries)")))
|
||||
{
|
||||
_ = _characterAnalyzer.ComputeAnalysis(print: false, recalculate: true);
|
||||
}
|
||||
@@ -143,7 +152,7 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
|
||||
ImGui.Separator();
|
||||
|
||||
ImGui.TextUnformatted("Total files:");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.TotalFiles", "Total files:"));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(_cachedAnalysis!.Values.Sum(c => c.Values.Count).ToString());
|
||||
ImGui.SameLine();
|
||||
@@ -153,17 +162,16 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
}
|
||||
if (ImGui.IsItemHovered())
|
||||
{
|
||||
string text = "";
|
||||
var groupedfiles = _cachedAnalysis.Values.SelectMany(f => f.Values).GroupBy(f => f.FileType, StringComparer.Ordinal);
|
||||
text = string.Join(Environment.NewLine, groupedfiles.OrderBy(f => f.Key, StringComparer.Ordinal)
|
||||
.Select(f => f.Key + ": " + f.Count() + " files, size: " + UiSharedService.ByteToString(f.Sum(v => v.OriginalSize))
|
||||
+ ", compressed: " + UiSharedService.ByteToString(f.Sum(v => v.CompressedSize))));
|
||||
ImGui.SetTooltip(text);
|
||||
var summary = string.Join(Environment.NewLine, groupedfiles.OrderBy(f => f.Key, StringComparer.Ordinal)
|
||||
.Select(f => L("DataAnalysis.Tooltip.FileSummary", "{0}: {1} files, size: {2}, compressed: {3}",
|
||||
f.Key, f.Count(), UiSharedService.ByteToString(f.Sum(v => v.OriginalSize)), UiSharedService.ByteToString(f.Sum(v => v.CompressedSize)))));
|
||||
ImGui.SetTooltip(summary);
|
||||
}
|
||||
ImGui.TextUnformatted("Total size (actual):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.TotalSizeActual", "Total size (actual):"));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(UiSharedService.ByteToString(_cachedAnalysis!.Sum(c => c.Value.Sum(c => c.Value.OriginalSize))));
|
||||
ImGui.TextUnformatted("Total size (download size):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.TotalSizeDownload", "Total size (download size):"));
|
||||
ImGui.SameLine();
|
||||
using (ImRaii.PushColor(ImGuiCol.Text, ImGuiColors.DalamudYellow, needAnalysis))
|
||||
{
|
||||
@@ -173,10 +181,11 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
using (ImRaii.PushFont(UiBuilder.IconFont))
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.ExclamationCircle.ToIconString());
|
||||
UiSharedService.AttachToolTip("Click \"Start analysis\" to calculate download size");
|
||||
UiSharedService.AttachToolTip(L("DataAnalysis.Tooltip.CalculateDownloadSize", "Click \"Start analysis\" to calculate download size"));
|
||||
}
|
||||
}
|
||||
ImGui.TextUnformatted($"Total modded model triangles: {UiSharedService.TrisToString(_cachedAnalysis.Sum(c => c.Value.Sum(f => f.Value.Triangles)))}");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.TotalTriangles", "Total modded model triangles: {0}",
|
||||
UiSharedService.TrisToString(_cachedAnalysis.Sum(c => c.Value.Sum(f => f.Value.Triangles)))));
|
||||
ImGui.Separator();
|
||||
|
||||
using var tabbar = ImRaii.TabBar("objectSelection");
|
||||
@@ -190,7 +199,7 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
var groupedfiles = kvp.Value.Select(v => v.Value).GroupBy(f => f.FileType, StringComparer.Ordinal)
|
||||
.OrderBy(k => k.Key, StringComparer.Ordinal).ToList();
|
||||
|
||||
ImGui.TextUnformatted("Files for " + kvp.Key);
|
||||
ImGui.TextUnformatted(L("DataAnalysis.FilesFor", "Files for {0}", kvp.Key));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(kvp.Value.Count.ToString());
|
||||
ImGui.SameLine();
|
||||
@@ -201,16 +210,15 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
}
|
||||
if (ImGui.IsItemHovered())
|
||||
{
|
||||
string text = "";
|
||||
text = string.Join(Environment.NewLine, groupedfiles
|
||||
.Select(f => f.Key + ": " + f.Count() + " files, size: " + UiSharedService.ByteToString(f.Sum(v => v.OriginalSize))
|
||||
+ ", compressed: " + UiSharedService.ByteToString(f.Sum(v => v.CompressedSize))));
|
||||
ImGui.SetTooltip(text);
|
||||
var summary = string.Join(Environment.NewLine, groupedfiles
|
||||
.Select(f => L("DataAnalysis.Tooltip.FileSummary", "{0}: {1} files, size: {2}, compressed: {3}",
|
||||
f.Key, f.Count(), UiSharedService.ByteToString(f.Sum(v => v.OriginalSize)), UiSharedService.ByteToString(f.Sum(v => v.CompressedSize)))));
|
||||
ImGui.SetTooltip(summary);
|
||||
}
|
||||
ImGui.TextUnformatted($"{kvp.Key} size (actual):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.Object.SizeActual", "{0} size (actual):", kvp.Key));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(UiSharedService.ByteToString(kvp.Value.Sum(c => c.Value.OriginalSize)));
|
||||
ImGui.TextUnformatted($"{kvp.Key} size (download size):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.Object.SizeDownload", "{0} size (download size):", kvp.Key));
|
||||
ImGui.SameLine();
|
||||
using (ImRaii.PushColor(ImGuiCol.Text, ImGuiColors.DalamudYellow, needAnalysis))
|
||||
{
|
||||
@@ -220,17 +228,18 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
using (ImRaii.PushFont(UiBuilder.IconFont))
|
||||
ImGui.TextUnformatted(FontAwesomeIcon.ExclamationCircle.ToIconString());
|
||||
UiSharedService.AttachToolTip("Click \"Start analysis\" to calculate download size");
|
||||
UiSharedService.AttachToolTip(L("DataAnalysis.Tooltip.CalculateDownloadSize", "Click \"Start analysis\" to calculate download size"));
|
||||
}
|
||||
}
|
||||
ImGui.TextUnformatted($"{kvp.Key} VRAM usage:");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.Object.Vram", "{0} VRAM usage:", kvp.Key));
|
||||
ImGui.SameLine();
|
||||
var vramUsage = groupedfiles.SingleOrDefault(v => string.Equals(v.Key, "tex", StringComparison.Ordinal));
|
||||
if (vramUsage != null)
|
||||
{
|
||||
ImGui.TextUnformatted(UiSharedService.ByteToString(vramUsage.Sum(f => f.OriginalSize)));
|
||||
}
|
||||
ImGui.TextUnformatted($"{kvp.Key} modded model triangles: {UiSharedService.TrisToString(kvp.Value.Sum(f => f.Value.Triangles))}");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.Object.Triangles", "{0} modded model triangles: {1}", kvp.Key,
|
||||
UiSharedService.TrisToString(kvp.Value.Sum(f => f.Value.Triangles))));
|
||||
|
||||
ImGui.Separator();
|
||||
if (_selectedObjectTab != kvp.Key)
|
||||
@@ -266,33 +275,35 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
_texturesToConvert.Clear();
|
||||
}
|
||||
|
||||
ImGui.TextUnformatted($"{fileGroup.Key} files");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.FileGroup.Count", "{0} files", fileGroup.Key));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(fileGroup.Count().ToString());
|
||||
|
||||
ImGui.TextUnformatted($"{fileGroup.Key} files size (actual):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.FileGroup.SizeActual", "{0} files size (actual):", fileGroup.Key));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(UiSharedService.ByteToString(fileGroup.Sum(c => c.OriginalSize)));
|
||||
|
||||
ImGui.TextUnformatted($"{fileGroup.Key} files size (download size):");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.FileGroup.SizeDownload", "{0} files size (download size):", fileGroup.Key));
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted(UiSharedService.ByteToString(fileGroup.Sum(c => c.CompressedSize)));
|
||||
|
||||
if (string.Equals(_selectedFileTypeTab, "tex", StringComparison.Ordinal))
|
||||
{
|
||||
ImGui.Checkbox("Enable BC7 Conversion Mode", ref _enableBc7ConversionMode);
|
||||
ImGui.Checkbox(L("DataAnalysis.Bc7.EnableMode", "Enable BC7 Conversion Mode"), ref _enableBc7ConversionMode);
|
||||
if (_enableBc7ConversionMode)
|
||||
{
|
||||
UiSharedService.ColorText("WARNING BC7 CONVERSION:", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorText(L("DataAnalysis.Bc7.WarningTitle", "WARNING BC7 CONVERSION:"), ImGuiColors.DalamudYellow);
|
||||
ImGui.SameLine();
|
||||
UiSharedService.ColorText("Converting textures to BC7 is irreversible!", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped("- Converting textures to BC7 will reduce their size (compressed and uncompressed) drastically. It is recommended to be used for large (4k+) textures." +
|
||||
Environment.NewLine + "- Some textures, especially ones utilizing colorsets, might not be suited for BC7 conversion and might produce visual artifacts." +
|
||||
Environment.NewLine + "- Before converting textures, make sure to have the original files of the mod you are converting so you can reimport it in case of issues." +
|
||||
Environment.NewLine + "- Conversion will convert all found texture duplicates (entries with more than 1 file path) automatically." +
|
||||
Environment.NewLine + "- Converting textures to BC7 is a very expensive operation and, depending on the amount of textures to convert, will take a while to complete."
|
||||
, ImGuiColors.DalamudYellow);
|
||||
if (_texturesToConvert.Count > 0 && _uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, "Start conversion of " + _texturesToConvert.Count + " texture(s)"))
|
||||
UiSharedService.ColorText(L("DataAnalysis.Bc7.WarningIrreversible", "Converting textures to BC7 is irreversible!"), ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("DataAnalysis.Bc7.WarningDetails",
|
||||
"- Converting textures to BC7 will reduce their size (compressed and uncompressed) drastically. It is recommended to be used for large (4k+) textures.\n"
|
||||
+ "- Some textures, especially ones utilizing colorsets, might not be suited for BC7 conversion and might produce visual artifacts.\n"
|
||||
+ "- Before converting textures, make sure to have the original files of the mod you are converting so you can reimport it in case of issues.\n"
|
||||
+ "- Conversion will convert all found texture duplicates (entries with more than 1 file path) automatically.\n"
|
||||
+ "- Converting textures to BC7 is a very expensive operation and, depending on the amount of textures to convert, will take a while to complete."),
|
||||
ImGuiColors.DalamudYellow);
|
||||
if (_texturesToConvert.Count > 0 && _uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle,
|
||||
L("DataAnalysis.Bc7.StartConversion", "Start conversion of {0} texture(s)", _texturesToConvert.Count)))
|
||||
{
|
||||
_conversionCancellationTokenSource = _conversionCancellationTokenSource.CancelRecreate();
|
||||
_conversionTask = _ipcManager.Penumbra.ConvertTextureFiles(_logger, _texturesToConvert, _conversionProgress, _conversionCancellationTokenSource.Token);
|
||||
@@ -310,33 +321,33 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
|
||||
ImGui.Separator();
|
||||
|
||||
ImGui.TextUnformatted("Selected file:");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.SelectedFile", "Selected file:"));
|
||||
ImGui.SameLine();
|
||||
UiSharedService.ColorText(_selectedHash, ImGuiColors.DalamudYellow);
|
||||
|
||||
if (_cachedAnalysis[_selectedObjectTab].TryGetValue(_selectedHash, out CharacterAnalyzer.FileDataEntry? item))
|
||||
{
|
||||
var filePaths = item.FilePaths;
|
||||
ImGui.TextUnformatted("Local file path:");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.LocalFilePath", "Local file path:"));
|
||||
ImGui.SameLine();
|
||||
UiSharedService.TextWrapped(filePaths[0]);
|
||||
if (filePaths.Count > 1)
|
||||
{
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted($"(and {filePaths.Count - 1} more)");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.MoreCount", "(and {0} more)", filePaths.Count - 1));
|
||||
ImGui.SameLine();
|
||||
_uiSharedService.IconText(FontAwesomeIcon.InfoCircle);
|
||||
UiSharedService.AttachToolTip(string.Join(Environment.NewLine, filePaths.Skip(1)));
|
||||
}
|
||||
|
||||
var gamepaths = item.GamePaths;
|
||||
ImGui.TextUnformatted("Used by game path:");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.GamePath", "Used by game path:"));
|
||||
ImGui.SameLine();
|
||||
UiSharedService.TextWrapped(gamepaths[0]);
|
||||
if (gamepaths.Count > 1)
|
||||
{
|
||||
ImGui.SameLine();
|
||||
ImGui.TextUnformatted($"(and {gamepaths.Count - 1} more)");
|
||||
ImGui.TextUnformatted(L("DataAnalysis.MoreCount", "(and {0} more)", gamepaths.Count - 1));
|
||||
ImGui.SameLine();
|
||||
_uiSharedService.IconText(FontAwesomeIcon.InfoCircle);
|
||||
UiSharedService.AttachToolTip(string.Join(Environment.NewLine, gamepaths.Skip(1)));
|
||||
@@ -372,20 +383,20 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
using var table = ImRaii.Table("Analysis", tableColumns, ImGuiTableFlags.Sortable | ImGuiTableFlags.RowBg | ImGuiTableFlags.ScrollY | ImGuiTableFlags.SizingFixedFit,
|
||||
new Vector2(0, 300));
|
||||
if (!table.Success) return;
|
||||
ImGui.TableSetupColumn("Hash");
|
||||
ImGui.TableSetupColumn("Filepaths", ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn("Gamepaths", ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn("File Size", ImGuiTableColumnFlags.DefaultSort | ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn("Download Size", ImGuiTableColumnFlags.PreferSortDescending);
|
||||
if (string.Equals(fileGroup.Key, "tex", StringComparison.Ordinal))
|
||||
{
|
||||
ImGui.TableSetupColumn("Format");
|
||||
if (_enableBc7ConversionMode) ImGui.TableSetupColumn("Convert to BC7");
|
||||
}
|
||||
if (string.Equals(fileGroup.Key, "mdl", StringComparison.Ordinal))
|
||||
{
|
||||
ImGui.TableSetupColumn("Triangles", ImGuiTableColumnFlags.PreferSortDescending);
|
||||
}
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.Hash", "Hash"));
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.Filepaths", "Filepaths"), ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.Gamepaths", "Gamepaths"), ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.FileSize", "File Size"), ImGuiTableColumnFlags.DefaultSort | ImGuiTableColumnFlags.PreferSortDescending);
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.DownloadSize", "Download Size"), ImGuiTableColumnFlags.PreferSortDescending);
|
||||
if (string.Equals(fileGroup.Key, "tex", StringComparison.Ordinal))
|
||||
{
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.Format", "Format"));
|
||||
if (_enableBc7ConversionMode) ImGui.TableSetupColumn(L("DataAnalysis.Table.ConvertToBc7", "Convert to BC7"));
|
||||
}
|
||||
if (string.Equals(fileGroup.Key, "mdl", StringComparison.Ordinal))
|
||||
{
|
||||
ImGui.TableSetupColumn(L("DataAnalysis.Table.Triangles", "Triangles"), ImGuiTableColumnFlags.PreferSortDescending);
|
||||
}
|
||||
ImGui.TableSetupScrollFreeze(0, 1);
|
||||
ImGui.TableHeadersRow();
|
||||
|
||||
@@ -489,4 +500,4 @@ public class DataAnalysisUi : WindowMediatorSubscriberBase
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -23,7 +23,7 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
|
||||
public DownloadUi(ILogger<DownloadUi> logger, DalamudUtilService dalamudUtilService, MareConfigService configService,
|
||||
FileUploadManager fileTransferManager, MareMediator mediator, UiSharedService uiShared, PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "Umbra Downloads", performanceCollectorService)
|
||||
: base(logger, mediator, uiShared.Localize("DownloadUi.WindowTitle", "Umbra Downloads"), performanceCollectorService)
|
||||
{
|
||||
_dalamudUtilService = dalamudUtilService;
|
||||
_configService = configService;
|
||||
@@ -69,6 +69,8 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
});
|
||||
}
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiShared.Localize(key, fallback, args);
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
if (_configService.Current.ShowTransferWindow)
|
||||
@@ -87,7 +89,7 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
UiSharedService.DrawOutlinedFont($"▲", ImGuiColors.DalamudWhite, new Vector4(0, 0, 0, 255), 1);
|
||||
ImGui.SameLine();
|
||||
var xDistance = ImGui.GetCursorPosX();
|
||||
UiSharedService.DrawOutlinedFont($"Compressing+Uploading {doneUploads}/{totalUploads}",
|
||||
UiSharedService.DrawOutlinedFont(L("DownloadUi.UploadStatus", "Compressing+Uploading {0}/{1}", doneUploads, totalUploads),
|
||||
ImGuiColors.DalamudWhite, new Vector4(0, 0, 0, 255), 1);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(xDistance);
|
||||
@@ -120,7 +122,7 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
var xDistance = ImGui.GetCursorPosX();
|
||||
UiSharedService.DrawOutlinedFont(
|
||||
$"{item.Key.Name} [W:{dlSlot}/Q:{dlQueue}/P:{dlProg}/D:{dlDecomp}]",
|
||||
L("DownloadUi.DownloadStatus", "{0} [W:{1}/Q:{2}/P:{3}/D:{4}]", item.Key.Name, dlSlot, dlQueue, dlProg, dlDecomp),
|
||||
ImGuiColors.DalamudWhite, new Vector4(0, 0, 0, 255), 1);
|
||||
ImGui.NewLine();
|
||||
ImGui.SameLine(xDistance);
|
||||
@@ -163,13 +165,13 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
UiSharedService.Color(0, 0, 0, transparency), 1);
|
||||
drawList.AddRectFilled(dlBarStart with { X = dlBarStart.X - dlBarBorder, Y = dlBarStart.Y - dlBarBorder },
|
||||
dlBarEnd with { X = dlBarEnd.X + dlBarBorder, Y = dlBarEnd.Y + dlBarBorder },
|
||||
UiSharedService.Color(220, 220, 255, transparency), 1);
|
||||
UiSharedService.Color(230, 200, 255, transparency), 1);
|
||||
drawList.AddRectFilled(dlBarStart, dlBarEnd,
|
||||
UiSharedService.Color(0, 0, 0, transparency), 1);
|
||||
var dlProgressPercent = transferredBytes / (double)totalBytes;
|
||||
drawList.AddRectFilled(dlBarStart,
|
||||
dlBarEnd with { X = dlBarStart.X + (float)(dlProgressPercent * dlBarWidth) },
|
||||
UiSharedService.Color(100, 100, 255, transparency), 1);
|
||||
UiSharedService.Color(160, 64, 255, transparency), 1);
|
||||
|
||||
if (_configService.Current.TransferBarsShowText)
|
||||
{
|
||||
@@ -191,7 +193,7 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
try
|
||||
{
|
||||
using var _ = _uiShared.UidFont.Push();
|
||||
var uploadText = "Uploading";
|
||||
var uploadText = L("DownloadUi.UploadingLabel", "Uploading");
|
||||
|
||||
var textSize = ImGui.CalcTextSize(uploadText);
|
||||
|
||||
@@ -245,4 +247,4 @@ public class DownloadUi : WindowMediatorSubscriberBase
|
||||
MaximumSize = new Vector2(300, maxHeight),
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
using Dalamud.Game.Text.SeStringHandling;
|
||||
using Dalamud.Game.Text.SeStringHandling.Payloads;
|
||||
using Dalamud.Plugin.Services;
|
||||
using Dalamud.Interface;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.MareConfiguration.Configurations;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
@@ -10,11 +11,14 @@ using MareSynchronos.WebAPI;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using System.Runtime.InteropServices;
|
||||
using MareSynchronos.Localization;
|
||||
using System.Globalization;
|
||||
|
||||
namespace MareSynchronos.UI;
|
||||
|
||||
public sealed class DtrEntry : IDisposable, IHostedService
|
||||
{
|
||||
public const string DefaultGlyph = "\u25CB";
|
||||
private enum DtrStyle
|
||||
{
|
||||
Default,
|
||||
@@ -44,6 +48,13 @@ public sealed class DtrEntry : IDisposable, IHostedService
|
||||
private string? _tooltip;
|
||||
private Colors _colors;
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public DtrEntry(ILogger<DtrEntry> logger, IDtrBar dtrBar, MareConfigService configService, MareMediator mareMediator, PairManager pairManager, ApiController apiController)
|
||||
{
|
||||
_logger = logger;
|
||||
@@ -104,7 +115,7 @@ public sealed class DtrEntry : IDisposable, IHostedService
|
||||
private IDtrBarEntry CreateEntry()
|
||||
{
|
||||
_logger.LogTrace("Creating new DtrBar entry");
|
||||
var entry = _dtrBar.Get("Umbra");
|
||||
var entry = _dtrBar.Get(L("DtrEntry.EntryName", "Umbra"));
|
||||
entry.OnClick = _ => _mareMediator.Publish(new UiToggleMessage(typeof(CompactUi)));
|
||||
|
||||
return entry;
|
||||
@@ -163,19 +174,20 @@ public sealed class DtrEntry : IDisposable, IHostedService
|
||||
.Select(x => string.Format("{0}", _configService.Current.PreferNoteInDtrTooltip ? x.GetNoteOrName() : x.PlayerName));
|
||||
}
|
||||
|
||||
tooltip = $"Umbra: Connected{Environment.NewLine}----------{Environment.NewLine}{string.Join(Environment.NewLine, visiblePairs)}";
|
||||
var header = L("DtrEntry.Tooltip.Connected", "Umbra: Connected");
|
||||
tooltip = header + Environment.NewLine + "----------" + Environment.NewLine + string.Join(Environment.NewLine, visiblePairs);
|
||||
colors = _configService.Current.DtrColorsPairsInRange;
|
||||
}
|
||||
else
|
||||
{
|
||||
tooltip = "Umbra: Connected";
|
||||
tooltip = L("DtrEntry.Tooltip.Connected", "Umbra: Connected");
|
||||
colors = _configService.Current.DtrColorsDefault;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
text = RenderDtrStyle(_configService.Current.DtrStyle, "\uE04C");
|
||||
tooltip = "Umbra: Not Connected";
|
||||
tooltip = L("DtrEntry.Tooltip.Disconnected", "Umbra: Not Connected");
|
||||
colors = _configService.Current.DtrColorsNotConnected;
|
||||
}
|
||||
|
||||
@@ -196,7 +208,8 @@ public sealed class DtrEntry : IDisposable, IHostedService
|
||||
{
|
||||
var style = (DtrStyle)styleNum;
|
||||
|
||||
return style switch {
|
||||
return style switch
|
||||
{
|
||||
DtrStyle.Style1 => $"\xE039 {text}",
|
||||
DtrStyle.Style2 => $"\xE0BC {text}",
|
||||
DtrStyle.Style3 => $"\xE0BD {text}",
|
||||
@@ -206,7 +219,7 @@ public sealed class DtrEntry : IDisposable, IHostedService
|
||||
DtrStyle.Style7 => $"\xE05D {text}",
|
||||
DtrStyle.Style8 => $"\xE03C{text}",
|
||||
DtrStyle.Style9 => $"\xE040 {text} \xE041",
|
||||
_ => $"\uE044 {text}"
|
||||
_ => DefaultGlyph + " " + text
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
@@ -35,7 +35,7 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
ApiController apiController, UiSharedService uiSharedService, FileDialogManager fileDialogManager,
|
||||
ServerConfigurationManager serverConfigurationManager,
|
||||
MareProfileManager mareProfileManager, PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "Umbra Edit Profile###UmbraSyncEditProfileUI", performanceCollectorService)
|
||||
: base(logger, mediator, uiSharedService.Localize("EditProfile.WindowTitle", "Umbra Edit Profile") + "###UmbraSyncEditProfileUI", performanceCollectorService)
|
||||
{
|
||||
IsOpen = false;
|
||||
this.SizeConstraints = new()
|
||||
@@ -62,9 +62,11 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
});
|
||||
}
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiSharedService.Localize(key, fallback, args);
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
_uiSharedService.BigText("Current Profile (as saved on server)");
|
||||
_uiSharedService.BigText(L("EditProfile.CurrentProfile", "Current Profile (as saved on server)"));
|
||||
|
||||
var profile = _mareProfileManager.GetMareProfile(new UserData(_apiController.UID));
|
||||
|
||||
@@ -125,9 +127,9 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
ImGui.Separator();
|
||||
_uiSharedService.BigText("Profile Settings");
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.FileUpload, "Upload new profile picture"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.FileUpload, L("EditProfile.Button.UploadPicture", "Upload new profile picture")))
|
||||
{
|
||||
_fileDialogManager.OpenFileDialog("Select new Profile picture", ".png", (success, file) =>
|
||||
_fileDialogManager.OpenFileDialog(L("EditProfile.Dialog.PictureTitle", "Select new Profile picture"), ".png", (success, file) =>
|
||||
{
|
||||
if (!success) return;
|
||||
_ = Task.Run(async () =>
|
||||
@@ -148,29 +150,29 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
});
|
||||
});
|
||||
}
|
||||
UiSharedService.AttachToolTip("Select and upload a new profile picture");
|
||||
UiSharedService.AttachToolTip(L("EditProfile.Tooltip.UploadPicture", "Select and upload a new profile picture"));
|
||||
ImGui.SameLine();
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, "Clear uploaded profile picture"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, L("EditProfile.Button.ClearPicture", "Clear uploaded profile picture")))
|
||||
{
|
||||
_ = _apiController.UserSetProfile(new UserProfileDto(new UserData(_apiController.UID), Disabled: false, IsNSFW: null, "", Description: null));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Clear your currently uploaded profile picture");
|
||||
UiSharedService.AttachToolTip(L("EditProfile.Tooltip.ClearPicture", "Clear your currently uploaded profile picture"));
|
||||
if (_showFileDialogError)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("The profile picture must be a PNG file with a maximum height and width of 256px and 250KiB size", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("EditProfile.Error.PictureTooLarge", "The profile picture must be a PNG file with a maximum height and width of 256px and 250KiB size"), ImGuiColors.DalamudRed);
|
||||
}
|
||||
var isNsfw = profile.IsNSFW;
|
||||
if (ImGui.Checkbox("Profile is NSFW", ref isNsfw))
|
||||
if (ImGui.Checkbox(L("EditProfile.Checkbox.Nsfw", "Profile is NSFW"), ref isNsfw))
|
||||
{
|
||||
_ = _apiController.UserSetProfile(new UserProfileDto(new UserData(_apiController.UID), Disabled: false, isNsfw, ProfilePictureBase64: null, Description: null));
|
||||
}
|
||||
_uiSharedService.DrawHelpText("If your profile description or image can be considered NSFW, toggle this to ON");
|
||||
_uiSharedService.DrawHelpText(L("EditProfile.Help.Nsfw", "If your profile description or image can be considered NSFW, toggle this to ON"));
|
||||
var widthTextBox = 400;
|
||||
var posX = ImGui.GetCursorPosX();
|
||||
ImGui.TextUnformatted($"Description {_descriptionText.Length}/1500");
|
||||
ImGui.TextUnformatted(L("EditProfile.DescriptionCounter", "Description {0}/1500", _descriptionText.Length));
|
||||
ImGui.SetCursorPosX(posX);
|
||||
ImGuiHelpers.ScaledRelativeSameLine(widthTextBox, ImGui.GetStyle().ItemSpacing.X);
|
||||
ImGui.TextUnformatted("Preview (approximate)");
|
||||
ImGui.TextUnformatted(L("EditProfile.PreviewLabel", "Preview (approximate)"));
|
||||
using (_uiSharedService.GameFont.Push())
|
||||
ImGui.InputTextMultiline("##description", ref _descriptionText, 1500, ImGuiHelpers.ScaledVector2(widthTextBox, 200));
|
||||
|
||||
@@ -199,17 +201,17 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
ImGui.EndChildFrame();
|
||||
}
|
||||
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Save, "Save Description"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Save, L("EditProfile.Button.SaveDescription", "Save Description")))
|
||||
{
|
||||
_ = _apiController.UserSetProfile(new UserProfileDto(new UserData(_apiController.UID), Disabled: false, IsNSFW: null, ProfilePictureBase64: null, _descriptionText));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Sets your profile description text");
|
||||
UiSharedService.AttachToolTip(L("EditProfile.Tooltip.SaveDescription", "Sets your profile description text"));
|
||||
ImGui.SameLine();
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, "Clear Description"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.Trash, L("EditProfile.Button.ClearDescription", "Clear Description")))
|
||||
{
|
||||
_ = _apiController.UserSetProfile(new UserProfileDto(new UserData(_apiController.UID), Disabled: false, IsNSFW: null, ProfilePictureBase64: null, ""));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Clears your profile description text");
|
||||
UiSharedService.AttachToolTip(L("EditProfile.Tooltip.ClearDescription", "Clears your profile description text"));
|
||||
}
|
||||
|
||||
protected override void Dispose(bool disposing)
|
||||
@@ -217,4 +219,4 @@ public class EditProfileUi : WindowMediatorSubscriberBase
|
||||
base.Dispose(disposing);
|
||||
_pfpTextureWrap?.Dispose();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -40,7 +40,7 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
public EventViewerUI(ILogger<EventViewerUI> logger, MareMediator mediator,
|
||||
EventAggregator eventAggregator, UiSharedService uiSharedService, MareConfigService configService,
|
||||
PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "Event Viewer", performanceCollectorService)
|
||||
: base(logger, mediator, uiSharedService.Localize("EventViewer.WindowTitle", "Event Viewer"), performanceCollectorService)
|
||||
{
|
||||
_eventAggregator = eventAggregator;
|
||||
_uiSharedService = uiSharedService;
|
||||
@@ -52,6 +52,8 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
_filteredEvents = RecreateFilter();
|
||||
}
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiSharedService.Localize(key, fallback, args);
|
||||
|
||||
private Lazy<List<Event>> RecreateFilter()
|
||||
{
|
||||
return new(() =>
|
||||
@@ -81,20 +83,22 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
{
|
||||
var newEventsAvailable = _eventAggregator.NewEventsAvailable;
|
||||
|
||||
var freezeSize = _uiSharedService.GetIconTextButtonSize(FontAwesomeIcon.PlayCircle, "Unfreeze View");
|
||||
var unfreezeLabel = L("EventViewer.Button.Unfreeze", "Unfreeze View");
|
||||
var freezeLabel = L("EventViewer.Button.Freeze", "Freeze View");
|
||||
var freezeSize = _uiSharedService.GetIconTextButtonSize(FontAwesomeIcon.PlayCircle, unfreezeLabel);
|
||||
if (_isPaused)
|
||||
{
|
||||
using (ImRaii.PushColor(ImGuiCol.Text, ImGuiColors.DalamudYellow, newEventsAvailable))
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, "Unfreeze View"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PlayCircle, unfreezeLabel))
|
||||
_isPaused = false;
|
||||
if (newEventsAvailable)
|
||||
UiSharedService.AttachToolTip("New events are available. Click to resume updating.");
|
||||
UiSharedService.AttachToolTip(L("EventViewer.Tooltip.NewEvents", "New events are available. Click to resume updating."));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PauseCircle, "Freeze View"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.PauseCircle, freezeLabel))
|
||||
_isPaused = true;
|
||||
}
|
||||
|
||||
@@ -105,7 +109,7 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
|
||||
bool changedFilter = false;
|
||||
ImGui.SetNextItemWidth(200);
|
||||
changedFilter |= ImGui.InputText("Filter lines", ref _filterFreeText, 50);
|
||||
changedFilter |= ImGui.InputText(L("EventViewer.FilterLabel", "Filter lines"), ref _filterFreeText, 50);
|
||||
if (changedFilter) _filteredEvents = RecreateFilter();
|
||||
|
||||
using (ImRaii.Disabled(_filterFreeText.IsNullOrEmpty()))
|
||||
@@ -120,10 +124,11 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
|
||||
if (_configService.Current.LogEvents)
|
||||
{
|
||||
var buttonSize = _uiSharedService.GetIconTextButtonSize(FontAwesomeIcon.FolderOpen, "Open EventLog Folder");
|
||||
var openLogLabel = L("EventViewer.Button.OpenLog", "Open EventLog folder");
|
||||
var buttonSize = _uiSharedService.GetIconTextButtonSize(FontAwesomeIcon.FolderOpen, openLogLabel);
|
||||
var dist = ImGui.GetWindowContentRegionMax().X - buttonSize;
|
||||
ImGui.SameLine(dist);
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.FolderOpen, "Open EventLog folder"))
|
||||
if (_uiSharedService.IconTextButton(FontAwesomeIcon.FolderOpen, openLogLabel))
|
||||
{
|
||||
ProcessStartInfo ps = new()
|
||||
{
|
||||
@@ -152,11 +157,11 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
{
|
||||
ImGui.TableSetupScrollFreeze(0, 1);
|
||||
ImGui.TableSetupColumn(string.Empty, ImGuiTableColumnFlags.NoSort);
|
||||
ImGui.TableSetupColumn("Time", ImGuiTableColumnFlags.None, timeColWidth);
|
||||
ImGui.TableSetupColumn("Source", ImGuiTableColumnFlags.None, sourceColWidth);
|
||||
ImGui.TableSetupColumn("UID", ImGuiTableColumnFlags.None, uidColWidth);
|
||||
ImGui.TableSetupColumn("Character", ImGuiTableColumnFlags.None, characterColWidth);
|
||||
ImGui.TableSetupColumn("Event", ImGuiTableColumnFlags.None);
|
||||
ImGui.TableSetupColumn(L("EventViewer.Column.Time", "Time"), ImGuiTableColumnFlags.None, timeColWidth);
|
||||
ImGui.TableSetupColumn(L("EventViewer.Column.Source", "Source"), ImGuiTableColumnFlags.None, sourceColWidth);
|
||||
ImGui.TableSetupColumn(L("EventViewer.Column.Uid", "UID"), ImGuiTableColumnFlags.None, uidColWidth);
|
||||
ImGui.TableSetupColumn(L("EventViewer.Column.Character", "Character"), ImGuiTableColumnFlags.None, characterColWidth);
|
||||
ImGui.TableSetupColumn(L("EventViewer.Column.Event", "Event"), ImGuiTableColumnFlags.None);
|
||||
ImGui.TableHeadersRow();
|
||||
int i = 0;
|
||||
foreach (var ev in _filteredEvents.Value)
|
||||
@@ -181,7 +186,7 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
|
||||
ImGui.TableNextColumn();
|
||||
_uiSharedService.IconText(icon, iconColor == new Vector4() ? null : iconColor);
|
||||
UiSharedService.AttachToolTip(ev.EventSeverity.ToString());
|
||||
UiSharedService.AttachToolTip(L($"EventViewer.Severity.{ev.EventSeverity}", ev.EventSeverity.ToString()));
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(ev.EventTime.ToString("T", CultureInfo.CurrentCulture));
|
||||
@@ -200,7 +205,7 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.TextUnformatted("--");
|
||||
ImGui.TextUnformatted(L("EventViewer.NoValue", "--"));
|
||||
}
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
@@ -214,7 +219,7 @@ internal class EventViewerUI : WindowMediatorSubscriberBase
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.TextUnformatted("--");
|
||||
ImGui.TextUnformatted(L("EventViewer.NoValue", "--"));
|
||||
}
|
||||
ImGui.TableNextColumn();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
|
||||
@@ -1,11 +1,13 @@
|
||||
using Dalamud.Bindings.ImGui;
|
||||
using Dalamud.Interface;
|
||||
using Dalamud.Interface.Utility.Raii;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.UI.Components;
|
||||
using System.Globalization;
|
||||
|
||||
namespace MareSynchronos.UI.Handlers;
|
||||
|
||||
@@ -22,6 +24,13 @@ public class UidDisplayHandler
|
||||
private bool _popupShown = false;
|
||||
private DateTime? _popupTime;
|
||||
|
||||
private static string L(string key, string fallback, params object[] args)
|
||||
{
|
||||
var safeArgs = args ?? Array.Empty<object>();
|
||||
return LocalizationService.Instance?.GetString(key, fallback, safeArgs)
|
||||
?? string.Format(CultureInfo.CurrentCulture, fallback, safeArgs);
|
||||
}
|
||||
|
||||
public UidDisplayHandler(MareMediator mediator, PairManager pairManager,
|
||||
ServerConfigurationManager serverManager, MareConfigService mareConfigService)
|
||||
{
|
||||
@@ -77,9 +86,7 @@ public class UidDisplayHandler
|
||||
|
||||
if (_popupTime > DateTime.UtcNow || !_mareConfigService.Current.ProfilesShow)
|
||||
{
|
||||
ImGui.SetTooltip("Left click to switch between UID display and nick" + Environment.NewLine
|
||||
+ "Right click to change nick for " + pair.UserData.AliasOrUID + Environment.NewLine
|
||||
+ "Middle Mouse Button to open their profile in a separate window");
|
||||
ImGui.SetTooltip(L("UidDisplay.Tooltip", "Left click to switch between UID display and nick\nRight click to change nick for {0}\nMiddle Mouse Button to open their profile in a separate window", pair.UserData.AliasOrUID));
|
||||
}
|
||||
else if (_popupTime < DateTime.UtcNow && !_popupShown)
|
||||
{
|
||||
@@ -125,7 +132,7 @@ public class UidDisplayHandler
|
||||
ImGui.SetCursorPosY(originalY);
|
||||
|
||||
ImGui.SetNextItemWidth(editBoxWidth.Invoke());
|
||||
if (ImGui.InputTextWithHint("##" + pair.UserData.UID, "Nick/Notes", ref _editUserComment, 255, ImGuiInputTextFlags.EnterReturnsTrue))
|
||||
if (ImGui.InputTextWithHint("##" + pair.UserData.UID, L("UidDisplay.EditNotes.Hint", "Nick/Notes"), ref _editUserComment, 255, ImGuiInputTextFlags.EnterReturnsTrue))
|
||||
{
|
||||
_serverManager.SetNoteForUid(pair.UserData.UID, _editUserComment);
|
||||
_serverManager.SaveNotes();
|
||||
@@ -136,7 +143,7 @@ public class UidDisplayHandler
|
||||
{
|
||||
_editNickEntry = string.Empty;
|
||||
}
|
||||
UiSharedService.AttachToolTip("Hit ENTER to save\nRight click to cancel");
|
||||
UiSharedService.AttachToolTip(L("GroupPanel.CommentTooltip", "Hit ENTER to save\nRight click to cancel"));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -201,4 +208,4 @@ public class UidDisplayHandler
|
||||
|
||||
return showUidInsteadOfName;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,6 +5,7 @@ using Dalamud.Interface.Utility.Raii;
|
||||
using Dalamud.Utility;
|
||||
using MareSynchronos.API.Dto.Account;
|
||||
using MareSynchronos.FileCache;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.MareConfiguration.Models;
|
||||
using MareSynchronos.Services;
|
||||
@@ -36,6 +37,8 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
private string? _registrationMessage;
|
||||
private RegisterReplyDto? _registrationReply;
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiShared.Localize(key, fallback, args);
|
||||
|
||||
public IntroUi(ILogger<IntroUi> logger, UiSharedService uiShared, MareConfigService configService,
|
||||
CacheMonitor fileCacheManager, ServerConfigurationManager serverConfigurationManager, MareMediator mareMediator,
|
||||
PerformanceCollectorService performanceCollectorService, DalamudUtilService dalamudUtilService, AccountRegistrationService registerService) : base(logger, mareMediator, "Umbra Setup", performanceCollectorService)
|
||||
@@ -87,17 +90,17 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
return _uiShared.ApiController.ServerState switch
|
||||
{
|
||||
ServerState.Reconnecting => "Reconnecting",
|
||||
ServerState.Connecting => "Connecting",
|
||||
ServerState.Disconnected => "Disconnected",
|
||||
ServerState.Disconnecting => "Disconnecting",
|
||||
ServerState.Unauthorized => "Unauthorized",
|
||||
ServerState.VersionMisMatch => "Version mismatch",
|
||||
ServerState.Offline => "Unavailable",
|
||||
ServerState.RateLimited => "Rate Limited",
|
||||
ServerState.NoSecretKey => "No Secret Key",
|
||||
ServerState.MultiChara => "Duplicate Characters",
|
||||
ServerState.Connected => "Connected",
|
||||
ServerState.Reconnecting => L("Compact.UidText.Reconnecting", "Reconnecting"),
|
||||
ServerState.Connecting => L("Compact.UidText.Connecting", "Connecting"),
|
||||
ServerState.Disconnected => L("Compact.UidText.Disconnected", "Disconnected"),
|
||||
ServerState.Disconnecting => L("Compact.UidText.Disconnecting", "Disconnecting"),
|
||||
ServerState.Unauthorized => L("Compact.UidText.Unauthorized", "Unauthorized"),
|
||||
ServerState.VersionMisMatch => L("Compact.UidText.VersionMismatch", "Version mismatch"),
|
||||
ServerState.Offline => L("Compact.UidText.Offline", "Unavailable"),
|
||||
ServerState.RateLimited => L("Compact.UidText.RateLimited", "Rate Limited"),
|
||||
ServerState.NoSecretKey => L("Compact.UidText.NoSecretKey", "No Secret Key"),
|
||||
ServerState.MultiChara => L("Compact.UidText.MultiChara", "Duplicate Characters"),
|
||||
ServerState.Connected => L("Intro.ConnectionStatus.Connected", "Connected"),
|
||||
_ => string.Empty
|
||||
};
|
||||
}
|
||||
@@ -108,18 +111,15 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
|
||||
if (!_configService.Current.AcceptedAgreement && !_readFirstPage)
|
||||
{
|
||||
_uiShared.BigText("Welcome to Umbra");
|
||||
_uiShared.BigText(L("Intro.Welcome.Title", "Welcome to Umbra"));
|
||||
ImGui.Separator();
|
||||
UiSharedService.TextWrapped("Umbra is a plugin that will replicate your full current character state including all Penumbra mods to other paired users. " +
|
||||
"Note that you will have to have Penumbra as well as Glamourer installed to use this plugin.");
|
||||
UiSharedService.TextWrapped("We will have to setup a few things first before you can start using this plugin. Click on next to continue.");
|
||||
UiSharedService.TextWrapped(L("Intro.Welcome.Paragraph1", "Umbra is a plugin that will replicate your full current character state including all Penumbra mods to other paired users. Note that you will have to have Penumbra as well as Glamourer installed to use this plugin."));
|
||||
UiSharedService.TextWrapped(L("Intro.Welcome.Paragraph2", "We will have to setup a few things first before you can start using this plugin. Click on next to continue."));
|
||||
|
||||
UiSharedService.ColorTextWrapped("Note: Any modifications you have applied through anything but Penumbra cannot be shared and your character state on other clients " +
|
||||
"might look broken because of this or others players mods might not apply on your end altogether. " +
|
||||
"If you want to use this plugin you will have to move your mods to Penumbra.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Welcome.Note", "Note: Any modifications you have applied through anything but Penumbra cannot be shared and your character state on other clients might look broken because of this or others players mods might not apply on your end altogether. If you want to use this plugin you will have to move your mods to Penumbra."), ImGuiColors.DalamudYellow);
|
||||
if (!_uiShared.DrawOtherPluginState(intro: true)) return;
|
||||
ImGui.Separator();
|
||||
if (ImGui.Button("Next##toAgreement"))
|
||||
if (ImGui.Button(L("Intro.Welcome.Next", "Next") + "##toAgreement"))
|
||||
{
|
||||
_readFirstPage = true;
|
||||
#if !DEBUG
|
||||
@@ -127,7 +127,7 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
for (int i = 10; i > 0; i--)
|
||||
{
|
||||
_timeoutLabel = $"'I agree' button will be available in {i}s";
|
||||
_timeoutLabel = L("Intro.Agreement.Timeout", "'I agree' button will be available in {0}s", i);
|
||||
await Task.Delay(TimeSpan.FromSeconds(1)).ConfigureAwait(false);
|
||||
}
|
||||
});
|
||||
@@ -140,52 +140,40 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
using (_uiShared.UidFont.Push())
|
||||
{
|
||||
ImGui.TextUnformatted("Agreement of Usage of Service");
|
||||
ImGui.TextUnformatted(L("Intro.Agreement.Title", "Agreement of Usage of Service"));
|
||||
}
|
||||
|
||||
ImGui.Separator();
|
||||
ImGui.SetWindowFontScale(1.5f);
|
||||
string readThis = "READ THIS CAREFULLY";
|
||||
string readThis = L("Intro.Agreement.Callout", "READ THIS CAREFULLY");
|
||||
Vector2 textSize = ImGui.CalcTextSize(readThis);
|
||||
ImGui.SetCursorPosX(ImGui.GetWindowSize().X / 2 - textSize.X / 2);
|
||||
UiSharedService.ColorText(readThis, ImGuiColors.DalamudRed);
|
||||
ImGui.SetWindowFontScale(1.0f);
|
||||
ImGui.Separator();
|
||||
UiSharedService.TextWrapped("""
|
||||
To use Umbra, you must be over the age of 18, or 21 in some jurisdictions.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
All of the mod files currently active on your character as well as your current character state will be uploaded to the service you registered yourself at automatically. The plugin will exclusively upload the necessary mod files and not the whole mod.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
If you are on a data capped internet connection, higher fees due to data usage depending on the amount of downloaded and uploaded mod files might occur. Mod files will be compressed on up- and download to save on bandwidth usage. Due to varying up- and download speeds, changes in characters might not be visible immediately. Files present on the service that already represent your active mod files will not be uploaded again.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
The mod files you are uploading are confidential and will not be distributed to parties other than the ones who are requesting the exact same mod files. Please think about who you are going to pair since it is unavoidable that they will receive and locally cache the necessary mod files that you have currently in use. Locally cached mod files will have arbitrary file names to discourage attempts at replicating the original mod.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
The plugin creator tried their best to keep you secure. However, there is no guarantee for 100% security. Do not blindly pair your client with everyone.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
Mod files that are saved on the service will remain on the service as long as there are requests for the files from clients. After a period of not being used, the mod files will be automatically deleted.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
Accounts that are inactive for ninety (90) days will be deleted for privacy reasons.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
Umbra is operated from servers located in the European Union. You agree not to upload any content to the service that violates EU law; and more specifically, German law.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
You may delete your account at any time from within the Settings panel of the plugin. Any mods unique to you will then be removed from the server within 14 days.
|
||||
""");
|
||||
UiSharedService.TextWrapped("""
|
||||
This service is provided as-is.
|
||||
""");
|
||||
var agreementParagraphs = new[]
|
||||
{
|
||||
L("Intro.Agreement.Paragraph1", "To use Umbra, you must be over the age of 18, or 21 in some jurisdictions."),
|
||||
L("Intro.Agreement.Paragraph2", "All of the mod files currently active on your character as well as your current character state will be uploaded to the service you registered yourself at automatically. The plugin will exclusively upload the necessary mod files and not the whole mod."),
|
||||
L("Intro.Agreement.Paragraph3", "If you are on a data capped internet connection, higher fees due to data usage depending on the amount of downloaded and uploaded mod files might occur. Mod files will be compressed on up- and download to save on bandwidth usage. Due to varying up- and download speeds, changes in characters might not be visible immediately. Files present on the service that already represent your active mod files will not be uploaded again."),
|
||||
L("Intro.Agreement.Paragraph4", "The mod files you are uploading are confidential and will not be distributed to parties other than the ones who are requesting the exact same mod files. Please think about who you are going to pair since it is unavoidable that they will receive and locally cache the necessary mod files that you have currently in use. Locally cached mod files will have arbitrary file names to discourage attempts at replicating the original mod."),
|
||||
L("Intro.Agreement.Paragraph5", "The plugin creator tried their best to keep you secure. However, there is no guarantee for 100% security. Do not blindly pair your client with everyone."),
|
||||
L("Intro.Agreement.Paragraph6", "Mod files that are saved on the service will remain on the service as long as there are requests for the files from clients. After a period of not being used, the mod files will be automatically deleted."),
|
||||
L("Intro.Agreement.Paragraph7", "Accounts that are inactive for ninety (90) days will be deleted for privacy reasons."),
|
||||
L("Intro.Agreement.Paragraph8", "Umbra is operated from servers located in the European Union. You agree not to upload any content to the service that violates EU law; and more specifically, German law."),
|
||||
L("Intro.Agreement.Paragraph9", "You may delete your account at any time from within the Settings panel of the plugin. Any mods unique to you will then be removed from the server within 14 days."),
|
||||
L("Intro.Agreement.Paragraph10", "This service is provided as-is."),
|
||||
};
|
||||
|
||||
foreach (var paragraph in agreementParagraphs)
|
||||
{
|
||||
UiSharedService.TextWrapped(paragraph);
|
||||
}
|
||||
|
||||
ImGui.Separator();
|
||||
if (_timeoutTask?.IsCompleted ?? true)
|
||||
{
|
||||
if (ImGui.Button("I agree##toSetup"))
|
||||
if (ImGui.Button(L("Intro.Agreement.Accept", "I agree") + "##toSetup"))
|
||||
{
|
||||
_configService.Current.AcceptedAgreement = true;
|
||||
_configService.Save();
|
||||
@@ -202,29 +190,26 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
|| !Directory.Exists(_configService.Current.CacheFolder)))
|
||||
{
|
||||
using (_uiShared.UidFont.Push())
|
||||
ImGui.TextUnformatted("File Storage Setup");
|
||||
ImGui.TextUnformatted(L("Intro.Storage.Title", "File Storage Setup"));
|
||||
|
||||
ImGui.Separator();
|
||||
|
||||
if (!_uiShared.HasValidPenumbraModPath)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("You do not have a valid Penumbra path set. Open Penumbra and set up a valid path for the mod directory.", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Storage.NoPenumbra", "You do not have a valid Penumbra path set. Open Penumbra and set up a valid path for the mod directory."), ImGuiColors.DalamudRed);
|
||||
}
|
||||
else
|
||||
{
|
||||
UiSharedService.TextWrapped("To not unnecessary download files already present on your computer, Umbra will have to scan your Penumbra mod directory. " +
|
||||
"Additionally, a local storage folder must be set where Umbra will download other character files to. " +
|
||||
"Once the storage folder is set and the scan complete, this page will automatically forward to registration at a service.");
|
||||
UiSharedService.TextWrapped("Note: The initial scan, depending on the amount of mods you have, might take a while. Please wait until it is completed.");
|
||||
UiSharedService.ColorTextWrapped("Warning: once past this step you should not delete the FileCache.csv of Umbra in the Plugin Configurations folder of Dalamud. " +
|
||||
"Otherwise on the next launch a full re-scan of the file cache database will be initiated.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorTextWrapped("Warning: if the scan is hanging and does nothing for a long time, chances are high your Penumbra folder is not set up properly.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.TextWrapped(L("Intro.Storage.Description", "To not unnecessarily download files already present on your computer, Umbra will have to scan your Penumbra mod directory. Additionally, a local storage folder must be set where Umbra will download other character files to. Once the storage folder is set and the scan complete, this page will automatically forward to registration at a service."));
|
||||
UiSharedService.TextWrapped(L("Intro.Storage.ScanNote", "Note: The initial scan, depending on the amount of mods you have, might take a while. Please wait until it is completed."));
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Storage.Warning.FileCache", "Warning: once past this step you should not delete the FileCache.csv of Umbra in the Plugin Configurations folder of Dalamud. Otherwise on the next launch a full re-scan of the file cache database will be initiated."), ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Storage.Warning.ScanHang", "Warning: if the scan is hanging and does nothing for a long time, chances are high your Penumbra folder is not set up properly."), ImGuiColors.DalamudYellow);
|
||||
_uiShared.DrawCacheDirectorySetting();
|
||||
}
|
||||
|
||||
if (!_cacheMonitor.IsScanRunning && !string.IsNullOrEmpty(_configService.Current.CacheFolder) && _uiShared.HasValidPenumbraModPath && Directory.Exists(_configService.Current.CacheFolder))
|
||||
{
|
||||
if (ImGui.Button("Start Scan##startScan"))
|
||||
if (ImGui.Button(L("Intro.Storage.StartScan", "Start Scan") + "##startScan"))
|
||||
{
|
||||
_cacheMonitor.InvokeScan();
|
||||
}
|
||||
@@ -236,22 +221,21 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
if (!_dalamudUtilService.IsWine)
|
||||
{
|
||||
var useFileCompactor = _configService.Current.UseCompactor;
|
||||
if (ImGui.Checkbox("Use File Compactor", ref useFileCompactor))
|
||||
if (ImGui.Checkbox(L("Intro.Storage.UseCompactor", "Use File Compactor"), ref useFileCompactor))
|
||||
{
|
||||
_configService.Current.UseCompactor = useFileCompactor;
|
||||
_configService.Save();
|
||||
}
|
||||
UiSharedService.ColorTextWrapped("The File Compactor can save a tremendeous amount of space on the hard disk for downloads through Umbra. It will incur a minor CPU penalty on download but can speed up " +
|
||||
"loading of other characters. It is recommended to keep it enabled. You can change this setting later anytime in the Umbra settings.", ImGuiColors.DalamudYellow);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Storage.CompactorDescription", "The File Compactor can save a tremendous amount of space on the hard disk for downloads through Umbra. It will incur a minor CPU penalty on download but can speed up loading of other characters. It is recommended to keep it enabled. You can change this setting later anytime in the Umbra settings."), ImGuiColors.DalamudYellow);
|
||||
}
|
||||
}
|
||||
else if (!_uiShared.ApiController.IsConnected)
|
||||
{
|
||||
using (_uiShared.UidFont.Push())
|
||||
ImGui.TextUnformatted("Service Registration");
|
||||
ImGui.TextUnformatted(L("Intro.Registration.Title", "Service Registration"));
|
||||
ImGui.Separator();
|
||||
UiSharedService.TextWrapped("To be able to use Umbra you will have to register an account.");
|
||||
UiSharedService.TextWrapped("Refer to the instructions at the location you obtained this plugin for more information or support.");
|
||||
UiSharedService.TextWrapped(L("Intro.Registration.Description", "To be able to use Umbra you will have to register an account."));
|
||||
UiSharedService.TextWrapped(L("Intro.Registration.Support", "Refer to the instructions at the location you obtained this plugin for more information or support."));
|
||||
|
||||
ImGui.Separator();
|
||||
|
||||
@@ -262,8 +246,8 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
ImGui.BeginDisabled(_registrationInProgress || _registrationSuccess || _secretKey.Length > 0);
|
||||
ImGui.Separator();
|
||||
ImGui.TextUnformatted("If you have not used Umbra before, click below to register a new account.");
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Plus, "Register a new Umbra account"))
|
||||
ImGui.TextUnformatted(L("Intro.Registration.NewAccountInfo", "If you have not used Umbra before, click below to register a new account."));
|
||||
if (_uiShared.IconTextButton(FontAwesomeIcon.Plus, L("Intro.Registration.RegisterButton", "Register a new Umbra account")))
|
||||
{
|
||||
_registrationInProgress = true;
|
||||
_ = Task.Run(async () => {
|
||||
@@ -273,12 +257,12 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
if (!reply.Success)
|
||||
{
|
||||
_logger.LogWarning("Registration failed: {err}", reply.ErrorMessage);
|
||||
_registrationMessage = reply.ErrorMessage;
|
||||
if (_registrationMessage.IsNullOrEmpty())
|
||||
_registrationMessage = "An unknown error occured. Please try again later.";
|
||||
_registrationMessage = string.IsNullOrEmpty(reply.ErrorMessage)
|
||||
? L("Intro.Registration.UnknownError", "An unknown error occured. Please try again later.")
|
||||
: reply.ErrorMessage;
|
||||
return;
|
||||
}
|
||||
_registrationMessage = "New account registered.\nPlease keep a copy of your secret key in case you need to reset your plugins, or to use it on another PC.";
|
||||
_registrationMessage = L("Intro.Registration.Success", "New account registered.\nPlease keep a copy of your secret key in case you need to reset your plugins, or to use it on another PC.");
|
||||
_secretKey = reply.SecretKey ?? "";
|
||||
_registrationReply = reply;
|
||||
_registrationSuccess = true;
|
||||
@@ -287,7 +271,7 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
{
|
||||
_logger.LogWarning(ex, "Registration failed");
|
||||
_registrationSuccess = false;
|
||||
_registrationMessage = "An unknown error occured. Please try again later.";
|
||||
_registrationMessage = L("Intro.Registration.UnknownError", "An unknown error occured. Please try again later.");
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -298,7 +282,7 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
ImGui.EndDisabled(); // _registrationInProgress || _registrationSuccess
|
||||
if (_registrationInProgress)
|
||||
{
|
||||
ImGui.TextUnformatted("Sending request...");
|
||||
ImGui.TextUnformatted(L("Intro.Registration.SendingRequest", "Sending request..."));
|
||||
}
|
||||
else if (!_registrationMessage.IsNullOrEmpty())
|
||||
{
|
||||
@@ -311,42 +295,40 @@ public partial class IntroUi : WindowMediatorSubscriberBase
|
||||
|
||||
ImGui.Separator();
|
||||
|
||||
var text = "Enter Secret Key";
|
||||
var secretKeyLabel = _registrationSuccess
|
||||
? L("Intro.Registration.SecretKeyLabelRegistered", "Secret Key")
|
||||
: L("Intro.Registration.SecretKeyLabel", "Enter Secret Key");
|
||||
|
||||
if (_registrationSuccess)
|
||||
if (!_registrationSuccess)
|
||||
{
|
||||
text = "Secret Key";
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.TextUnformatted("If you already have a registered account, you can enter its secret key below to use it instead.");
|
||||
ImGui.TextUnformatted(L("Intro.Registration.SecretKeyInstructions", "If you already have a registered account, you can enter its secret key below to use it instead."));
|
||||
}
|
||||
|
||||
var textSize = ImGui.CalcTextSize(text);
|
||||
var textSize = ImGui.CalcTextSize(secretKeyLabel);
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(text);
|
||||
ImGui.TextUnformatted(secretKeyLabel);
|
||||
ImGui.SameLine();
|
||||
ImGui.SetNextItemWidth(UiSharedService.GetWindowContentRegionWidth() - ImGui.GetWindowContentRegionMin().X - textSize.X);
|
||||
ImGui.InputText("", ref _secretKey, 64);
|
||||
if (_secretKey.Length > 0 && _secretKey.Length != 64)
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("Your secret key must be exactly 64 characters long.", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Registration.SecretKeyLength", "Your secret key must be exactly 64 characters long."), ImGuiColors.DalamudRed);
|
||||
}
|
||||
else if (_secretKey.Length == 64 && !HexRegex().IsMatch(_secretKey))
|
||||
{
|
||||
UiSharedService.ColorTextWrapped("Your secret key can only contain ABCDEF and the numbers 0-9.", ImGuiColors.DalamudRed);
|
||||
UiSharedService.ColorTextWrapped(L("Intro.Registration.SecretKeyCharacters", "Your secret key can only contain ABCDEF and the numbers 0-9."), ImGuiColors.DalamudRed);
|
||||
}
|
||||
else if (_secretKey.Length == 64)
|
||||
{
|
||||
using var saveDisabled = ImRaii.Disabled(_uiShared.ApiController.ServerState == ServerState.Connecting || _uiShared.ApiController.ServerState == ServerState.Reconnecting);
|
||||
if (ImGui.Button("Save and Connect"))
|
||||
if (ImGui.Button(L("Intro.Registration.SaveAndConnect", "Save and Connect")))
|
||||
{
|
||||
string keyName;
|
||||
if (_serverConfigurationManager.CurrentServer == null) _serverConfigurationManager.SelectServer(0);
|
||||
if (_registrationReply != null && _secretKey.Equals(_registrationReply.SecretKey, StringComparison.Ordinal))
|
||||
keyName = _registrationReply.UID + $" (registered {DateTime.Now:yyyy-MM-dd})";
|
||||
keyName = _registrationReply.UID + " " + L("Intro.Registration.SavedKeyRegistered", "(registered {0})", DateTime.Now.ToString("yyyy-MM-dd"));
|
||||
else
|
||||
keyName = $"Secret Key added on Setup ({DateTime.Now:yyyy-MM-dd})";
|
||||
keyName = L("Intro.Registration.SavedKeySetup", "Secret Key added on Setup ({0})", DateTime.Now.ToString("yyyy-MM-dd"));
|
||||
_serverConfigurationManager.CurrentServer!.SecretKeys.Add(_serverConfigurationManager.CurrentServer.SecretKeys.Select(k => k.Key).LastOrDefault() + 1, new SecretKey()
|
||||
{
|
||||
FriendlyName = keyName,
|
||||
|
||||
@@ -22,7 +22,9 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
|
||||
public PermissionWindowUI(ILogger<PermissionWindowUI> logger, Pair pair, MareMediator mediator, UiSharedService uiSharedService,
|
||||
ApiController apiController, PerformanceCollectorService performanceCollectorService)
|
||||
: base(logger, mediator, "Permissions for " + pair.UserData.AliasOrUID + "###UmbraSyncPermissions" + pair.UserData.UID, performanceCollectorService)
|
||||
: base(logger, mediator,
|
||||
uiSharedService.Localize("PermissionWindow.Title", "Permissions for {0}", pair.UserData.AliasOrUID) + "###UmbraSyncPermissions" + pair.UserData.UID,
|
||||
performanceCollectorService)
|
||||
{
|
||||
Pair = pair;
|
||||
_uiSharedService = uiSharedService;
|
||||
@@ -37,6 +39,8 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
IsOpen = true;
|
||||
}
|
||||
|
||||
private string L(string key, string fallback, params object[] args) => _uiSharedService.Localize(key, fallback, args);
|
||||
|
||||
protected override void DrawInternal()
|
||||
{
|
||||
var paused = _ownPermissions.IsPaused();
|
||||
@@ -46,18 +50,19 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
var style = ImGui.GetStyle();
|
||||
var indentSize = ImGui.GetFrameHeight() + style.ItemSpacing.X;
|
||||
|
||||
_uiSharedService.BigText("Permissions for " + Pair.UserData.AliasOrUID);
|
||||
_uiSharedService.BigText(L("PermissionWindow.Title", "Permissions for {0}", Pair.UserData.AliasOrUID));
|
||||
ImGuiHelpers.ScaledDummy(1f);
|
||||
|
||||
if (Pair.UserPair == null)
|
||||
return;
|
||||
|
||||
if (ImGui.Checkbox("Pause Sync", ref paused))
|
||||
if (ImGui.Checkbox(L("PermissionWindow.Pause.Label", "Pause Sync"), ref paused))
|
||||
{
|
||||
_ownPermissions.SetPaused(paused);
|
||||
}
|
||||
_uiSharedService.DrawHelpText("Pausing will completely cease any sync with this user." + UiSharedService.TooltipSeparator
|
||||
+ "Note: this is bidirectional, either user pausing will cease sync completely.");
|
||||
_uiSharedService.DrawHelpText(L("PermissionWindow.Pause.HelpMain", "Pausing will completely cease any sync with this user.")
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("PermissionWindow.Pause.HelpNote", "Note: this is bidirectional, either user pausing will cease sync completely."));
|
||||
var otherPerms = Pair.UserPair.OtherPermissions;
|
||||
|
||||
var otherIsPaused = otherPerms.IsPaused();
|
||||
@@ -70,53 +75,68 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
_uiSharedService.BooleanToColoredIcon(!otherIsPaused, false);
|
||||
ImGui.SameLine();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(Pair.UserData.AliasOrUID + " has " + (!otherIsPaused ? "not " : string.Empty) + "paused you");
|
||||
var pausedText = otherIsPaused
|
||||
? L("PermissionWindow.OtherPaused.True", "{0} has paused you", Pair.UserData.AliasOrUID)
|
||||
: L("PermissionWindow.OtherPaused.False", "{0} has not paused you", Pair.UserData.AliasOrUID);
|
||||
ImGui.TextUnformatted(pausedText);
|
||||
}
|
||||
|
||||
ImGuiHelpers.ScaledDummy(0.5f);
|
||||
ImGui.Separator();
|
||||
ImGuiHelpers.ScaledDummy(0.5f);
|
||||
|
||||
if (ImGui.Checkbox("Disable Sounds", ref disableSounds))
|
||||
if (ImGui.Checkbox(L("PermissionWindow.Sounds.Label", "Disable Sounds"), ref disableSounds))
|
||||
{
|
||||
_ownPermissions.SetDisableSounds(disableSounds);
|
||||
}
|
||||
_uiSharedService.DrawHelpText("Disabling sounds will remove all sounds synced with this user on both sides." + UiSharedService.TooltipSeparator
|
||||
+ "Note: this is bidirectional, either user disabling sound sync will stop sound sync on both sides.");
|
||||
_uiSharedService.DrawHelpText(L("PermissionWindow.Sounds.HelpMain", "Disabling sounds will remove all sounds synced with this user on both sides.")
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("PermissionWindow.Sounds.HelpNote", "Note: this is bidirectional, either user disabling sound sync will stop sound sync on both sides."));
|
||||
using (ImRaii.PushIndent(indentSize, false))
|
||||
{
|
||||
_uiSharedService.BooleanToColoredIcon(!otherDisableSounds, false);
|
||||
ImGui.SameLine();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(Pair.UserData.AliasOrUID + " has " + (!otherDisableSounds ? "not " : string.Empty) + "disabled sound sync with you");
|
||||
var soundText = otherDisableSounds
|
||||
? L("PermissionWindow.OtherSoundDisabled.True", "{0} has disabled sound sync with you", Pair.UserData.AliasOrUID)
|
||||
: L("PermissionWindow.OtherSoundDisabled.False", "{0} has not disabled sound sync with you", Pair.UserData.AliasOrUID);
|
||||
ImGui.TextUnformatted(soundText);
|
||||
}
|
||||
|
||||
if (ImGui.Checkbox("Disable Animations", ref disableAnimations))
|
||||
if (ImGui.Checkbox(L("PermissionWindow.Animations.Label", "Disable Animations"), ref disableAnimations))
|
||||
{
|
||||
_ownPermissions.SetDisableAnimations(disableAnimations);
|
||||
}
|
||||
_uiSharedService.DrawHelpText("Disabling sounds will remove all animations synced with this user on both sides." + UiSharedService.TooltipSeparator
|
||||
+ "Note: this is bidirectional, either user disabling animation sync will stop animation sync on both sides.");
|
||||
_uiSharedService.DrawHelpText(L("PermissionWindow.Animations.HelpMain", "Disabling sounds will remove all animations synced with this user on both sides.")
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("PermissionWindow.Animations.HelpNote", "Note: this is bidirectional, either user disabling animation sync will stop animation sync on both sides."));
|
||||
using (ImRaii.PushIndent(indentSize, false))
|
||||
{
|
||||
_uiSharedService.BooleanToColoredIcon(!otherDisableAnimations, false);
|
||||
ImGui.SameLine();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(Pair.UserData.AliasOrUID + " has " + (!otherDisableAnimations ? "not " : string.Empty) + "disabled animation sync with you");
|
||||
var animText = otherDisableAnimations
|
||||
? L("PermissionWindow.OtherAnimationDisabled.True", "{0} has disabled animation sync with you", Pair.UserData.AliasOrUID)
|
||||
: L("PermissionWindow.OtherAnimationDisabled.False", "{0} has not disabled animation sync with you", Pair.UserData.AliasOrUID);
|
||||
ImGui.TextUnformatted(animText);
|
||||
}
|
||||
|
||||
if (ImGui.Checkbox("Disable VFX", ref disableVfx))
|
||||
if (ImGui.Checkbox(L("PermissionWindow.Vfx.Label", "Disable VFX"), ref disableVfx))
|
||||
{
|
||||
_ownPermissions.SetDisableVFX(disableVfx);
|
||||
}
|
||||
_uiSharedService.DrawHelpText("Disabling sounds will remove all VFX synced with this user on both sides." + UiSharedService.TooltipSeparator
|
||||
+ "Note: this is bidirectional, either user disabling VFX sync will stop VFX sync on both sides.");
|
||||
_uiSharedService.DrawHelpText(L("PermissionWindow.Vfx.HelpMain", "Disabling sounds will remove all VFX synced with this user on both sides.")
|
||||
+ UiSharedService.TooltipSeparator
|
||||
+ L("PermissionWindow.Vfx.HelpNote", "Note: this is bidirectional, either user disabling VFX sync will stop VFX sync on both sides."));
|
||||
using (ImRaii.PushIndent(indentSize, false))
|
||||
{
|
||||
_uiSharedService.BooleanToColoredIcon(!otherDisableVFX, false);
|
||||
ImGui.SameLine();
|
||||
ImGui.AlignTextToFramePadding();
|
||||
ImGui.TextUnformatted(Pair.UserData.AliasOrUID + " has " + (!otherDisableVFX ? "not " : string.Empty) + "disabled VFX sync with you");
|
||||
var vfxText = otherDisableVFX
|
||||
? L("PermissionWindow.OtherVfxDisabled.True", "{0} has disabled VFX sync with you", Pair.UserData.AliasOrUID)
|
||||
: L("PermissionWindow.OtherVfxDisabled.False", "{0} has not disabled VFX sync with you", Pair.UserData.AliasOrUID);
|
||||
ImGui.TextUnformatted(vfxText);
|
||||
}
|
||||
|
||||
ImGuiHelpers.ScaledDummy(0.5f);
|
||||
@@ -126,27 +146,27 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
bool hasChanges = _ownPermissions != Pair.UserPair.OwnPermissions;
|
||||
|
||||
using (ImRaii.Disabled(!hasChanges))
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.Save, "Save"))
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.Save, L("PermissionWindow.Button.Save", "Save")))
|
||||
{
|
||||
_ = _apiController.UserSetPairPermissions(new(Pair.UserData, _ownPermissions));
|
||||
}
|
||||
UiSharedService.AttachToolTip("Save and apply all changes");
|
||||
UiSharedService.AttachToolTip(L("PermissionWindow.Tooltip.Save", "Save and apply all changes"));
|
||||
|
||||
var rightSideButtons = _uiSharedService.GetIconTextButtonSize(Dalamud.Interface.FontAwesomeIcon.Undo, "Revert") +
|
||||
_uiSharedService.GetIconTextButtonSize(Dalamud.Interface.FontAwesomeIcon.ArrowsSpin, "Reset to Default");
|
||||
var rightSideButtons = _uiSharedService.GetIconTextButtonSize(Dalamud.Interface.FontAwesomeIcon.Undo, L("PermissionWindow.Button.Revert", "Revert")) +
|
||||
_uiSharedService.GetIconTextButtonSize(Dalamud.Interface.FontAwesomeIcon.ArrowsSpin, L("PermissionWindow.Button.Reset", "Reset to Default"));
|
||||
var availableWidth = ImGui.GetWindowContentRegionMax().X - ImGui.GetWindowContentRegionMin().X;
|
||||
|
||||
ImGui.SameLine(availableWidth - rightSideButtons);
|
||||
|
||||
using (ImRaii.Disabled(!hasChanges))
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.Undo, "Revert"))
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.Undo, L("PermissionWindow.Button.Revert", "Revert")))
|
||||
{
|
||||
_ownPermissions = Pair.UserPair.OwnPermissions.DeepClone();
|
||||
}
|
||||
UiSharedService.AttachToolTip("Revert all changes");
|
||||
UiSharedService.AttachToolTip(L("PermissionWindow.Tooltip.Revert", "Revert all changes"));
|
||||
|
||||
ImGui.SameLine();
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.ArrowsSpin, "Reset to Default"))
|
||||
if (_uiSharedService.IconTextButton(Dalamud.Interface.FontAwesomeIcon.ArrowsSpin, L("PermissionWindow.Button.Reset", "Reset to Default")))
|
||||
{
|
||||
_ownPermissions.SetPaused(false);
|
||||
_ownPermissions.SetDisableVFX(false);
|
||||
@@ -154,7 +174,7 @@ public class PermissionWindowUI : WindowMediatorSubscriberBase
|
||||
_ownPermissions.SetDisableAnimations(false);
|
||||
_ = _apiController.UserSetPairPermissions(new(Pair.UserData, _ownPermissions));
|
||||
}
|
||||
UiSharedService.AttachToolTip("This will set all permissions to their default setting");
|
||||
UiSharedService.AttachToolTip(L("PermissionWindow.Tooltip.Reset", "This will set all permissions to their default setting"));
|
||||
|
||||
var ySize = ImGui.GetCursorPosY() + style.FramePadding.Y * ImGuiHelpers.GlobalScale + style.FrameBorderSize;
|
||||
ImGui.SetWindowSize(new(400, ySize));
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -12,6 +12,7 @@ using Dalamud.Plugin.Services;
|
||||
using Dalamud.Utility;
|
||||
using MareSynchronos.FileCache;
|
||||
using MareSynchronos.Interop.Ipc;
|
||||
using MareSynchronos.Localization;
|
||||
using MareSynchronos.MareConfiguration;
|
||||
using MareSynchronos.MareConfiguration.Models;
|
||||
using MareSynchronos.PlayerData.Pairs;
|
||||
@@ -20,6 +21,7 @@ using MareSynchronos.Services.Mediator;
|
||||
using MareSynchronos.Services.ServerConfiguration;
|
||||
using MareSynchronos.WebAPI;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using System.Globalization;
|
||||
using System.Numerics;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
@@ -36,7 +38,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGuiWindowFlags.NoScrollbar |
|
||||
ImGuiWindowFlags.NoScrollWithMouse;
|
||||
|
||||
public static Vector4 AccentColor { get; set; } = ImGuiColors.DalamudYellow;
|
||||
public static Vector4 AccentColor { get; set; } = ImGuiColors.DalamudViolet;
|
||||
|
||||
public readonly FileDialogManager FileDialogManager;
|
||||
|
||||
@@ -53,6 +55,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
private readonly DalamudUtilService _dalamudUtil;
|
||||
private readonly IpcManager _ipcManager;
|
||||
private readonly IDalamudPluginInterface _pluginInterface;
|
||||
private readonly LocalizationService _localizationService;
|
||||
private readonly ITextureProvider _textureProvider;
|
||||
private readonly Dictionary<string, object> _selectedComboItems = new(StringComparer.Ordinal);
|
||||
private readonly ServerConfigurationManager _serverConfigurationManager;
|
||||
@@ -84,7 +87,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
public UiSharedService(ILogger<UiSharedService> logger, IpcManager ipcManager, ApiController apiController,
|
||||
CacheMonitor cacheMonitor, FileDialogManager fileDialogManager,
|
||||
MareConfigService configService, DalamudUtilService dalamudUtil, IDalamudPluginInterface pluginInterface,
|
||||
ITextureProvider textureProvider,
|
||||
LocalizationService localizationService, ITextureProvider textureProvider,
|
||||
ServerConfigurationManager serverManager, MareMediator mediator) : base(logger, mediator)
|
||||
{
|
||||
_ipcManager = ipcManager;
|
||||
@@ -94,6 +97,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
_configService = configService;
|
||||
_dalamudUtil = dalamudUtil;
|
||||
_pluginInterface = pluginInterface;
|
||||
_localizationService = localizationService;
|
||||
_textureProvider = textureProvider;
|
||||
_serverConfigurationManager = serverManager;
|
||||
|
||||
@@ -124,6 +128,10 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
}
|
||||
|
||||
public ApiController ApiController => _apiController;
|
||||
public LocalizationService Localization => _localizationService;
|
||||
|
||||
public string Localize(string key, string fallback, params object[] args) => _localizationService.GetString(key, fallback, args);
|
||||
public string Localize(string fallback, params object[] args) => _localizationService.GetString(fallback, args);
|
||||
|
||||
public bool EditTrackerPosition { get; set; }
|
||||
|
||||
@@ -310,7 +318,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
}
|
||||
}
|
||||
|
||||
public static Vector4 GetBoolColor(bool input) => input ? ImGuiColors.ParsedGreen : ImGuiColors.DalamudRed;
|
||||
public static Vector4 GetBoolColor(bool input) => input ? AccentColor : ImGuiColors.DalamudRed;
|
||||
|
||||
public float GetIconTextButtonSize(FontAwesomeIcon icon, string text)
|
||||
{
|
||||
@@ -517,7 +525,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
|
||||
public void BooleanToColoredIcon(bool value, bool inline = true)
|
||||
{
|
||||
using var colorgreen = ImRaii.PushColor(ImGuiCol.Text, ImGuiColors.HealerGreen, value);
|
||||
using var colorgreen = ImRaii.PushColor(ImGuiCol.Text, AccentColor, value);
|
||||
using var colorred = ImRaii.PushColor(ImGuiCol.Text, ImGuiColors.DalamudRed, !value);
|
||||
|
||||
if (inline) ImGui.SameLine();
|
||||
@@ -761,15 +769,19 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
var check = FontAwesomeIcon.Check;
|
||||
var cross = FontAwesomeIcon.SquareXmark;
|
||||
|
||||
var availableTemplate = Localize("Settings.Plugins.Tooltip.Available", "{0} is available and up to date.");
|
||||
var unavailableTemplate = Localize("Settings.Plugins.Tooltip.Unavailable", "{0} is unavailable or not up to date.");
|
||||
string FormatTooltip(bool exists, string pluginName) => string.Format(CultureInfo.CurrentCulture, exists ? availableTemplate : unavailableTemplate, pluginName);
|
||||
|
||||
if (intro)
|
||||
{
|
||||
ImGui.SetWindowFontScale(0.8f);
|
||||
BigText("Mandatory Plugins");
|
||||
BigText(Localize("Settings.Plugins.MandatoryHeading", "Mandatory Plugins"));
|
||||
ImGui.SetWindowFontScale(1.0f);
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.TextUnformatted("Mandatory Plugins:");
|
||||
ImGui.TextUnformatted(Localize("Settings.Plugins.MandatoryLabel", "Mandatory Plugins:"));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
|
||||
@@ -777,23 +789,23 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_penumbraExists ? check : cross, GetBoolColor(_penumbraExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"Penumbra is " + (_penumbraExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_penumbraExists, "Penumbra"));
|
||||
|
||||
ImGui.TextUnformatted("Glamourer");
|
||||
ImGui.SameLine();
|
||||
IconText(_glamourerExists ? check : cross, GetBoolColor(_glamourerExists));
|
||||
AttachToolTip($"Glamourer is " + (_glamourerExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_glamourerExists, "Glamourer"));
|
||||
|
||||
if (intro)
|
||||
{
|
||||
ImGui.SetWindowFontScale(0.8f);
|
||||
BigText("Optional Addons");
|
||||
BigText(Localize("Settings.Plugins.OptionalHeading", "Optional Addons"));
|
||||
ImGui.SetWindowFontScale(1.0f);
|
||||
UiSharedService.TextWrapped("These addons are not required for basic operation, but without them you may not see others as intended.");
|
||||
UiSharedService.TextWrapped(Localize("Settings.Plugins.OptionalDescription", "These addons are not required for basic operation, but without them you may not see others as intended."));
|
||||
}
|
||||
else
|
||||
{
|
||||
ImGui.TextUnformatted("Optional Addons:");
|
||||
ImGui.TextUnformatted(Localize("Settings.Plugins.OptionalLabel", "Optional Addons:"));
|
||||
ImGui.SameLine();
|
||||
}
|
||||
|
||||
@@ -803,7 +815,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_heelsExists ? check : cross, GetBoolColor(_heelsExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"SimpleHeels is " + (_heelsExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_heelsExists, "SimpleHeels"));
|
||||
ImGui.Spacing();
|
||||
|
||||
ImGui.SameLine();
|
||||
@@ -811,7 +823,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_customizePlusExists ? check : cross, GetBoolColor(_customizePlusExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"Customize+ is " + (_customizePlusExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_customizePlusExists, "Customize+"));
|
||||
ImGui.Spacing();
|
||||
|
||||
ImGui.SameLine();
|
||||
@@ -819,7 +831,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_honorificExists ? check : cross, GetBoolColor(_honorificExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"Honorific is " + (_honorificExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_honorificExists, "Honorific"));
|
||||
ImGui.Spacing();
|
||||
|
||||
ImGui.SameLine();
|
||||
@@ -827,7 +839,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_petNamesExists ? check : cross, GetBoolColor(_petNamesExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"PetNicknames is " + (_petNamesExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_petNamesExists, "PetNicknames"));
|
||||
ImGui.Spacing();
|
||||
|
||||
ImGui.SetCursorPosX(alignPos);
|
||||
@@ -835,7 +847,7 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_moodlesExists ? check : cross, GetBoolColor(_moodlesExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"Moodles is " + (_moodlesExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_moodlesExists, "Moodles"));
|
||||
ImGui.Spacing();
|
||||
|
||||
ImGui.SameLine();
|
||||
@@ -843,12 +855,12 @@ public partial class UiSharedService : DisposableMediatorSubscriberBase
|
||||
ImGui.SameLine();
|
||||
IconText(_brioExists ? check : cross, GetBoolColor(_brioExists));
|
||||
ImGui.SameLine();
|
||||
AttachToolTip($"Brio is " + (_moodlesExists ? "available and up to date." : "unavailable or not up to date."));
|
||||
AttachToolTip(FormatTooltip(_brioExists, "Brio"));
|
||||
ImGui.Spacing();
|
||||
|
||||
if (!_penumbraExists || !_glamourerExists)
|
||||
{
|
||||
ImGui.TextColored(ImGuiColors.DalamudRed, "You need to install both Penumbra and Glamourer and keep them up to date to use Umbra.");
|
||||
ImGui.TextColored(ImGuiColors.DalamudRed, Localize("Settings.Plugins.WarningMandatoryMissing", "You need to install both Penumbra and Glamourer and keep them up to date to use Umbra."));
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
@@ -1,14 +1,14 @@
|
||||
{
|
||||
"Author": "SirConstance",
|
||||
"Author": "Keda",
|
||||
"Name": "UmbraSync",
|
||||
"Punchline": "Share your true self.",
|
||||
"Description": "This plugin will synchronize your Penumbra mods and current Glamourer state with other paired clients automatically.",
|
||||
"Punchline": "Parce que nous le valons bien.",
|
||||
"Description": "Ce plugin synchronisera automatiquement vos mods Penumbra et l'état actuel de Glamourer avec les autres clients appairés.",
|
||||
"InternalName": "UmbraSync",
|
||||
"ApplicableVersion": "any",
|
||||
"Tags": [
|
||||
"customization"
|
||||
],
|
||||
"IconUrl": "https://repo.umbra-sync.net/logo.png",
|
||||
"IconUrl": "https://repo.umbra-sync.net/images/logo.png",
|
||||
"RepoUrl": "https://repo.umbra-sync.net/plugin.json",
|
||||
"CanUnloadAsync": true
|
||||
}
|
||||
|
||||
209
MareSynchronos/WebAPI/AutoDetect/DiscoveryApiClient.cs
Normal file
209
MareSynchronos/WebAPI/AutoDetect/DiscoveryApiClient.cs
Normal file
@@ -0,0 +1,209 @@
|
||||
using System.Net.Http.Headers;
|
||||
using System.Text;
|
||||
using System.Text.Json;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using MareSynchronos.WebAPI.SignalR;
|
||||
using MareSynchronos.Services.AutoDetect;
|
||||
|
||||
namespace MareSynchronos.WebAPI.AutoDetect;
|
||||
|
||||
public class DiscoveryApiClient
|
||||
{
|
||||
private readonly ILogger<DiscoveryApiClient> _logger;
|
||||
private readonly TokenProvider _tokenProvider;
|
||||
private readonly DiscoveryConfigProvider _configProvider;
|
||||
private readonly HttpClient _httpClient = new();
|
||||
private static readonly JsonSerializerOptions JsonOpt = new() { PropertyNameCaseInsensitive = true };
|
||||
|
||||
public DiscoveryApiClient(ILogger<DiscoveryApiClient> logger, TokenProvider tokenProvider, DiscoveryConfigProvider configProvider)
|
||||
{
|
||||
_logger = logger;
|
||||
_tokenProvider = tokenProvider;
|
||||
_configProvider = configProvider;
|
||||
_httpClient.Timeout = TimeSpan.FromSeconds(30);
|
||||
}
|
||||
|
||||
public async Task<List<ServerMatch>> QueryAsync(string endpoint, IEnumerable<string> hashes, CancellationToken ct)
|
||||
{
|
||||
try
|
||||
{
|
||||
var token = await _tokenProvider.GetOrUpdateToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(token)) return [];
|
||||
var distinctHashes = hashes.Distinct(StringComparer.Ordinal).ToArray();
|
||||
using var req = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
||||
var body = JsonSerializer.Serialize(new
|
||||
{
|
||||
hashes = distinctHashes,
|
||||
salt = _configProvider.SaltB64
|
||||
});
|
||||
req.Content = new StringContent(body, Encoding.UTF8, "application/json");
|
||||
var resp = await _httpClient.SendAsync(req, ct).ConfigureAwait(false);
|
||||
if (resp.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
||||
{
|
||||
var token2 = await _tokenProvider.ForceRefreshToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(token2)) return [];
|
||||
using var req2 = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req2.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token2);
|
||||
var body2 = JsonSerializer.Serialize(new
|
||||
{
|
||||
hashes = distinctHashes,
|
||||
salt = _configProvider.SaltB64
|
||||
});
|
||||
req2.Content = new StringContent(body2, Encoding.UTF8, "application/json");
|
||||
resp = await _httpClient.SendAsync(req2, ct).ConfigureAwait(false);
|
||||
}
|
||||
resp.EnsureSuccessStatusCode();
|
||||
var json = await resp.Content.ReadAsStringAsync(ct).ConfigureAwait(false);
|
||||
var result = JsonSerializer.Deserialize<List<ServerMatch>>(json, JsonOpt) ?? [];
|
||||
return result;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Discovery query failed");
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
public async Task<bool> SendRequestAsync(string endpoint, string token, string? displayName, CancellationToken ct)
|
||||
{
|
||||
try
|
||||
{
|
||||
var jwt = await _tokenProvider.GetOrUpdateToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt)) return false;
|
||||
using var req = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt);
|
||||
var body = JsonSerializer.Serialize(new { token, displayName });
|
||||
req.Content = new StringContent(body, Encoding.UTF8, "application/json");
|
||||
var resp = await _httpClient.SendAsync(req, ct).ConfigureAwait(false);
|
||||
if (resp.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
||||
{
|
||||
var jwt2 = await _tokenProvider.ForceRefreshToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt2)) return false;
|
||||
using var req2 = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req2.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt2);
|
||||
var body2 = JsonSerializer.Serialize(new { token, displayName });
|
||||
req2.Content = new StringContent(body2, Encoding.UTF8, "application/json");
|
||||
resp = await _httpClient.SendAsync(req2, ct).ConfigureAwait(false);
|
||||
}
|
||||
if (!resp.IsSuccessStatusCode)
|
||||
{
|
||||
string txt = string.Empty;
|
||||
try { txt = await resp.Content.ReadAsStringAsync(ct).ConfigureAwait(false); } catch { }
|
||||
_logger.LogWarning("Discovery request failed: {code} {reason} {body}", (int)resp.StatusCode, resp.ReasonPhrase, txt);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Discovery send request failed");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task<bool> PublishAsync(string endpoint, IEnumerable<string> hashes, string? displayName, CancellationToken ct, bool allowRequests = true)
|
||||
{
|
||||
try
|
||||
{
|
||||
var jwt = await _tokenProvider.GetOrUpdateToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt)) return false;
|
||||
using var req = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt);
|
||||
var bodyObj = new
|
||||
{
|
||||
hashes = hashes.Distinct(StringComparer.Ordinal).ToArray(),
|
||||
displayName,
|
||||
salt = _configProvider.SaltB64,
|
||||
allowRequests
|
||||
};
|
||||
var body = JsonSerializer.Serialize(bodyObj);
|
||||
req.Content = new StringContent(body, Encoding.UTF8, "application/json");
|
||||
var resp = await _httpClient.SendAsync(req, ct).ConfigureAwait(false);
|
||||
if (resp.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
||||
{
|
||||
var jwt2 = await _tokenProvider.ForceRefreshToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt2)) return false;
|
||||
using var req2 = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req2.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt2);
|
||||
var body2 = JsonSerializer.Serialize(bodyObj);
|
||||
req2.Content = new StringContent(body2, Encoding.UTF8, "application/json");
|
||||
resp = await _httpClient.SendAsync(req2, ct).ConfigureAwait(false);
|
||||
}
|
||||
return resp.IsSuccessStatusCode;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Discovery publish failed");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task<bool> SendAcceptAsync(string endpoint, string targetUid, string? displayName, CancellationToken ct)
|
||||
{
|
||||
try
|
||||
{
|
||||
var jwt = await _tokenProvider.GetOrUpdateToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt)) return false;
|
||||
using var req = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt);
|
||||
var bodyObj = new { targetUid, displayName };
|
||||
var body = JsonSerializer.Serialize(bodyObj);
|
||||
req.Content = new StringContent(body, Encoding.UTF8, "application/json");
|
||||
var resp = await _httpClient.SendAsync(req, ct).ConfigureAwait(false);
|
||||
if (resp.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
||||
{
|
||||
var jwt2 = await _tokenProvider.ForceRefreshToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt2)) return false;
|
||||
using var req2 = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req2.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt2);
|
||||
var body2 = JsonSerializer.Serialize(bodyObj);
|
||||
req2.Content = new StringContent(body2, Encoding.UTF8, "application/json");
|
||||
resp = await _httpClient.SendAsync(req2, ct).ConfigureAwait(false);
|
||||
}
|
||||
return resp.IsSuccessStatusCode;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Discovery accept notify failed");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
public async Task DisableAsync(string endpoint, CancellationToken ct)
|
||||
{
|
||||
try
|
||||
{
|
||||
var jwt = await _tokenProvider.GetOrUpdateToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt)) return;
|
||||
using var req = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt);
|
||||
var resp = await _httpClient.SendAsync(req, ct).ConfigureAwait(false);
|
||||
if (resp.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
||||
{
|
||||
var jwt2 = await _tokenProvider.ForceRefreshToken(ct).ConfigureAwait(false);
|
||||
if (string.IsNullOrEmpty(jwt2)) return;
|
||||
using var req2 = new HttpRequestMessage(HttpMethod.Post, endpoint);
|
||||
req2.Headers.Authorization = new AuthenticationHeaderValue("Bearer", jwt2);
|
||||
resp = await _httpClient.SendAsync(req2, ct).ConfigureAwait(false);
|
||||
}
|
||||
if (!resp.IsSuccessStatusCode)
|
||||
{
|
||||
string txt = string.Empty;
|
||||
try { txt = await resp.Content.ReadAsStringAsync(ct).ConfigureAwait(false); } catch { }
|
||||
_logger.LogWarning("Discovery disable failed: {code} {reason} {body}", (int)resp.StatusCode, resp.ReasonPhrase, txt);
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, "Discovery disable failed");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public sealed class ServerMatch
|
||||
{
|
||||
public string Hash { get; set; } = string.Empty;
|
||||
public string? Token { get; set; }
|
||||
public string? Uid { get; set; }
|
||||
public string? DisplayName { get; set; }
|
||||
}
|
||||
@@ -49,7 +49,7 @@ public partial class FileDownloadManager : DisposableMediatorSubscriberBase
|
||||
|
||||
public List<FileTransfer> ForbiddenTransfers => _orchestrator.ForbiddenTransfers;
|
||||
|
||||
public bool IsDownloading => !CurrentDownloads.Any();
|
||||
public bool IsDownloading => CurrentDownloads.Any();
|
||||
|
||||
public void ClearDownload()
|
||||
{
|
||||
@@ -507,4 +507,4 @@ public partial class FileDownloadManager : DisposableMediatorSubscriberBase
|
||||
_orchestrator.ClearDownloadRequest(requestId);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -49,10 +49,10 @@ public partial class ApiController
|
||||
await _mareHub!.SendAsync(nameof(GroupClear), group).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
public async Task<GroupPasswordDto> GroupCreate()
|
||||
public async Task<GroupPasswordDto> GroupCreate(string? alias = null)
|
||||
{
|
||||
CheckConnection();
|
||||
return await _mareHub!.InvokeAsync<GroupPasswordDto>(nameof(GroupCreate)).ConfigureAwait(false);
|
||||
return await _mareHub!.InvokeAsync<GroupPasswordDto>(nameof(GroupCreate), string.IsNullOrWhiteSpace(alias) ? null : alias.Trim()).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
public async Task<List<string>> GroupCreateTempInvite(GroupDto group, int amount)
|
||||
@@ -125,4 +125,4 @@ public partial class ApiController
|
||||
{
|
||||
if (ServerState is not (ServerState.Connected or ServerState.Connecting or ServerState.Reconnecting)) throw new InvalidDataException("Not connected");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -222,7 +222,7 @@ public sealed partial class ApiController : DisposableMediatorSubscriberBase, IM
|
||||
#endif
|
||||
}
|
||||
|
||||
await LoadIninitialPairs().ConfigureAwait(false);
|
||||
await LoadInitialPairs().ConfigureAwait(false);
|
||||
await LoadOnlinePairs().ConfigureAwait(false);
|
||||
}
|
||||
catch (OperationCanceledException)
|
||||
@@ -375,7 +375,7 @@ public sealed partial class ApiController : DisposableMediatorSubscriberBase, IM
|
||||
_initialized = true;
|
||||
}
|
||||
|
||||
private async Task LoadIninitialPairs()
|
||||
private async Task LoadInitialPairs()
|
||||
{
|
||||
foreach (var userPair in await UserGetPairedClients().ConfigureAwait(false))
|
||||
{
|
||||
@@ -435,7 +435,7 @@ public sealed partial class ApiController : DisposableMediatorSubscriberBase, IM
|
||||
return;
|
||||
}
|
||||
ServerState = ServerState.Connected;
|
||||
await LoadIninitialPairs().ConfigureAwait(false);
|
||||
await LoadInitialPairs().ConfigureAwait(false);
|
||||
await LoadOnlinePairs().ConfigureAwait(false);
|
||||
Mediator.Publish(new ConnectedMessage(_connectionDto));
|
||||
}
|
||||
|
||||
@@ -172,6 +172,16 @@ public sealed class TokenProvider : IDisposable, IMediatorSubscriber
|
||||
return await GetNewToken(jwtIdentifier, ct).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
public async Task<string?> ForceRefreshToken(CancellationToken ct)
|
||||
{
|
||||
JwtIdentifier? jwtIdentifier = await GetIdentifier().ConfigureAwait(false);
|
||||
if (jwtIdentifier == null) return null;
|
||||
|
||||
_tokenCache.TryRemove(jwtIdentifier, out _);
|
||||
_logger.LogTrace("ForceRefresh: Getting new token");
|
||||
return await GetNewToken(jwtIdentifier, ct).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
public string? GetStapledWellKnown(string apiUrl)
|
||||
{
|
||||
_wellKnownCache.TryGetValue(apiUrl, out var wellKnown);
|
||||
@@ -180,4 +190,4 @@ public sealed class TokenProvider : IDisposable, IMediatorSubscriber
|
||||
return null;
|
||||
return wellKnown;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
1
Penumbra.Api
Submodule
1
Penumbra.Api
Submodule
Submodule Penumbra.Api added at dd14131793
File diff suppressed because it is too large
Load Diff
3
Penumbra.Api/.gitignore
vendored
3
Penumbra.Api/.gitignore
vendored
@@ -1,3 +0,0 @@
|
||||
bin/
|
||||
obj/
|
||||
.vs/
|
||||
@@ -1,66 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to collection management. </summary>
|
||||
public interface IPenumbraApiCollection
|
||||
{
|
||||
/// <returns> A list of the GUIDs of all currently installed collections together with their display names, excluding the empty collection. </returns>
|
||||
public Dictionary<Guid, string> GetCollections();
|
||||
|
||||
/// <summary> Returns all collections for which either
|
||||
/// <list type="number">
|
||||
/// <item> the name is equal to the given identifier up to case, </item>
|
||||
/// <item> the identifier is parsable to a GUID and the GUID corresponds to an existing collection, </item>
|
||||
/// <item> or the identifier is at least 8 characters long and the GUID as a hex-string starts with the identifier. </item>
|
||||
/// </list>
|
||||
/// </summary>
|
||||
public List<(Guid Id, string Name)> GetCollectionsByIdentifier(string identifier);
|
||||
|
||||
/// <returns>A dictionary of affected items in <paramref name="collectionId"/> via GUID and known objects or null.</returns>
|
||||
public Dictionary<string, object?> GetChangedItemsForCollection(Guid collectionId);
|
||||
|
||||
/// <returns> The GUID and name of the collection assigned to the given <paramref name="type"/>, the empty GUID for the empty collection, or null if nothing is assigned. </returns>
|
||||
public (Guid Id, string Name)? GetCollection(ApiCollectionType type);
|
||||
|
||||
/// <returns>Return whether the object at <paramref name="gameObjectIdx" /> produces a valid identifier, if the identifier has a collection assigned, and the collection that affects the object.</returns>
|
||||
public (bool ObjectValid, bool IndividualSet, (Guid Id, string Name) EffectiveCollection) GetCollectionForObject(int gameObjectIdx);
|
||||
|
||||
/// <summary>
|
||||
/// Set a collection by GUID for a specific type.
|
||||
/// </summary>
|
||||
/// <param name="type">The collection type to set.</param>
|
||||
/// <param name="collectionId">The GUID of the collection to set it to, null to remove the association if allowed. </param>
|
||||
/// <param name="allowCreateNew">Allow only setting existing types or also creating an unset type.</param>
|
||||
/// <param name="allowDelete">Allow deleting existing collections if <paramref name="collectionId"/> is empty.</param>
|
||||
/// <returns>InvalidArgument if type is invalid,
|
||||
/// NothingChanged if the new collection is the same as the old,<br />
|
||||
/// AssignmentDeletionDisallowed if <paramref name="collectionId"/> is null and <paramref name="allowDelete"/> is false, and the assignment exists,<br />
|
||||
/// or if Default, Current or Interface would be deleted.<br />
|
||||
/// CollectionMissing if the new collection can not be found,<br />
|
||||
/// AssignmentCreationDisallowed if <paramref name="allowCreateNew"/> is false and the assignment does not exist,<br />
|
||||
/// or Success, as well as the GUID of the previous collection (empty if no assignment existed).
|
||||
/// </returns>
|
||||
public (PenumbraApiEc, (Guid Id, string Name)? OldCollection) SetCollection(ApiCollectionType type, Guid? collectionId, bool allowCreateNew,
|
||||
bool allowDelete);
|
||||
|
||||
/// <summary>
|
||||
/// Set a collection by GUID for a specific game object.
|
||||
/// </summary>
|
||||
/// <param name="gameObjectIdx">The index of the desired game object in the object table.</param>
|
||||
/// <param name="collectionId">The GUID of the collection to set it to, null to remove the association if allowed. </param>
|
||||
/// <param name="allowCreateNew">Allow only setting existing individuals or also creating a new individual assignment.</param>
|
||||
/// <param name="allowDelete">Allow deleting existing individual assignments if <paramref name="collectionId"/> is null.</param>
|
||||
/// <returns>InvalidIdentifier if <paramref name="gameObjectIdx"/> does not produce an existing game object or the object is not identifiable,
|
||||
/// NothingChanged if the new collection is the same as the old,<br />
|
||||
/// AssignmentDeletionDisallowed if <paramref name="collectionId"/> is null and <paramref name="allowDelete"/> is false, and the assignment exists,<br />
|
||||
/// CollectionMissing if the new collection can not be found,<br />
|
||||
/// AssignmentCreationDisallowed if <paramref name="allowCreateNew"/> is false and the assignment does not exist,<br />
|
||||
/// or Success, as well as the name of the previous collection (empty if no assignment existed).</returns>
|
||||
public (PenumbraApiEc, (Guid Id, string Name)? OldCollection) SetCollectionForObject(int gameObjectIdx, Guid? collectionId, bool allowCreateNew,
|
||||
bool allowDelete);
|
||||
|
||||
/// <summary> Obtain a function object that can check if the current collection contains a given changed item by listing the mods changing it. </summary>
|
||||
/// <remarks> Throws an <seealso cref="ObjectDisposedException"/> on invocation if the collection storage is not valid anymore, so clear this on <seealso cref="IpcSubscribers.Disposed"/>. </remarks>
|
||||
public Func<string, (string ModDirectory, string ModName)[]> CheckCurrentChangedItemFunc();
|
||||
}
|
||||
@@ -1,28 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the editing of mods or game files. </summary>
|
||||
public interface IPenumbraApiEditing
|
||||
{
|
||||
/// <summary>
|
||||
/// Convert the given texture file into a different type or format and potentially add mip maps.
|
||||
/// </summary>
|
||||
/// <param name="inputFile"> The path to the input file, which may be of .dds, .tex or .png format. </param>
|
||||
/// <param name="outputFile"> The desired output path. Can be the same as input. </param>
|
||||
/// <param name="textureType"> The file type and format to convert the data to. </param>
|
||||
/// <param name="mipMaps"> Whether to add mip maps or not. Ignored for .png. </param>
|
||||
/// <returns> A task for when the conversion is finished or has failed. </returns>
|
||||
public Task ConvertTextureFile(string inputFile, string outputFile, TextureType textureType, bool mipMaps);
|
||||
|
||||
/// <summary>
|
||||
/// Convert the given RGBA32 texture data into a different type or format and potentially add mip maps.
|
||||
/// </summary>
|
||||
/// <param name="rgbaData"> The input byte data for a picture given in RGBA32 format. </param>
|
||||
/// <param name="width"> The width of the input picture. The height is computed from the size of <paramref name="rgbaData"/> and this. </param>
|
||||
/// <param name="outputFile"> The desired output path. Can be the same as input. </param>
|
||||
/// <param name="textureType"> The file type and format to convert the data to. </param>
|
||||
/// <param name="mipMaps"> Whether to add mip maps or not. Ignored for .png. </param>
|
||||
/// <returns> A task for when the conversion is finished or has failed. </returns>
|
||||
public Task ConvertTextureData(byte[] rgbaData, int width, string outputFile, TextureType textureType, bool mipMaps);
|
||||
}
|
||||
@@ -1,52 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the currently tracked game state. </summary>
|
||||
public interface IPenumbraApiGameState
|
||||
{
|
||||
/// <param name="drawObject"></param>
|
||||
/// <returns>The game object associated with the given <paramref name="drawObject">draw object</paramref>
|
||||
/// and the GUID and name of the collection associated with this game object.</returns>
|
||||
public (nint GameObject, (Guid Id, string Name) Collection) GetDrawObjectInfo(nint drawObject);
|
||||
|
||||
/// <summary>
|
||||
/// Obtain the parent game object index for an unnamed cutscene actor by its <paramref name="actorIdx">index</paramref>.
|
||||
/// </summary>
|
||||
/// <param name="actorIdx"></param>
|
||||
/// <returns>The parent game object index.</returns>
|
||||
public int GetCutsceneParentIndex(int actorIdx);
|
||||
|
||||
/// <summary>
|
||||
/// Set the cutscene parent of <paramref name="copyIdx"/> in Penumbras internal state to a new value.
|
||||
/// </summary>
|
||||
/// <param name="copyIdx"> The index of the cutscene actor to be changed. </param>
|
||||
/// <param name="newParentIdx"> The new index of the cutscene actors parent or -1 for no parent. </param>
|
||||
/// <returns> Success when the new parent could be set, or InvalidArgument if either index is out of its respective range. </returns>
|
||||
/// <remarks>
|
||||
/// Checks that the new parent exists as a game object if the value is not -1 before assigning. If it does not, InvalidArgument is given, too.
|
||||
/// Please only use this for good reason and if you know what you are doing, probably only for actor copies you actually create yourself.
|
||||
/// </remarks>
|
||||
public PenumbraApiEc SetCutsceneParentIndex(int copyIdx, int newParentIdx);
|
||||
|
||||
/// <summary>
|
||||
/// Triggered when a character base is created and a corresponding gameObject could be found,
|
||||
/// before the Draw Object is actually created, so customize and equipdata can be manipulated beforehand.
|
||||
/// </summary>
|
||||
/// <returns><inheritdoc cref="CreatingCharacterBaseDelegate"/></returns>
|
||||
public event CreatingCharacterBaseDelegate? CreatingCharacterBase;
|
||||
|
||||
/// <summary>
|
||||
/// Triggered after a character base was created if a corresponding gameObject could be found,
|
||||
/// so you can apply flag changes after finishing.
|
||||
/// </summary>
|
||||
/// <returns><inheritdoc cref="CreatedCharacterBaseDelegate"/></returns>
|
||||
public event CreatedCharacterBaseDelegate? CreatedCharacterBase;
|
||||
|
||||
/// <summary>
|
||||
/// Triggered whenever a resource is redirected by Penumbra for a specific, identified game object.
|
||||
/// Does not trigger if the resource is not requested for a known game object.
|
||||
/// </summary>
|
||||
/// <returns><inheritdoc cref="GameObjectResourceResolvedDelegate"/></returns>
|
||||
public event GameObjectResourceResolvedDelegate? GameObjectResourceResolved;
|
||||
}
|
||||
@@ -1,41 +0,0 @@
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> The entire API. </summary>
|
||||
public interface IPenumbraApi : IPenumbraApiBase
|
||||
{
|
||||
/// <inheritdoc cref="IPenumbraApiCollection"/>
|
||||
public IPenumbraApiCollection Collection { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiEditing"/>
|
||||
public IPenumbraApiEditing Editing { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiGameState"/>
|
||||
public IPenumbraApiGameState GameState { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiMeta"/>
|
||||
public IPenumbraApiMeta Meta { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiMods"/>
|
||||
public IPenumbraApiMods Mods { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiModSettings"/>
|
||||
public IPenumbraApiModSettings ModSettings { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiPluginState"/>
|
||||
public IPenumbraApiPluginState PluginState { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiRedraw"/>
|
||||
public IPenumbraApiRedraw Redraw { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiResolve"/>
|
||||
public IPenumbraApiResolve Resolve { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiResourceTree"/>
|
||||
public IPenumbraApiResourceTree ResourceTree { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiTemporary"/>
|
||||
public IPenumbraApiTemporary Temporary { get; }
|
||||
|
||||
/// <inheritdoc cref="IPenumbraApiUi"/>
|
||||
public IPenumbraApiUi Ui { get; }
|
||||
}
|
||||
@@ -1,16 +0,0 @@
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> Base interface for the API that is always available, regardless of version. </summary>
|
||||
public interface IPenumbraApiBase
|
||||
{
|
||||
/// <summary>
|
||||
/// The API version is staggered in two parts.
|
||||
/// The major/Breaking version only increments if there are changes breaking backwards compatibility.
|
||||
/// The minor/Feature version increments any time there is something added
|
||||
/// and resets when Breaking is incremented.
|
||||
/// </summary>
|
||||
public (int Breaking, int Feature) ApiVersion { get; }
|
||||
|
||||
/// <summary> Whether the API is still usable. </summary>
|
||||
public bool Valid { get; }
|
||||
}
|
||||
@@ -1,13 +0,0 @@
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to current metadata manipulations. </summary>
|
||||
public interface IPenumbraApiMeta
|
||||
{
|
||||
/// <returns>A base64 encoded, zipped json-string with a prepended version-byte of the current manipulations
|
||||
/// in the collection currently associated with the player.</returns>
|
||||
public string GetPlayerMetaManipulations();
|
||||
|
||||
/// <returns>A base64 encoded, zipped json-string with a prepended version-byte of the current manipulations
|
||||
/// in the given collection applying to the given game object or the default collection if it does not exist.</returns>
|
||||
public string GetMetaManipulations(int gameObjectIdx);
|
||||
}
|
||||
@@ -1,77 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
using Penumbra.Api.Helpers;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the management of mod settings. </summary>
|
||||
public interface IPenumbraApiModSettings
|
||||
{
|
||||
/// <summary>
|
||||
/// Obtain the potential settings of a mod given by its <paramref name="modDirectory" /> name or <paramref name="modName" />.
|
||||
/// </summary>
|
||||
/// <returns>A dictionary of group names to lists of option names and the group type. Null if the mod could not be found.</returns>
|
||||
public AvailableModSettings? GetAvailableModSettings(string modDirectory, string modName);
|
||||
|
||||
/// <summary>
|
||||
/// Obtain the enabled state, the priority, the settings of a mod given by its <paramref name="modDirectory" /> name or <paramref name="modName" /> in the specified collection.
|
||||
/// </summary>
|
||||
/// <param name="collectionId">Specify the collection.</param>
|
||||
/// <param name="modDirectory">Specify the mod via its directory name.</param>
|
||||
/// <param name="modName">Specify the mod via its (non-unique) display name.</param>
|
||||
/// <param name="ignoreInheritance">Whether the settings need to be from the given collection or can be inherited from any other by it. (True: given collection only)</param>
|
||||
/// <param name="ignoreTemporary"> Whether the settings need to be actual settings or can be temporary. </param>
|
||||
/// <param name="key"> The key for the settings lock. If <paramref name="ignoreTemporary"/> is false, settings with a key greater than 0 that is different from this will be ignored. </param>
|
||||
/// <returns>ModMissing, CollectionMissing or Success. <para />
|
||||
/// On Success, a tuple of Enabled State, Priority, a dictionary of option group names and lists of enabled option names and a bool whether the settings are inherited (true) or not.</returns>
|
||||
public (PenumbraApiEc, (bool, int, Dictionary<string, List<string>>, bool, bool)?) GetCurrentModSettingsWithTemp(Guid collectionId,
|
||||
string modDirectory, string modName, bool ignoreInheritance, bool ignoreTemporary, int key);
|
||||
|
||||
/// <inheritdoc cref="GetCurrentModSettingsWithTemp"/>
|
||||
public (PenumbraApiEc, (bool, int, Dictionary<string, List<string>>, bool)?) GetCurrentModSettings(Guid collectionId,
|
||||
string modDirectory, string modName, bool ignoreInheritance);
|
||||
|
||||
/// <summary> Obtain the enabled state, the priority, the settings of all mods in the specified collection. </summary>
|
||||
/// <param name="collectionId"> Specify the collection. </param>
|
||||
/// <param name="ignoreInheritance"> Whether the settings need to be from the given collection or can be inherited from any other by it. (True: given collection only) </param>
|
||||
/// <param name="ignoreTemporary"> Whether the settings need to be actual settings or can be temporary. </param>
|
||||
/// <param name="key"> The key for the settings lock. If <paramref name="ignoreTemporary"/> is false, settings with a key greater than 0 that is different from this will be ignored. </param>
|
||||
/// <returns> CollectionMissing or Success, on Success, a dictionary of mod directory names to a tuple of (Enabled, Priority, Settings, Inherited, Temporary). Mods that have no settings at all are left out. </returns>
|
||||
public (PenumbraApiEc, Dictionary<string, (bool, int, Dictionary<string, List<string>>, bool, bool)>?) GetAllModSettings(Guid collectionId,
|
||||
bool ignoreInheritance, bool ignoreTemporary, int key);
|
||||
|
||||
/// <summary> Try to set the inheritance state of a mod in a collection. </summary>
|
||||
/// <returns>ModMissing, CollectionMissing, InvalidArgument (GUID is nil), NothingChanged or Success.</returns>
|
||||
public PenumbraApiEc TryInheritMod(Guid collectionId, string modDirectory, string modName, bool inherit);
|
||||
|
||||
/// <summary> Try to set the enabled state of a mod in a collection. </summary>
|
||||
/// <returns>ModMissing, CollectionMissing, InvalidArgument (GUID is nil), NothingChanged or Success.</returns>
|
||||
public PenumbraApiEc TrySetMod(Guid collectionId, string modDirectory, string modName, bool enabled);
|
||||
|
||||
/// <summary> Try to set the priority of a mod in a collection. </summary>
|
||||
/// <returns>ModMissing, CollectionMissing, InvalidArgument (GUID is nil), NothingChanged or Success.</returns>
|
||||
public PenumbraApiEc TrySetModPriority(Guid collectionId, string modDirectory, string modName, int priority);
|
||||
|
||||
/// <summary> Try to set a specific option group of a mod in the given collection to a specific value. </summary>
|
||||
/// <remarks>Removes inheritance. Single Selection groups should provide a single option, Multi Selection can provide multiple.
|
||||
/// If any setting can not be found, it will not change anything.</remarks>
|
||||
/// <returns>ModMissing, CollectionMissing, OptionGroupMissing, SettingMissing, InvalidArgument (GUID is nil), NothingChanged or Success.</returns>
|
||||
public PenumbraApiEc TrySetModSetting(Guid collectionId, string modDirectory, string modName, string optionGroupName, string optionName);
|
||||
|
||||
/// <inheritdoc cref="TrySetModSetting"/>
|
||||
public PenumbraApiEc TrySetModSettings(Guid collectionId, string modDirectory, string modName, string optionGroupName,
|
||||
IReadOnlyList<string> optionNames);
|
||||
|
||||
/// <summary> This event gets fired when any setting in any collection changes. </summary>
|
||||
/// <returns><inheritdoc cref="ModSettingChangedDelegate" /></returns>
|
||||
public event ModSettingChangedDelegate? ModSettingChanged;
|
||||
|
||||
/// <summary>
|
||||
/// Copy all current settings for a mod to another mod.
|
||||
/// </summary>
|
||||
/// <param name="collectionId">Specify the collection to work in, leave null to do it in all collections.</param>
|
||||
/// <param name="modDirectoryFrom">Specify the mod to take the settings from via its directory name.</param>
|
||||
/// <param name="modDirectoryTo">Specify the mod to put the settings on via its directory name. If the mod does not exist, it will be added as unused settings.</param>
|
||||
/// <returns>CollectionMissing if collectionName is not empty but does not exist or Success.</returns>
|
||||
/// <remarks>If the target mod exists, the settings will be fixed before being applied. If the source mod does not exist, it will use unused settings if available and remove existing settings otherwise.</remarks>
|
||||
public PenumbraApiEc CopyModSettings(Guid? collectionId, string modDirectoryFrom, string modDirectoryTo);
|
||||
}
|
||||
@@ -1,78 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to management of mods. </summary>
|
||||
public interface IPenumbraApiMods
|
||||
{
|
||||
/// <returns>A list of all installed mods. The first string is their directory name, the second string is their mod name.</returns>
|
||||
public Dictionary<string, string> GetModList();
|
||||
|
||||
/// <summary> Try to unpack and install a valid mod file (.pmp, .ttmp, .ttmp2) as if installed manually. </summary>
|
||||
/// <param name="modFilePackagePath">The file that should be unpacked.</param>
|
||||
/// <returns>Success, MissingFile. Success does not indicate successful installing, just successful queueing for install.</returns>
|
||||
public PenumbraApiEc InstallMod(string modFilePackagePath);
|
||||
|
||||
/// <summary> Try to reload an existing mod given by its <paramref name="modDirectory" /> name or <paramref name="modName" />.</summary>
|
||||
/// <remarks>Reload is the same as if triggered by button press and might delete the mod if it is not valid anymore.</remarks>
|
||||
/// <returns>ModMissing if the mod can not be found or Success</returns>
|
||||
public PenumbraApiEc ReloadMod(string modDirectory, string modName);
|
||||
|
||||
/// <summary> Try to add a new mod inside the mod root directory.</summary>
|
||||
/// <remarks>Note that success does only imply a successful call, not a successful mod load.</remarks>
|
||||
/// <param name="modDirectory">The name (not full name) of the mod directory.</param>
|
||||
/// <returns>FileMissing if <paramref name="modDirectory" /> does not exist, InvalidArgument if the path leads outside the root directory, Success otherwise.</returns>
|
||||
public PenumbraApiEc AddMod(string modDirectory);
|
||||
|
||||
/// <summary>Try to delete a mod given by its <paramref name="modDirectory" /> name or <paramref name="modName" />.</summary>
|
||||
/// <remarks>Note that success does only imply a successful call, not successful deletion.</remarks>
|
||||
/// <returns>NothingDone if the mod can not be found, Success otherwise.</returns>
|
||||
public PenumbraApiEc DeleteMod(string modDirectory, string modName);
|
||||
|
||||
/// <summary> Triggers whenever a mod is deleted. </summary>
|
||||
/// <returns>The base directory name of the deleted mod.</returns>
|
||||
public event Action<string>? ModDeleted;
|
||||
|
||||
/// <summary> Triggers whenever a mod is deleted. </summary>
|
||||
/// <returns>The base directory name of the new mod.</returns>
|
||||
public event Action<string>? ModAdded;
|
||||
|
||||
/// <summary> Triggers whenever a mods base name is changed from inside Penumbra. </summary>
|
||||
/// <returns>The previous base directory name of the mod and the new base directory name of the mod.</returns>
|
||||
public event Action<string, string>? ModMoved;
|
||||
|
||||
/// <summary>
|
||||
/// Get the internal full filesystem path including search order for the specified mod
|
||||
/// given by its <paramref name="modDirectory" /> name or <paramref name="modName" />.
|
||||
/// </summary>
|
||||
/// <returns>On Success, the full path, a bool indicating whether the entire path is default (true) or manually set (false),
|
||||
/// and a bool indicating whether the sort order name ignoring the folder path is default (true) or manually set (false).
|
||||
/// Otherwise, returns ModMissing if the mod can not be found.</returns>
|
||||
public (PenumbraApiEc, string, bool, bool) GetModPath(string modDirectory, string modName);
|
||||
|
||||
/// <summary>
|
||||
/// Set the internal search order and filesystem path of the specified mod
|
||||
/// given by its <paramref name="modDirectory" /> name or <paramref name="modName" />
|
||||
/// to the <paramref name="newPath" />.
|
||||
/// </summary>
|
||||
/// <returns>InvalidArgument if <paramref name="newPath" /> is empty, ModMissing if the mod can not be found,
|
||||
/// PathRenameFailed if <paramref name="newPath"/> could not be set or Success.</returns>
|
||||
public PenumbraApiEc SetModPath(string modDirectory, string modName, string newPath);
|
||||
|
||||
/// <summary> Get the overall changed items of a single mod given by its <paramref name="modDirectory"/> name or <paramref name="modName"/>, regardless of settings. </summary>
|
||||
/// <returns> A possibly empty dictionary of affected items and known objects or null. </returns>
|
||||
public Dictionary<string, object?> GetChangedItems(string modDirectory, string modName);
|
||||
|
||||
/// <summary> Get a dictionary of dictionaries to check all mods changed items. </summary>
|
||||
/// <returns> A dictionary of mod identifier to changed item dictionary. </returns>
|
||||
/// <remarks> Throws an <seealso cref="ObjectDisposedException"/> on access if the mod storage is not valid anymore, so clear this on <seealso cref="IpcSubscribers.Disposed"/>. </remarks>
|
||||
public IReadOnlyDictionary<string, IReadOnlyDictionary<string, object?>> GetChangedItemAdapterDictionary();
|
||||
|
||||
/// <summary> Get a list of dictionaries to check all mods changed items. </summary>
|
||||
/// <returns> A list all mods changed item dictionaries. </returns>
|
||||
/// <remarks>
|
||||
/// The order of mods is unspecified, but the same as in GetModList (assuming no changes in mods have taken place between calls). <br/>
|
||||
/// Throws an <seealso cref="ObjectDisposedException"/> on access if the mod storage is not valid anymore, so clear this on <seealso cref="IpcSubscribers.Disposed"/>.
|
||||
/// </remarks>
|
||||
public IReadOnlyList<(string ModDirectory, IReadOnlyDictionary<string, object?> ChangedItems)> GetChangedItemAdapterList();
|
||||
}
|
||||
@@ -1,26 +0,0 @@
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to Penumbras own state. </summary>
|
||||
public interface IPenumbraApiPluginState
|
||||
{
|
||||
/// <returns> The full path of the current penumbra root directory. </returns>
|
||||
public string GetModDirectory();
|
||||
|
||||
/// <returns> The entire current penumbra configuration as a json encoded string. </returns>
|
||||
public string GetConfiguration();
|
||||
|
||||
/// <summary>
|
||||
/// Fired whenever a mod directory change is finished.
|
||||
/// </summary>
|
||||
/// <returns>The full path of the mod directory and whether Penumbra treats it as valid.</returns>
|
||||
public event Action<string, bool>? ModDirectoryChanged;
|
||||
|
||||
/// <returns>True if Penumbra is enabled, false otherwise.</returns>
|
||||
public bool GetEnabledState();
|
||||
|
||||
/// <summary>
|
||||
/// Fired whenever the enabled state of Penumbra changes.
|
||||
/// </summary>
|
||||
/// <returns>True if the new state is enabled, false if the new state is disabled</returns>
|
||||
public event Action<bool>? EnabledChange;
|
||||
}
|
||||
@@ -1,23 +0,0 @@
|
||||
using Penumbra.Api.Enums;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the redrawing of actors. </summary>
|
||||
public interface IPenumbraApiRedraw
|
||||
{
|
||||
/// <summary>
|
||||
/// Queue redrawing of the actor with the given object <paramref name="gameObjectIndex" />, if it exists, with the given RedrawType <paramref name="setting"/>.
|
||||
/// </summary>
|
||||
public void RedrawObject(int gameObjectIndex, RedrawType setting);
|
||||
|
||||
/// <summary>
|
||||
/// Queue redrawing of all currently available actors with the given RedrawType <paramref name="setting"/>.
|
||||
/// </summary>
|
||||
public void RedrawAll(RedrawType setting);
|
||||
|
||||
/// <summary>
|
||||
/// Triggered whenever a game object is redrawn via Penumbra.
|
||||
/// </summary>
|
||||
/// /<returns><inheritdoc cref="GameObjectRedrawnDelegate"/></returns>
|
||||
public event GameObjectRedrawnDelegate? GameObjectRedrawn;
|
||||
}
|
||||
@@ -1,64 +0,0 @@
|
||||
using Lumina.Data;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the resolving of paths. </summary>
|
||||
public interface IPenumbraApiResolve
|
||||
{
|
||||
/// <summary>
|
||||
/// Resolve a given <paramref name="gamePath" /> via Penumbra using the Base collection.
|
||||
/// </summary>
|
||||
/// <returns>The resolved path, or the given path if Penumbra would not manipulate it.</returns>
|
||||
public string ResolveDefaultPath(string gamePath);
|
||||
|
||||
/// <summary>
|
||||
/// Resolve a given <paramref name="gamePath" /> via Penumbra using the Interface collection.
|
||||
/// </summary>
|
||||
/// <returns>The resolved path, or the given path if Penumbra would not manipulate it.</returns>
|
||||
public string ResolveInterfacePath(string gamePath);
|
||||
|
||||
/// <summary>
|
||||
/// Resolve a given <paramref name="gamePath" /> via Penumbra using collection applying to the <paramref name="gameObjectIdx"/>
|
||||
/// given by its index in the game object table.
|
||||
/// </summary>
|
||||
/// <remarks>If the object does not exist in the table, the default collection is used.</remarks>
|
||||
/// <returns>The resolved path, or the given path if Penumbra would not manipulate it.</returns>
|
||||
public string ResolveGameObjectPath(string gamePath, int gameObjectIdx);
|
||||
|
||||
/// <summary>
|
||||
/// Resolve a given <paramref name="gamePath" /> via Penumbra using the collection currently applying to the player character.
|
||||
/// </summary>
|
||||
/// <returns>The resolved path, or the given path if Penumbra would not manipulate it.</returns>
|
||||
public string ResolvePlayerPath(string gamePath);
|
||||
|
||||
/// <summary>
|
||||
/// Reverse resolves a given local <paramref name="moddedPath" /> into its replacement in form of all applicable game paths
|
||||
/// for the collection applying to the <paramref name="gameObjectIdx"/>th game object in the game object table.
|
||||
/// </summary>
|
||||
/// <remarks>If the object does not exist in the table, the default collection is used.</remarks>
|
||||
/// <returns>A list of game paths resolving to the modded path.</returns>
|
||||
public string[] ReverseResolveGameObjectPath(string moddedPath, int gameObjectIdx);
|
||||
|
||||
/// <summary>
|
||||
/// Reverse resolves a given local <paramref name="moddedPath" /> into its replacement in form of all applicable game paths
|
||||
/// for the collection currently applying to the player character.
|
||||
/// </summary>
|
||||
/// <returns>A list of game paths resolving to the modded path.</returns>
|
||||
public string[] ReverseResolvePlayerPath(string moddedPath);
|
||||
|
||||
/// <summary>
|
||||
/// Resolve all game paths in <paramref name="forward"/> and reserve all paths in <paramref name="reverse"/> at once.
|
||||
/// </summary>
|
||||
/// <param name="forward">Paths to forward-resolve.</param>
|
||||
/// <param name="reverse">Paths to reverse-resolve.</param>
|
||||
/// <returns>A pair of an array of forward-resolved single paths of the same length as <paramref name="forward"/> and an array of arrays of reverse-resolved paths.
|
||||
/// The outer array has the same length as <paramref name="reverse"/> while each inner array can have arbitrary length.</returns>
|
||||
public (string[], string[][]) ResolvePlayerPaths(string[] forward, string[] reverse);
|
||||
|
||||
/// <summary>
|
||||
/// Resolve all game paths in <paramref name="forward"/> and reserve all paths in <paramref name="reverse"/> at once asynchronously.
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="ResolvePlayerPaths"/>
|
||||
/// <remarks> Can be called from outside of framework. Can theoretically produce incoherent state when collections change during evaluation. </remarks>
|
||||
public Task<(string[], string[][])> ResolvePlayerPathsAsync(string[] forward, string[] reverse);
|
||||
}
|
||||
@@ -1,73 +0,0 @@
|
||||
using Newtonsoft.Json.Linq;
|
||||
using Penumbra.Api.Enums;
|
||||
using Penumbra.Api.Helpers;
|
||||
|
||||
namespace Penumbra.Api.Api;
|
||||
|
||||
/// <summary> API methods pertaining to the tracking of resources in use by actors. </summary>
|
||||
public interface IPenumbraApiResourceTree
|
||||
{
|
||||
/// <summary>
|
||||
/// Get the given game objects' resources, as dictionaries of actual paths (that may be FS paths for redirected resources, or game paths for swapped or vanilla resources) to game paths.
|
||||
/// </summary>
|
||||
/// <param name="gameObjects"> The game object indices for which to get the resources. </param>
|
||||
/// <returns> An array of resource path dictionaries, of the same length and in the same order as the given game object index array. </returns>
|
||||
/// <remarks> This function is best called right after the game objects are redrawn, as it may fail to resolve paths if relevant mod settings have changed since then. </remarks>
|
||||
public Dictionary<string, HashSet<string>>?[] GetGameObjectResourcePaths(params ushort[] gameObjects);
|
||||
|
||||
/// <summary>
|
||||
/// Get the player and player-owned game objects' resources, as dictionaries of actual paths (that may be FS paths for redirected resources, or game paths for swapped or vanilla resources) to game paths.
|
||||
/// </summary>
|
||||
/// <returns> A dictionary of game object indices to resource path dictionaries. </returns>
|
||||
/// <remarks> This function is best called right after the game objects are redrawn, as it may fail to resolve paths if relevant mod settings have changed since then. </remarks>
|
||||
public Dictionary<ushort, Dictionary<string, HashSet<string>>> GetPlayerResourcePaths();
|
||||
|
||||
/// <summary>
|
||||
/// Get the given game objects' resources of a given type, as dictionaries of resource handles to actual paths and, optionally, names and icons.
|
||||
/// </summary>
|
||||
/// <param name="type"> Type of the resources to get, for example <see cref="ResourceType.Mtrl"/> for materials. </param>
|
||||
/// <param name="withUiData"> Whether to get names and icons along with the paths. </param>
|
||||
/// <param name="gameObjects"> The game object indices for which to get the resources. </param>
|
||||
/// <returns> An array of resource information dictionaries, of the same length and in the same order as the given game object index array. </returns>
|
||||
/// <remarks>
|
||||
/// It is the caller's responsibility to make sure the returned resource handles are still in use on the game object's draw object before using them. <para />
|
||||
/// Also, callers should not use UI data for non-UI purposes.
|
||||
/// </remarks>
|
||||
public GameResourceDict?[] GetGameObjectResourcesOfType(ResourceType type, bool withUiData,
|
||||
params ushort[] gameObjects);
|
||||
|
||||
/// <summary>
|
||||
/// Get the player and player-owned game objects' resources of a given type, as dictionaries of resource handles to actual paths and, optionally, names and icons.
|
||||
/// </summary>
|
||||
/// <param name="type"> Type of the resources to get, for example <see cref="ResourceType.Mtrl"/> for materials. </param>
|
||||
/// <param name="withUiData"> Whether to get names and icons along with the paths. </param>
|
||||
/// <returns> A dictionary of game object indices to resource information dictionaries. </returns>
|
||||
/// <remarks>
|
||||
/// It is the caller's responsibility to make sure the returned resource handles are still in use on the game object's draw object before using them. <para />
|
||||
/// Also, callers should not use UI data for non-UI purposes.
|
||||
/// </remarks>
|
||||
public Dictionary<ushort, GameResourceDict> GetPlayerResourcesOfType(ResourceType type, bool withUiData);
|
||||
|
||||
/// <summary>
|
||||
/// Get the given game objects' resource tree.
|
||||
/// </summary>
|
||||
/// <param name="withUiData"> Whether to get names and icons along with the paths. </param>
|
||||
/// <param name="gameObjects"> The game object indices for which to get the resources. </param>
|
||||
/// <returns> An array of resource tree JObjects, of the same length and in the same order as the given game object index array. </returns>
|
||||
/// <remarks>
|
||||
/// It is the caller's responsibility to make sure the returned resource handles are still in use on the game object's draw object before using them. <para />
|
||||
/// Also, callers should not use UI data for non-UI purposes.
|
||||
/// </remarks>
|
||||
public JObject?[] GetGameObjectResourceTrees(bool withUiData, params ushort[] gameObjects);
|
||||
|
||||
/// <summary>
|
||||
/// Get the player and player-owned game objects' resource trees.
|
||||
/// </summary>
|
||||
/// <param name="withUiData"> Whether to get names and icons along with the paths. </param>
|
||||
/// <returns> A dictionary of game object indices to resource trees. </returns>
|
||||
/// <remarks>
|
||||
/// It is the caller's responsibility to make sure the returned resource handles are still in use on the game object's draw object before using them. <para />
|
||||
/// Also, callers should not use UI data for non-UI purposes.
|
||||
/// </remarks>
|
||||
public Dictionary<ushort, JObject> GetPlayerResourceTrees(bool withUiData);
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user