2020-01-25 01:57:47 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.ComponentModel;
|
2020-05-26 22:58:48 +00:00
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.IO;
|
2020-01-25 01:57:47 +00:00
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
namespace PKHeX.Core
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Provides reflection utility for manipulating blocks, providing block names and value wrapping.
|
|
|
|
|
/// </summary>
|
2020-09-07 20:51:13 +00:00
|
|
|
|
public sealed class SCBlockMetadata
|
2020-01-25 01:57:47 +00:00
|
|
|
|
{
|
|
|
|
|
private readonly Dictionary<SaveBlock, string> BlockList;
|
|
|
|
|
private readonly Dictionary<uint, string> ValueList;
|
|
|
|
|
private readonly SCBlockAccessor Accessor;
|
|
|
|
|
|
|
|
|
|
public SCBlockMetadata(SCBlockAccessor accessor)
|
|
|
|
|
{
|
|
|
|
|
var aType = accessor.GetType();
|
|
|
|
|
|
|
|
|
|
BlockList = aType.GetAllPropertiesOfType<SaveBlock>(accessor);
|
|
|
|
|
ValueList = aType.GetAllConstantsOfType<uint>();
|
2020-05-26 22:58:48 +00:00
|
|
|
|
AddExtraKeyNames(ValueList);
|
2020-01-25 01:57:47 +00:00
|
|
|
|
Accessor = accessor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IEnumerable<ComboItem> GetSortedBlockKeyList()
|
|
|
|
|
{
|
|
|
|
|
var list = Accessor.BlockInfo
|
|
|
|
|
.Select((z, i) => new ComboItem(GetBlockHint(z, i), (int)z.Key))
|
|
|
|
|
.OrderBy(z => !z.Text.StartsWith("*"))
|
|
|
|
|
.ThenBy(z => GetSortKey(z));
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-26 22:58:48 +00:00
|
|
|
|
public static void AddExtraKeyNames(IDictionary<uint, string> names, string extra = "SCBlocks.txt")
|
|
|
|
|
{
|
|
|
|
|
if (!File.Exists(extra))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
var lines = File.ReadLines(extra);
|
|
|
|
|
foreach (var line in lines)
|
|
|
|
|
{
|
|
|
|
|
var split = line.IndexOf('\t');
|
|
|
|
|
if (split < 0)
|
|
|
|
|
continue;
|
|
|
|
|
var hex = line.Substring(0, split);
|
|
|
|
|
if (!ulong.TryParse(hex, NumberStyles.HexNumber, CultureInfo.CurrentCulture, out var value))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
var name = line.Substring(split + 1);
|
|
|
|
|
if (!names.ContainsKey((uint)value))
|
|
|
|
|
names[(uint)value] = name;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-25 01:57:47 +00:00
|
|
|
|
private static string GetSortKey(in ComboItem item)
|
|
|
|
|
{
|
|
|
|
|
var text = item.Text;
|
|
|
|
|
if (text.StartsWith("*"))
|
|
|
|
|
return text;
|
|
|
|
|
// key:X8, " - ", "####", " ", type
|
|
|
|
|
return text.Substring(8 + 3 + 4 + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetBlockHint(SCBlock z, int i)
|
|
|
|
|
{
|
|
|
|
|
var blockName = GetBlockName(z, out _);
|
2020-01-29 03:08:26 +00:00
|
|
|
|
var isBool = z.Type.IsBoolean();
|
|
|
|
|
var type = (isBool ? "Bool" : z.Type.ToString());
|
2020-01-25 01:57:47 +00:00
|
|
|
|
if (blockName != null)
|
|
|
|
|
return $"*{type} {blockName}";
|
2020-01-29 03:08:26 +00:00
|
|
|
|
var result = $"{z.Key:X8} - {i:0000} {type}";
|
|
|
|
|
if (z.Type == SCTypeCode.Object || z.Type == SCTypeCode.Array)
|
|
|
|
|
result += $" 0x{z.Data.Length:X3}";
|
|
|
|
|
else if (!isBool)
|
|
|
|
|
result += $" {z.GetValue()}";
|
|
|
|
|
return result;
|
2020-01-25 01:57:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string? GetBlockName(SCBlock block, out SaveBlock? saveBlock)
|
|
|
|
|
{
|
|
|
|
|
// See if we have a Block object for this block
|
2020-09-06 18:21:23 +00:00
|
|
|
|
if (block.Data.Length != 0)
|
2020-01-25 01:57:47 +00:00
|
|
|
|
{
|
2020-09-06 18:21:23 +00:00
|
|
|
|
var obj = BlockList.FirstOrDefault(z => ReferenceEquals(z.Key.Data, block.Data));
|
|
|
|
|
if (obj.Key != null)
|
|
|
|
|
{
|
|
|
|
|
saveBlock = obj.Key;
|
|
|
|
|
return obj.Value;
|
|
|
|
|
}
|
2020-01-25 01:57:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// See if it's a single-value declaration
|
|
|
|
|
if (ValueList.TryGetValue(block.Key, out var blockName))
|
|
|
|
|
{
|
|
|
|
|
saveBlock = null;
|
|
|
|
|
return blockName;
|
|
|
|
|
}
|
|
|
|
|
saveBlock = null;
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns an object that wraps the block with a Value property to get/set via a PropertyGrid/etc control.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>Returns null if no wrapping is supported.</returns>
|
|
|
|
|
public static object? GetEditableBlockObject(SCBlock block)
|
|
|
|
|
{
|
|
|
|
|
return block.Type switch
|
|
|
|
|
{
|
|
|
|
|
SCTypeCode.Byte => new WrappedValueView<byte>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.UInt16 => new WrappedValueView<ushort>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.UInt32 => new WrappedValueView<uint>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.UInt64 => new WrappedValueView<ulong>(block, block.GetValue()),
|
|
|
|
|
|
|
|
|
|
SCTypeCode.SByte => new WrappedValueView<sbyte>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.Int16 => new WrappedValueView<short>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.Int32 => new WrappedValueView<int>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.Int64 => new WrappedValueView<long>(block, block.GetValue()),
|
|
|
|
|
|
|
|
|
|
SCTypeCode.Single => new WrappedValueView<float>(block, block.GetValue()),
|
|
|
|
|
SCTypeCode.Double => new WrappedValueView<double>(block, block.GetValue()),
|
|
|
|
|
|
|
|
|
|
_ => null,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-07 20:51:13 +00:00
|
|
|
|
private sealed class WrappedValueView<T> where T : struct
|
2020-01-25 01:57:47 +00:00
|
|
|
|
{
|
|
|
|
|
private readonly SCBlock Parent;
|
|
|
|
|
private T _value;
|
|
|
|
|
|
|
|
|
|
[Description("Stored Value for this Block")]
|
|
|
|
|
public T Value
|
|
|
|
|
{
|
|
|
|
|
get => _value;
|
|
|
|
|
set => Parent.SetValue(_value = value);
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-09 19:47:24 +00:00
|
|
|
|
// ReSharper disable once UnusedMember.Local
|
2020-01-25 01:57:47 +00:00
|
|
|
|
[Description("Type of Value this Block stores")]
|
|
|
|
|
public string ValueType => typeof(T).Name;
|
|
|
|
|
|
|
|
|
|
public WrappedValueView(SCBlock block, object currentValue)
|
|
|
|
|
{
|
|
|
|
|
Parent = block;
|
|
|
|
|
_value = (T)Convert.ChangeType(currentValue, typeof(T));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|