2017-10-06 13:29:58 -07:00
|
|
|
|
using System;
|
2020-09-07 14:47:27 -07:00
|
|
|
|
using System.Xml.Serialization;
|
2024-02-28 19:19:50 -05:00
|
|
|
|
using Newtonsoft.Json;
|
2020-12-08 13:23:59 -08:00
|
|
|
|
using SabreTools.Core;
|
2024-03-12 22:52:36 -04:00
|
|
|
|
using SabreTools.Core.Filter;
|
2020-12-08 16:37:08 -08:00
|
|
|
|
using SabreTools.Core.Tools;
|
2017-06-16 16:24:26 -07:00
|
|
|
|
|
2020-12-08 16:37:08 -08:00
|
|
|
|
namespace SabreTools.DatFiles
|
2017-06-16 16:24:26 -07:00
|
|
|
|
{
|
2019-01-08 11:06:26 -08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Represents all possible DAT header information
|
|
|
|
|
|
/// </summary>
|
2020-09-08 10:12:41 -07:00
|
|
|
|
[JsonObject("header"), XmlRoot("header")]
|
2024-03-12 22:26:54 -04:00
|
|
|
|
public sealed class DatHeader : ModelBackedItem<Models.Metadata.Header>, ICloneable
|
2019-01-08 11:06:26 -08:00
|
|
|
|
{
|
2024-03-10 16:49:07 -04:00
|
|
|
|
#region Constants
|
2017-06-16 16:24:26 -07:00
|
|
|
|
|
2024-03-10 21:54:07 -04:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Read or write format
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public const string DatFormatKey = "DATFORMAT";
|
|
|
|
|
|
|
2024-03-10 21:41:49 -04:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// External name of the DAT
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public const string FileNameKey = "FILENAME";
|
|
|
|
|
|
|
2024-03-10 16:49:07 -04:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
#region Fields
|
2019-01-08 11:49:31 -08:00
|
|
|
|
|
2024-03-10 16:49:07 -04:00
|
|
|
|
[JsonIgnore]
|
2024-03-12 11:53:58 -04:00
|
|
|
|
public bool CanOpenSpecified
|
2024-03-10 16:49:07 -04:00
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-03-12 11:53:58 -04:00
|
|
|
|
var canOpen = GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey);
|
|
|
|
|
|
return canOpen != null && canOpen.Length > 0;
|
2024-03-10 16:49:07 -04:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[JsonIgnore]
|
2024-03-12 11:53:58 -04:00
|
|
|
|
public bool ImagesSpecified
|
2024-03-10 16:49:07 -04:00
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-03-12 11:53:58 -04:00
|
|
|
|
return GetFieldValue<Models.OfflineList.Images?>(Models.Metadata.Header.ImagesKey) != null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[JsonIgnore]
|
|
|
|
|
|
public bool InfosSpecified
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-10-30 14:11:35 -04:00
|
|
|
|
return GetFieldValue<Models.OfflineList.Infos?>(Models.Metadata.Header.InfosKey) != null;
|
2024-03-10 16:49:07 -04:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2019-01-08 11:49:31 -08:00
|
|
|
|
|
2024-03-11 01:37:47 -04:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
|
public bool NewDatSpecified
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-10-30 14:11:35 -04:00
|
|
|
|
return GetFieldValue<Models.OfflineList.NewDat?>(Models.Metadata.Header.NewDatKey) != null;
|
2024-03-11 01:37:47 -04:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-12 00:09:48 -04:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
|
public bool SearchSpecified
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
|
|
|
|
|
return GetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey) != null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2019-01-08 11:06:26 -08:00
|
|
|
|
#endregion
|
2017-10-06 13:29:58 -07:00
|
|
|
|
|
2024-03-10 22:49:15 -04:00
|
|
|
|
#region Constructors
|
|
|
|
|
|
|
|
|
|
|
|
public DatHeader() { }
|
|
|
|
|
|
|
|
|
|
|
|
public DatHeader(Models.Metadata.Header header)
|
|
|
|
|
|
{
|
2025-01-30 13:01:07 -05:00
|
|
|
|
// Create a new internal model
|
|
|
|
|
|
_internal = new Models.Metadata.Header();
|
|
|
|
|
|
|
2024-03-10 22:49:15 -04:00
|
|
|
|
// Get all fields to automatically copy without processing
|
|
|
|
|
|
var nonItemFields = TypeHelper.GetConstants(typeof(Models.Metadata.Header));
|
2025-01-30 13:01:07 -05:00
|
|
|
|
if (nonItemFields != null)
|
2024-03-10 22:49:15 -04:00
|
|
|
|
{
|
2025-01-30 13:01:07 -05:00
|
|
|
|
// Populate the internal machine from non-filter fields
|
|
|
|
|
|
foreach (string fieldName in nonItemFields)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (header.ContainsKey(fieldName))
|
|
|
|
|
|
_internal[fieldName] = header[fieldName];
|
|
|
|
|
|
}
|
2024-03-10 22:49:15 -04:00
|
|
|
|
}
|
2024-09-30 19:47:56 -04:00
|
|
|
|
|
|
|
|
|
|
// Get all fields specific to the DatFiles implementation
|
|
|
|
|
|
var nonStandardFields = TypeHelper.GetConstants(typeof(DatHeader));
|
2025-01-30 13:01:07 -05:00
|
|
|
|
if (nonStandardFields != null)
|
2024-09-30 19:47:56 -04:00
|
|
|
|
{
|
2025-01-30 13:01:07 -05:00
|
|
|
|
// Populate the internal machine from filter fields
|
|
|
|
|
|
foreach (string fieldName in nonStandardFields)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (header.ContainsKey(fieldName))
|
|
|
|
|
|
_internal[fieldName] = header[fieldName];
|
|
|
|
|
|
}
|
2024-09-30 19:47:56 -04:00
|
|
|
|
}
|
2025-01-30 14:05:12 -05:00
|
|
|
|
|
|
|
|
|
|
// Get all no-filter fields
|
|
|
|
|
|
if (header.ContainsKey(Models.Metadata.Header.CanOpenKey))
|
|
|
|
|
|
_internal[Models.Metadata.Header.CanOpenKey] = header[Models.Metadata.Header.CanOpenKey];
|
|
|
|
|
|
if (header.ContainsKey(Models.Metadata.Header.ImagesKey))
|
|
|
|
|
|
_internal[Models.Metadata.Header.ImagesKey] = header[Models.Metadata.Header.ImagesKey];
|
|
|
|
|
|
if (header.ContainsKey(Models.Metadata.Header.InfosKey))
|
|
|
|
|
|
_internal[Models.Metadata.Header.InfosKey] = header[Models.Metadata.Header.InfosKey];
|
|
|
|
|
|
if (header.ContainsKey(Models.Metadata.Header.NewDatKey))
|
|
|
|
|
|
_internal[Models.Metadata.Header.NewDatKey] = header[Models.Metadata.Header.NewDatKey];
|
|
|
|
|
|
if (header.ContainsKey(Models.Metadata.Header.SearchKey))
|
|
|
|
|
|
_internal[Models.Metadata.Header.SearchKey] = header[Models.Metadata.Header.SearchKey];
|
2024-03-10 22:49:15 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
2019-01-08 11:06:26 -08:00
|
|
|
|
#region Cloning Methods
|
2017-10-06 13:29:58 -07:00
|
|
|
|
|
2019-01-08 11:06:26 -08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Clone the current header
|
|
|
|
|
|
/// </summary>
|
2024-03-12 22:09:16 -04:00
|
|
|
|
public object Clone() => new DatHeader(GetInternalClone());
|
2017-10-06 13:29:58 -07:00
|
|
|
|
|
2020-07-15 09:41:59 -07:00
|
|
|
|
/// <summary>
|
2025-01-29 22:52:22 -05:00
|
|
|
|
/// Clone just the format from the current header
|
2020-07-15 09:41:59 -07:00
|
|
|
|
/// </summary>
|
2025-01-29 22:52:22 -05:00
|
|
|
|
public DatHeader CloneFormat()
|
2020-07-15 09:41:59 -07:00
|
|
|
|
{
|
2024-03-10 22:08:08 -04:00
|
|
|
|
var header = new DatHeader();
|
2025-01-29 22:52:22 -05:00
|
|
|
|
|
2024-07-15 12:48:26 -04:00
|
|
|
|
header.SetFieldValue<DatFormat>(DatHeader.DatFormatKey,
|
|
|
|
|
|
GetFieldValue<DatFormat>(DatHeader.DatFormatKey));
|
2024-03-10 16:49:07 -04:00
|
|
|
|
|
|
|
|
|
|
return header;
|
2020-07-15 09:41:59 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-11 23:34:26 -05:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Get a clone of the current internal model
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public Models.Metadata.Header GetInternalClone()
|
|
|
|
|
|
{
|
|
|
|
|
|
var header = (_internal.Clone() as Models.Metadata.Header)!;
|
|
|
|
|
|
|
|
|
|
|
|
// Remove fields with default values
|
2025-05-11 22:55:38 -04:00
|
|
|
|
if (header.ReadString(Models.Metadata.Header.ForceMergingKey).AsMergingFlag() == MergingFlag.None)
|
2025-01-11 23:34:26 -05:00
|
|
|
|
header.Remove(Models.Metadata.Header.ForceMergingKey);
|
2025-05-11 22:55:38 -04:00
|
|
|
|
if (header.ReadString(Models.Metadata.Header.ForceNodumpKey).AsNodumpFlag() == NodumpFlag.None)
|
2025-01-11 23:34:26 -05:00
|
|
|
|
header.Remove(Models.Metadata.Header.ForceNodumpKey);
|
2025-05-11 22:55:38 -04:00
|
|
|
|
if (header.ReadString(Models.Metadata.Header.ForcePackingKey).AsPackingFlag() == PackingFlag.None)
|
2025-01-11 23:34:26 -05:00
|
|
|
|
header.Remove(Models.Metadata.Header.ForcePackingKey);
|
2025-05-12 22:40:55 -04:00
|
|
|
|
if (header.ReadString(Models.Metadata.Header.BiosModeKey).AsMergingFlag() == MergingFlag.None)
|
|
|
|
|
|
header.Remove(Models.Metadata.Header.BiosModeKey);
|
|
|
|
|
|
if (header.ReadString(Models.Metadata.Header.RomModeKey).AsMergingFlag() == MergingFlag.None)
|
|
|
|
|
|
header.Remove(Models.Metadata.Header.RomModeKey);
|
|
|
|
|
|
if (header.ReadString(Models.Metadata.Header.SampleModeKey).AsMergingFlag() == MergingFlag.None)
|
|
|
|
|
|
header.Remove(Models.Metadata.Header.SampleModeKey);
|
2025-01-11 23:34:26 -05:00
|
|
|
|
|
|
|
|
|
|
// Convert subheader values
|
|
|
|
|
|
if (CanOpenSpecified)
|
|
|
|
|
|
header[Models.Metadata.Header.CanOpenKey] = new Models.OfflineList.CanOpen { Extension = GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey) };
|
|
|
|
|
|
if (ImagesSpecified)
|
|
|
|
|
|
header[Models.Metadata.Header.ImagesKey] = GetFieldValue<Models.OfflineList.Images>(Models.Metadata.Header.ImagesKey);
|
|
|
|
|
|
if (InfosSpecified)
|
|
|
|
|
|
header[Models.Metadata.Header.InfosKey] = GetFieldValue<Models.OfflineList.Infos>(Models.Metadata.Header.InfosKey);
|
|
|
|
|
|
if (NewDatSpecified)
|
|
|
|
|
|
header[Models.Metadata.Header.NewDatKey] = GetFieldValue<Models.OfflineList.NewDat>(Models.Metadata.Header.NewDatKey);
|
|
|
|
|
|
if (SearchSpecified)
|
|
|
|
|
|
header[Models.Metadata.Header.SearchKey] = GetFieldValue<Models.OfflineList.Search>(Models.Metadata.Header.SearchKey);
|
|
|
|
|
|
|
|
|
|
|
|
return header;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-15 09:41:59 -07:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2025-01-08 11:36:43 -05:00
|
|
|
|
#region Comparision Methods
|
|
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc/>
|
|
|
|
|
|
public override bool Equals(ModelBackedItem? other)
|
|
|
|
|
|
{
|
|
|
|
|
|
// If other is null
|
|
|
|
|
|
if (other == null)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
// If the type is mismatched
|
|
|
|
|
|
if (other is not DatHeader otherItem)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
// Compare internal models
|
|
|
|
|
|
return _internal.EqualTo(otherItem._internal);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc/>
|
|
|
|
|
|
public override bool Equals(ModelBackedItem<Models.Metadata.Header>? other)
|
|
|
|
|
|
{
|
|
|
|
|
|
// If other is null
|
|
|
|
|
|
if (other == null)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
// If the type is mismatched
|
|
|
|
|
|
if (other is not DatHeader otherItem)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
// Compare internal models
|
|
|
|
|
|
return _internal.EqualTo(otherItem._internal);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
2024-03-06 00:33:45 -05:00
|
|
|
|
#region Manipulation
|
|
|
|
|
|
|
2024-03-10 21:03:53 -04:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Runs a filter and determines if it passes or not
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="filterRunner">Filter runner to use for checking</param>
|
|
|
|
|
|
/// <returns>True if the Machine passes the filter, false otherwise</returns>
|
2024-03-12 22:22:12 -04:00
|
|
|
|
public bool PassesFilter(FilterRunner filterRunner) => filterRunner.Run(_internal);
|
2024-03-10 21:03:53 -04:00
|
|
|
|
|
2024-03-06 00:33:45 -05:00
|
|
|
|
#endregion
|
2019-01-08 11:06:26 -08:00
|
|
|
|
}
|
2017-06-16 16:24:26 -07:00
|
|
|
|
}
|