Files
SabreTools/SabreTools.DatFiles/DatHeader.cs

232 lines
8.2 KiB
C#
Raw Normal View History

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;
using SabreTools.Core.Filter;
using SabreTools.Core.Tools;
namespace SabreTools.DatFiles
{
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
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
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
}
}
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
}
}
[JsonIgnore]
public bool SearchSpecified
{
get
{
return GetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey) != null;
}
}
2019-01-08 11:06:26 -08:00
#endregion
#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();
// 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)
{
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];
}
}
// 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)
{
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];
}
}
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];
}
#endregion
2019-01-08 11:06:26 -08:00
#region Cloning Methods
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());
/// <summary>
2025-01-29 22:52:22 -05:00
/// Clone just the format from the current header
/// </summary>
2025-01-29 22:52:22 -05:00
public DatHeader CloneFormat()
{
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;
}
/// <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)
header.Remove(Models.Metadata.Header.ForceMergingKey);
2025-05-11 22:55:38 -04:00
if (header.ReadString(Models.Metadata.Header.ForceNodumpKey).AsNodumpFlag() == NodumpFlag.None)
header.Remove(Models.Metadata.Header.ForceNodumpKey);
2025-05-11 22:55:38 -04:00
if (header.ReadString(Models.Metadata.Header.ForcePackingKey).AsPackingFlag() == PackingFlag.None)
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);
// 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;
}
#endregion
#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>
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
}
}