mirror of
https://github.com/goatcorp/Dalamud.git
synced 2026-02-13 11:27:42 +01:00
* Use new Lock objects * Fix CA1513: Use ObjectDisposedException.ThrowIf * Fix CA1860: Avoid using 'Enumerable.Any()' extension method * Fix IDE0028: Use collection initializers or expressions * Fix CA2263: Prefer generic overload when type is known * Fix CA1862: Use the 'StringComparison' method overloads to perform case-insensitive string comparisons * Fix IDE0270: Null check can be simplified * Fix IDE0280: Use 'nameof' * Fix IDE0009: Add '.this' * Fix IDE0007: Use 'var' instead of explicit type * Fix IDE0062: Make local function static * Fix CA1859: Use concrete types when possible for improved performance * Fix IDE0066: Use switch expression Only applied to where it doesn't look horrendous. * Use is over switch * Fix CA1847: Use String.Contains(char) instead of String.Contains(string) with single characters * Fix SYSLIB1045: Use 'GeneratedRegexAttribute' to generate the regular expression implementation at compile-time. * Fix CA1866: Use 'string.EndsWith(char)' instead of 'string.EndsWith(string)' when you have a string with a single char * Fix IDE0057: Substring can be simplified * Fix IDE0059: Remove unnecessary value assignment * Fix CA1510: Use ArgumentNullException throw helper * Fix IDE0300: Use collection expression for array * Fix IDE0250: Struct can be made 'readonly' * Fix IDE0018: Inline variable declaration * Fix CA1850: Prefer static HashData method over ComputeHash * Fi CA1872: Prefer 'Convert.ToHexString' and 'Convert.ToHexStringLower' over call chains based on 'BitConverter.ToString' * Update ModuleLog instantiations * Organize usings
113 lines
4.6 KiB
C#
113 lines
4.6 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Runtime.InteropServices;
|
|
|
|
using Dalamud.Utility;
|
|
|
|
using FFXIVClientStructs.FFXIV.Component.GUI;
|
|
|
|
namespace Dalamud.Game.NativeWrapper;
|
|
|
|
/// <summary>
|
|
/// A readonly wrapper for AtkValue.
|
|
/// </summary>
|
|
/// <param name="address">The address to the AtkValue.</param>
|
|
[StructLayout(LayoutKind.Explicit, Size = 0x08)]
|
|
public readonly unsafe struct AtkValuePtr(nint address) : IEquatable<AtkValuePtr>
|
|
{
|
|
/// <summary>
|
|
/// The address to the AtkValue.
|
|
/// </summary>
|
|
[FieldOffset(0x00)]
|
|
public readonly nint Address = address;
|
|
|
|
/// <summary>
|
|
/// Gets a value indicating whether the underlying pointer is a nullptr.
|
|
/// </summary>
|
|
public readonly bool IsNull => this.Address == 0;
|
|
|
|
/// <summary>
|
|
/// Gets the value type.
|
|
/// </summary>
|
|
public readonly AtkValueType ValueType => (AtkValueType)this.Struct->Type;
|
|
|
|
/// <summary>
|
|
/// Gets the AtkValue*.
|
|
/// </summary>
|
|
/// <remarks> Internal use only. </remarks>
|
|
internal readonly AtkValue* Struct => (AtkValue*)this.Address;
|
|
|
|
public static implicit operator nint(AtkValuePtr wrapper) => wrapper.Address;
|
|
|
|
public static implicit operator AtkValuePtr(nint address) => new(address);
|
|
|
|
public static implicit operator AtkValuePtr(void* ptr) => new((nint)ptr);
|
|
|
|
public static bool operator ==(AtkValuePtr left, AtkValuePtr right) => left.Address == right.Address;
|
|
|
|
public static bool operator !=(AtkValuePtr left, AtkValuePtr right) => left.Address != right.Address;
|
|
|
|
/// <summary>
|
|
/// Gets the value of the underlying <see cref="AtkValue"/> as a boxed object, based on its <see cref="AtkValueType"/>.
|
|
/// </summary>
|
|
/// <returns>
|
|
/// The boxed value represented by this <see cref="AtkValuePtr"/>, or <c>null</c> if the value is null or undefined.
|
|
/// </returns>
|
|
/// <exception cref="NotImplementedException">
|
|
/// Thrown if the value type is not currently handled by this implementation.
|
|
/// </exception>
|
|
public unsafe object? GetValue()
|
|
{
|
|
if (this.Struct == null)
|
|
return null;
|
|
|
|
return this.ValueType switch
|
|
{
|
|
AtkValueType.Undefined or AtkValueType.Null => null,
|
|
AtkValueType.Bool => this.Struct->Bool,
|
|
AtkValueType.Int => this.Struct->Int,
|
|
AtkValueType.Int64 => this.Struct->Int64,
|
|
AtkValueType.UInt => this.Struct->UInt,
|
|
AtkValueType.UInt64 => this.Struct->UInt64,
|
|
AtkValueType.Float => this.Struct->Float,
|
|
AtkValueType.String or AtkValueType.String8 or AtkValueType.ManagedString => this.Struct->String.HasValue ? this.Struct->String.AsReadOnlySeString() : default,
|
|
AtkValueType.WideString => this.Struct->WideString == null ? string.Empty : new string(this.Struct->WideString),
|
|
AtkValueType.Pointer => (nint)this.Struct->Pointer,
|
|
_ => throw new NotImplementedException($"AtkValueType {this.ValueType} is currently not supported"),
|
|
};
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to retrieve the value as a strongly-typed object if the underlying type matches.
|
|
/// </summary>
|
|
/// <typeparam name="T">The expected value type to extract.</typeparam>
|
|
/// <param name="result">
|
|
/// When this method returns <c>true</c>, contains the extracted value of type <typeparamref name="T"/>.
|
|
/// Otherwise, contains the default value of type <typeparamref name="T"/>.
|
|
/// </param>
|
|
/// <returns>
|
|
/// <c>true</c> if the value was successfully extracted and matched <typeparamref name="T"/>; otherwise, <c>false</c>.
|
|
/// </returns>
|
|
public unsafe bool TryGet<T>([NotNullWhen(true)] out T? result) where T : struct
|
|
{
|
|
var value = this.GetValue();
|
|
if (value is T typed)
|
|
{
|
|
result = typed;
|
|
return true;
|
|
}
|
|
|
|
result = default;
|
|
return false;
|
|
}
|
|
|
|
/// <summary>Determines whether the specified AtkValuePtr is equal to the current AtkValuePtr.</summary>
|
|
/// <param name="other">The AtkValuePtr to compare with the current AtkValuePtr.</param>
|
|
/// <returns><c>true</c> if the specified AtkValuePtr is equal to the current AtkValuePtr; otherwise, <c>false</c>.</returns>
|
|
public readonly bool Equals(AtkValuePtr other) => this.Address == other.Address || this.Struct->EqualTo(other.Struct);
|
|
|
|
/// <inheritdoc cref="object.Equals(object?)"/>
|
|
public override readonly bool Equals(object obj) => obj is AtkValuePtr wrapper && this.Equals(wrapper);
|
|
|
|
/// <inheritdoc cref="object.GetHashCode()"/>
|
|
public override readonly int GetHashCode() => ((nuint)this.Address).GetHashCode();
|
|
}
|