2020-12-14 10:58:43 -08:00
|
|
|
|
using System.Xml.Serialization;
|
2022-11-03 12:22:17 -07:00
|
|
|
|
using Newtonsoft.Json;
|
2020-12-08 13:23:59 -08:00
|
|
|
|
using SabreTools.Core;
|
2020-12-08 16:37:08 -08:00
|
|
|
|
using SabreTools.Core.Tools;
|
2016-09-19 20:36:12 -07:00
|
|
|
|
|
2021-02-02 10:23:43 -08:00
|
|
|
|
namespace SabreTools.DatItems.Formats
|
2016-09-19 18:04:24 -07:00
|
|
|
|
{
|
2019-01-08 17:40:12 -08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Represents Compressed Hunks of Data (CHD) formatted disks which use internal hashes
|
|
|
|
|
|
/// </summary>
|
2020-09-08 10:12:41 -07:00
|
|
|
|
[JsonObject("disk"), XmlRoot("disk")]
|
2024-03-10 20:39:54 -04:00
|
|
|
|
public sealed class Disk : DatItem<Models.Metadata.Disk>
|
2019-01-08 17:40:12 -08:00
|
|
|
|
{
|
2024-03-10 16:49:07 -04:00
|
|
|
|
#region Constants
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Non-standard key for inverted logic
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public const string DiskAreaKey = "DISKAREA";
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Non-standard key for inverted logic
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public const string PartKey = "PART";
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
2020-08-20 13:17:14 -07:00
|
|
|
|
#region Fields
|
2019-01-08 17:40:12 -08:00
|
|
|
|
|
2024-03-10 20:39:54 -04:00
|
|
|
|
/// <inheritdoc>/>
|
|
|
|
|
|
protected override ItemType ItemType => ItemType.Disk;
|
|
|
|
|
|
|
2020-09-07 22:00:02 -07:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
|
public bool DiskAreaSpecified
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-03-10 16:49:07 -04:00
|
|
|
|
var diskArea = GetFieldValue<DiskArea?>(Disk.DiskAreaKey);
|
2024-03-09 21:34:26 -05:00
|
|
|
|
return diskArea != null && !string.IsNullOrEmpty(diskArea.GetName());
|
2020-09-07 22:00:02 -07:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[JsonIgnore]
|
|
|
|
|
|
public bool PartSpecified
|
|
|
|
|
|
{
|
|
|
|
|
|
get
|
|
|
|
|
|
{
|
2024-03-10 16:49:07 -04:00
|
|
|
|
var part = GetFieldValue<Part?>(Disk.PartKey);
|
2024-03-09 21:34:26 -05:00
|
|
|
|
return part != null
|
|
|
|
|
|
&& (!string.IsNullOrEmpty(part.GetName())
|
2024-03-11 15:46:44 -04:00
|
|
|
|
|| !string.IsNullOrEmpty(part.GetStringFieldValue(Models.Metadata.Part.InterfaceKey)));
|
2020-09-07 22:00:02 -07:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-09-03 11:02:06 -07:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
2019-01-08 17:40:12 -08:00
|
|
|
|
#region Constructors
|
|
|
|
|
|
|
2024-03-10 20:39:54 -04:00
|
|
|
|
public Disk() : base()
|
2019-01-08 17:40:12 -08:00
|
|
|
|
{
|
2024-03-10 16:49:07 -04:00
|
|
|
|
SetFieldValue<DupeType>(DatItem.DupeTypeKey, 0x00);
|
2024-03-12 00:20:36 -04:00
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.StatusKey, ItemStatus.None.AsStringValue());
|
2019-01-08 17:40:12 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-10 20:39:54 -04:00
|
|
|
|
public Disk(Models.Metadata.Disk item) : base(item)
|
2019-01-08 17:40:12 -08:00
|
|
|
|
{
|
2024-03-10 20:39:54 -04:00
|
|
|
|
SetFieldValue<DupeType>(DatItem.DupeTypeKey, 0x00);
|
2025-01-11 22:00:26 -05:00
|
|
|
|
|
|
|
|
|
|
// Process flag values
|
|
|
|
|
|
if (GetBoolFieldValue(Models.Metadata.Disk.OptionalKey) != null)
|
|
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.OptionalKey, GetBoolFieldValue(Models.Metadata.Disk.OptionalKey).FromYesNo());
|
|
|
|
|
|
if (GetStringFieldValue(Models.Metadata.Disk.StatusKey) != null)
|
|
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.StatusKey, GetStringFieldValue(Models.Metadata.Disk.StatusKey).AsEnumValue<ItemStatus>().AsStringValue());
|
|
|
|
|
|
if (GetBoolFieldValue(Models.Metadata.Disk.WritableKey) != null)
|
|
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.WritableKey, GetBoolFieldValue(Models.Metadata.Disk.WritableKey).FromYesNo());
|
|
|
|
|
|
|
|
|
|
|
|
// Process hash values
|
|
|
|
|
|
if (GetStringFieldValue(Models.Metadata.Disk.MD5Key) != null)
|
|
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.MD5Key, TextHelper.NormalizeMD5(GetStringFieldValue(Models.Metadata.Disk.MD5Key)));
|
|
|
|
|
|
if (GetStringFieldValue(Models.Metadata.Disk.SHA1Key) != null)
|
|
|
|
|
|
SetFieldValue<string?>(Models.Metadata.Disk.SHA1Key, TextHelper.NormalizeSHA1(GetStringFieldValue(Models.Metadata.Disk.SHA1Key)));
|
2024-03-10 20:39:54 -04:00
|
|
|
|
}
|
2023-08-15 01:38:01 -04:00
|
|
|
|
|
2019-01-08 17:40:12 -08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
#region Cloning Methods
|
|
|
|
|
|
|
2019-09-20 10:30:30 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Convert a disk to the closest Rom approximation
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public Rom ConvertToRom()
|
|
|
|
|
|
{
|
2024-03-10 17:00:11 -04:00
|
|
|
|
var rom = new Rom(_internal.ConvertToRom()!);
|
2025-01-08 15:22:54 -05:00
|
|
|
|
|
|
|
|
|
|
// Create a DataArea if there was an existing DiskArea
|
|
|
|
|
|
var diskArea = GetFieldValue<DiskArea?>(Disk.DiskAreaKey);
|
|
|
|
|
|
if (diskArea != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
var dataArea = new DataArea();
|
|
|
|
|
|
|
|
|
|
|
|
string? diskAreaName = diskArea.GetStringFieldValue(Models.Metadata.DiskArea.NameKey);
|
|
|
|
|
|
dataArea.SetFieldValue(Models.Metadata.DataArea.NameKey, diskAreaName);
|
|
|
|
|
|
|
|
|
|
|
|
rom.SetFieldValue<DataArea?>(Rom.DataAreaKey, dataArea);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-10 16:49:07 -04:00
|
|
|
|
rom.SetFieldValue<DupeType>(DatItem.DupeTypeKey, GetFieldValue<DupeType>(DatItem.DupeTypeKey));
|
2025-01-08 15:22:54 -05:00
|
|
|
|
rom.SetFieldValue<Machine>(DatItem.MachineKey, GetFieldValue<Machine>(DatItem.MachineKey)?.Clone() as Machine);
|
|
|
|
|
|
rom.SetFieldValue<Part>(Rom.PartKey, GetFieldValue<Part>(Disk.PartKey)?.Clone() as Part);
|
2024-03-11 15:46:44 -04:00
|
|
|
|
rom.SetFieldValue<bool?>(DatItem.RemoveKey, GetBoolFieldValue(DatItem.RemoveKey));
|
2025-01-08 15:22:54 -05:00
|
|
|
|
rom.SetFieldValue<Source?>(DatItem.SourceKey, GetFieldValue<Source?>(DatItem.SourceKey)?.Clone() as Source);
|
2024-03-08 20:42:24 -05:00
|
|
|
|
|
2019-09-20 10:30:30 -07:00
|
|
|
|
return rom;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2019-01-08 17:40:12 -08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
#region Comparision Methods
|
|
|
|
|
|
|
2020-08-17 17:28:32 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Fill any missing size and hash information from another Disk
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="other">Disk to fill information from</param>
|
2025-01-06 15:37:32 -05:00
|
|
|
|
public void FillMissingInformation(Disk other)
|
2025-01-07 14:55:56 -05:00
|
|
|
|
=> _internal.FillMissingHashes(other._internal);
|
2020-08-17 17:28:32 -07:00
|
|
|
|
|
2025-01-08 16:19:31 -05:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Returns if the Rom contains any hashes
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns>True if any hash exists, false otherwise</returns>
|
|
|
|
|
|
public bool HasHashes() => _internal.HasHashes();
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Returns if all of the hashes are set to their 0-byte values
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns>True if any hash matches the 0-byte value, false otherwise</returns>
|
|
|
|
|
|
public bool HasZeroHash() => _internal.HasZeroHash();
|
|
|
|
|
|
|
2024-03-05 01:42:42 -05:00
|
|
|
|
#endregion
|
2025-01-05 21:35:06 -05:00
|
|
|
|
|
2020-08-17 17:28:32 -07:00
|
|
|
|
#region Sorting and Merging
|
|
|
|
|
|
|
2020-12-14 15:31:28 -08:00
|
|
|
|
/// <inheritdoc/>
|
|
|
|
|
|
public override string GetKey(ItemKey bucketedBy, bool lower = true, bool norename = true)
|
2020-08-17 17:28:32 -07:00
|
|
|
|
{
|
|
|
|
|
|
// Set the output key as the default blank string
|
2023-08-14 13:17:51 -04:00
|
|
|
|
string? key;
|
2020-08-17 17:28:32 -07:00
|
|
|
|
|
|
|
|
|
|
// Now determine what the key should be based on the bucketedBy value
|
|
|
|
|
|
switch (bucketedBy)
|
|
|
|
|
|
{
|
2020-12-14 15:31:28 -08:00
|
|
|
|
case ItemKey.MD5:
|
2024-03-11 15:46:44 -04:00
|
|
|
|
key = GetStringFieldValue(Models.Metadata.Disk.MD5Key);
|
2020-08-17 17:28:32 -07:00
|
|
|
|
break;
|
|
|
|
|
|
|
2020-12-14 15:31:28 -08:00
|
|
|
|
case ItemKey.SHA1:
|
2024-03-11 15:46:44 -04:00
|
|
|
|
key = GetStringFieldValue(Models.Metadata.Disk.SHA1Key);
|
2020-08-17 17:28:32 -07:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
// Let the base handle generic stuff
|
|
|
|
|
|
default:
|
|
|
|
|
|
return base.GetKey(bucketedBy, lower, norename);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Double and triple check the key for corner cases
|
2023-04-19 16:39:58 -04:00
|
|
|
|
key ??= string.Empty;
|
2025-01-08 13:20:59 -05:00
|
|
|
|
if (lower)
|
|
|
|
|
|
key = key.ToLowerInvariant();
|
2020-08-17 17:28:32 -07:00
|
|
|
|
|
|
|
|
|
|
return key;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-08 13:32:09 -05:00
|
|
|
|
/// <inheritdoc/>
|
2025-01-14 14:22:18 -05:00
|
|
|
|
public override string GetKeyDB(ItemKey bucketedBy, Machine? machine, Source? source, bool lower = true, bool norename = true)
|
2025-01-08 13:32:09 -05:00
|
|
|
|
{
|
|
|
|
|
|
// Set the output key as the default blank string
|
|
|
|
|
|
string? key;
|
|
|
|
|
|
|
|
|
|
|
|
// Now determine what the key should be based on the bucketedBy value
|
|
|
|
|
|
switch (bucketedBy)
|
|
|
|
|
|
{
|
|
|
|
|
|
case ItemKey.MD5:
|
|
|
|
|
|
key = GetStringFieldValue(Models.Metadata.Disk.MD5Key);
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case ItemKey.SHA1:
|
|
|
|
|
|
key = GetStringFieldValue(Models.Metadata.Disk.SHA1Key);
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
// Let the base handle generic stuff
|
|
|
|
|
|
default:
|
2025-01-14 14:22:18 -05:00
|
|
|
|
return base.GetKeyDB(bucketedBy, machine, source, lower, norename);
|
2025-01-08 13:32:09 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Double and triple check the key for corner cases
|
|
|
|
|
|
key ??= string.Empty;
|
|
|
|
|
|
if (lower)
|
|
|
|
|
|
key = key.ToLowerInvariant();
|
|
|
|
|
|
|
|
|
|
|
|
return key;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-08-17 17:28:32 -07:00
|
|
|
|
#endregion
|
2019-01-08 17:40:12 -08:00
|
|
|
|
}
|
2016-09-19 18:04:24 -07:00
|
|
|
|
}
|