2024-03-11 01:39:55 -04:00
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using SabreTools.Core.Tools;
|
2024-03-13 00:02:19 -04:00
|
|
|
using SabreTools.DatItems;
|
2024-03-11 01:39:55 -04:00
|
|
|
|
|
|
|
|
namespace SabreTools.DatFiles
|
|
|
|
|
{
|
|
|
|
|
public partial class DatFile
|
|
|
|
|
{
|
|
|
|
|
#region To Metadata
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Convert metadata information
|
|
|
|
|
/// </summary>
|
2025-01-10 22:35:47 -05:00
|
|
|
internal Models.Metadata.MetadataFile? ConvertToMetadata(bool ignoreblanks = false)
|
2024-03-11 01:39:55 -04:00
|
|
|
{
|
|
|
|
|
// If we don't have items, we can't do anything
|
2025-01-14 10:38:46 -05:00
|
|
|
if (DatStatistics.TotalCount == 0)
|
2024-03-11 01:39:55 -04:00
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
// Create an object to hold the data
|
|
|
|
|
var metadataFile = new Models.Metadata.MetadataFile();
|
|
|
|
|
|
|
|
|
|
// Convert and assign the header
|
2025-01-11 23:34:26 -05:00
|
|
|
var header = Header.GetInternalClone();
|
2024-03-11 01:39:55 -04:00
|
|
|
if (header != null)
|
|
|
|
|
metadataFile[Models.Metadata.MetadataFile.HeaderKey] = header;
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
// Convert and assign the machines
|
|
|
|
|
var machines = ConvertMachines(ignoreblanks);
|
|
|
|
|
if (machines != null)
|
2024-03-11 14:31:02 -04:00
|
|
|
metadataFile[Models.Metadata.MetadataFile.MachineKey] = machines;
|
2024-03-11 01:39:55 -04:00
|
|
|
|
|
|
|
|
return metadataFile;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-20 11:36:14 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Convert metadata information
|
|
|
|
|
/// </summary>
|
2025-01-10 22:35:47 -05:00
|
|
|
internal Models.Metadata.MetadataFile? ConvertToMetadataDB(bool ignoreblanks = false)
|
2024-03-20 11:36:14 -04:00
|
|
|
{
|
|
|
|
|
// If we don't have items, we can't do anything
|
2025-01-14 10:38:46 -05:00
|
|
|
if (ItemsDB.DatStatistics.TotalCount == 0)
|
2024-03-20 11:36:14 -04:00
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
// Create an object to hold the data
|
|
|
|
|
var metadataFile = new Models.Metadata.MetadataFile();
|
|
|
|
|
|
|
|
|
|
// Convert and assign the header
|
2025-01-11 23:34:26 -05:00
|
|
|
var header = Header.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
if (header != null)
|
|
|
|
|
metadataFile[Models.Metadata.MetadataFile.HeaderKey] = header;
|
|
|
|
|
|
|
|
|
|
// Convert and assign the machines
|
|
|
|
|
var machines = ConvertMachinesDB(ignoreblanks);
|
|
|
|
|
if (machines != null)
|
|
|
|
|
metadataFile[Models.Metadata.MetadataFile.MachineKey] = machines;
|
|
|
|
|
|
|
|
|
|
return metadataFile;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Convert machines information
|
|
|
|
|
/// </summary>
|
|
|
|
|
private Models.Metadata.Machine[]? ConvertMachines(bool ignoreblanks = false)
|
|
|
|
|
{
|
|
|
|
|
// Create a machine list to hold all outputs
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Machine> machines = [];
|
2024-03-11 11:07:21 -04:00
|
|
|
|
|
|
|
|
// Loop through the sorted items and create games for them
|
|
|
|
|
foreach (string key in Items.SortedKeys)
|
|
|
|
|
{
|
2025-01-12 23:15:30 -05:00
|
|
|
var items = Items.GetItemsForBucket(key, filter: true);
|
2024-03-19 14:35:43 -04:00
|
|
|
if (items == null || items.Count == 0)
|
2024-03-11 11:07:21 -04:00
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// Create a machine to hold everything
|
2025-05-02 16:46:20 -04:00
|
|
|
var machine = items[0].GetMachine()!.GetInternalClone();
|
2024-03-11 11:07:21 -04:00
|
|
|
|
2024-03-12 11:53:58 -04:00
|
|
|
// Handle Trurip object, if it exists
|
|
|
|
|
if (machine.ContainsKey(Models.Metadata.Machine.TruripKey))
|
|
|
|
|
{
|
|
|
|
|
var trurip = machine.Read<DatItems.Trurip>(Models.Metadata.Machine.TruripKey);
|
|
|
|
|
if (trurip != null)
|
|
|
|
|
{
|
|
|
|
|
var truripItem = trurip.ConvertToLogiqx();
|
|
|
|
|
truripItem.Publisher = machine.ReadString(Models.Metadata.Machine.PublisherKey);
|
|
|
|
|
truripItem.Year = machine.ReadString(Models.Metadata.Machine.YearKey);
|
|
|
|
|
truripItem.Players = machine.ReadString(Models.Metadata.Machine.PlayersKey);
|
|
|
|
|
truripItem.Source = machine.ReadString(Models.Metadata.Machine.SourceFileKey);
|
|
|
|
|
truripItem.CloneOf = machine.ReadString(Models.Metadata.Machine.CloneOfKey);
|
|
|
|
|
machine[Models.Metadata.Machine.TruripKey] = truripItem;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 14:16:40 -04:00
|
|
|
// Create mapping dictionaries for the Parts, DataAreas, and DiskAreas associated with this machine
|
2024-03-12 14:34:52 -04:00
|
|
|
Dictionary<Models.Metadata.Part, Models.Metadata.DatItem> partMappings = [];
|
2024-03-12 14:54:19 -04:00
|
|
|
Dictionary<Models.Metadata.Part, (Models.Metadata.DataArea, Models.Metadata.Rom)> dataAreaMappings = [];
|
|
|
|
|
Dictionary<Models.Metadata.Part, (Models.Metadata.DiskArea, Models.Metadata.Disk)> diskAreaMappings = [];
|
2024-03-12 14:16:40 -04:00
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
// Loop through and convert the items to respective lists
|
|
|
|
|
for (int index = 0; index < items.Count; index++)
|
|
|
|
|
{
|
|
|
|
|
// Get the item
|
|
|
|
|
var item = items[index];
|
|
|
|
|
|
|
|
|
|
// Check for a "null" item
|
|
|
|
|
item = ProcessNullifiedItem(item);
|
|
|
|
|
|
|
|
|
|
// Skip if we're ignoring the item
|
|
|
|
|
if (ShouldIgnore(item, ignoreblanks))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
switch (item)
|
|
|
|
|
{
|
|
|
|
|
case DatItems.Formats.Adjuster adjuster:
|
2025-01-11 23:34:26 -05:00
|
|
|
var adjusterItem = adjuster.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Adjuster?>(machine, Models.Metadata.Machine.AdjusterKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.AdjusterKey, adjusterItem);
|
2024-03-20 11:36:14 -04:00
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Archive archive:
|
|
|
|
|
var archiveItem = archive.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Archive?>(machine, Models.Metadata.Machine.ArchiveKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ArchiveKey, archiveItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.BiosSet biosSet:
|
|
|
|
|
var biosSetItem = biosSet.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.BiosSet?>(machine, Models.Metadata.Machine.BiosSetKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.BiosSetKey, biosSetItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Chip chip:
|
|
|
|
|
var chipItem = chip.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Chip?>(machine, Models.Metadata.Machine.ChipKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ChipKey, chipItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Configuration configuration:
|
2025-01-11 23:34:26 -05:00
|
|
|
var configurationItem = configuration.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Configuration?>(machine, Models.Metadata.Machine.ConfigurationKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ConfigurationKey, configurationItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Device device:
|
2025-01-11 23:34:26 -05:00
|
|
|
var deviceItem = device.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Device?>(machine, Models.Metadata.Machine.DeviceKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DeviceKey, deviceItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.DeviceRef deviceRef:
|
|
|
|
|
var deviceRefItem = deviceRef.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.DeviceRef?>(machine, Models.Metadata.Machine.DeviceRefKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DeviceRefKey, deviceRefItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.DipSwitch dipSwitch:
|
2025-01-11 23:34:26 -05:00
|
|
|
var dipSwitchItem = dipSwitch.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.DipSwitch?>(machine, Models.Metadata.Machine.DipSwitchKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DipSwitchKey, dipSwitchItem);
|
|
|
|
|
|
|
|
|
|
// Add Part mapping
|
|
|
|
|
bool dipSwitchContainsPart = dipSwitchItem.ContainsKey(DatItems.Formats.DipSwitch.PartKey);
|
|
|
|
|
if (dipSwitchContainsPart)
|
|
|
|
|
{
|
|
|
|
|
var partItem = dipSwitchItem.Read<DatItems.Formats.Part>(DatItems.Formats.DipSwitch.PartKey);
|
|
|
|
|
if (partItem != null)
|
|
|
|
|
partMappings[partItem.GetInternalClone()] = dipSwitchItem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Disk disk:
|
|
|
|
|
var diskItem = disk.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Disk?>(machine, Models.Metadata.Machine.DiskKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DiskKey, diskItem);
|
|
|
|
|
|
|
|
|
|
// Add Part and DiskArea mappings
|
|
|
|
|
bool diskContainsPart = diskItem.ContainsKey(DatItems.Formats.Disk.PartKey);
|
|
|
|
|
bool diskContainsDiskArea = diskItem.ContainsKey(DatItems.Formats.Disk.DiskAreaKey);
|
|
|
|
|
if (diskContainsPart && diskContainsDiskArea)
|
|
|
|
|
{
|
|
|
|
|
var partItem = diskItem.Read<DatItems.Formats.Part>(DatItems.Formats.Disk.PartKey);
|
|
|
|
|
if (partItem != null)
|
|
|
|
|
{
|
|
|
|
|
var partItemInternal = partItem.GetInternalClone();
|
|
|
|
|
partMappings[partItemInternal] = diskItem;
|
|
|
|
|
|
|
|
|
|
var diskAreaItem = diskItem.Read<DatItems.Formats.DiskArea>(DatItems.Formats.Disk.DiskAreaKey);
|
|
|
|
|
if (diskAreaItem != null)
|
|
|
|
|
diskAreaMappings[partItemInternal] = (diskAreaItem.GetInternalClone(), diskItem);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Display display:
|
|
|
|
|
var displayItem = ProcessItem(display, machine);
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Display?>(machine, Models.Metadata.Machine.DisplayKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DisplayKey, displayItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Driver driver:
|
|
|
|
|
var driverItem = driver.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Driver?>(machine, Models.Metadata.Machine.DriverKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DriverKey, driverItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Feature feature:
|
|
|
|
|
var featureItem = feature.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Feature?>(machine, Models.Metadata.Machine.FeatureKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.FeatureKey, featureItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Info info:
|
|
|
|
|
var infoItem = info.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Info?>(machine, Models.Metadata.Machine.InfoKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.InfoKey, infoItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Input input:
|
2025-01-11 23:34:26 -05:00
|
|
|
var inputItem = input.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Input?>(machine, Models.Metadata.Machine.InputKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.InputKey, inputItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Media media:
|
|
|
|
|
var mediaItem = media.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Media?>(machine, Models.Metadata.Machine.MediaKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.MediaKey, mediaItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.PartFeature partFeature:
|
|
|
|
|
var partFeatureItem = partFeature.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Feature?>(machine, Models.Metadata.Machine.FeatureKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.FeatureKey, partFeatureItem);
|
|
|
|
|
|
|
|
|
|
// Add Part mapping
|
|
|
|
|
bool partFeatureContainsPart = partFeatureItem.ContainsKey(DatItems.Formats.PartFeature.PartKey);
|
|
|
|
|
if (partFeatureContainsPart)
|
|
|
|
|
{
|
|
|
|
|
var partItem = partFeatureItem.Read<DatItems.Formats.Part>(DatItems.Formats.PartFeature.PartKey);
|
|
|
|
|
if (partItem != null)
|
|
|
|
|
partMappings[partItem.GetInternalClone()] = partFeatureItem;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Port port:
|
2025-01-11 23:34:26 -05:00
|
|
|
var portItem = port.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Port?>(machine, Models.Metadata.Machine.PortKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.PortKey, portItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.RamOption ramOption:
|
|
|
|
|
var ramOptionItem = ramOption.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.RamOption?>(machine, Models.Metadata.Machine.RamOptionKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.RamOptionKey, ramOptionItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Release release:
|
|
|
|
|
var releaseItem = release.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Release?>(machine, Models.Metadata.Machine.ReleaseKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ReleaseKey, releaseItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Rom rom:
|
|
|
|
|
var romItem = ProcessItem(rom, machine);
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Rom?>(machine, Models.Metadata.Machine.RomKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.RomKey, romItem);
|
|
|
|
|
|
|
|
|
|
// Add Part and DataArea mappings
|
|
|
|
|
bool romContainsPart = romItem.ContainsKey(DatItems.Formats.Rom.PartKey);
|
|
|
|
|
bool romContainsDataArea = romItem.ContainsKey(DatItems.Formats.Rom.DataAreaKey);
|
|
|
|
|
if (romContainsPart && romContainsDataArea)
|
|
|
|
|
{
|
|
|
|
|
var partItem = romItem.Read<DatItems.Formats.Part>(DatItems.Formats.Rom.PartKey);
|
|
|
|
|
if (partItem != null)
|
|
|
|
|
{
|
|
|
|
|
var partItemInternal = partItem.GetInternalClone();
|
|
|
|
|
partMappings[partItemInternal] = romItem;
|
|
|
|
|
|
|
|
|
|
var dataAreaItem = romItem.Read<DatItems.Formats.DataArea>(DatItems.Formats.Rom.DataAreaKey);
|
|
|
|
|
if (dataAreaItem != null)
|
|
|
|
|
dataAreaMappings[partItemInternal] = (dataAreaItem.GetInternalClone(), romItem);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Sample sample:
|
|
|
|
|
var sampleItem = sample.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Sample?>(machine, Models.Metadata.Machine.SampleKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SampleKey, sampleItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.SharedFeat sharedFeat:
|
|
|
|
|
var sharedFeatItem = sharedFeat.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.SharedFeat?>(machine, Models.Metadata.Machine.SharedFeatKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SharedFeatKey, sharedFeatItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Slot slot:
|
2025-01-11 23:34:26 -05:00
|
|
|
var slotItem = slot.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Slot?>(machine, Models.Metadata.Machine.SlotKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SlotKey, slotItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.SoftwareList softwareList:
|
|
|
|
|
var softwareListItem = softwareList.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.SoftwareList?>(machine, Models.Metadata.Machine.SoftwareListKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SoftwareListKey, softwareListItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Sound sound:
|
|
|
|
|
var soundItem = sound.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Sound?>(machine, Models.Metadata.Machine.SoundKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SoundKey, soundItem);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Handle Part, DiskItem, and DatItem mappings, if they exist
|
|
|
|
|
if (partMappings.Count != 0)
|
|
|
|
|
{
|
|
|
|
|
// Create a collection to hold the inverted Parts
|
|
|
|
|
Dictionary<string, Models.Metadata.Part> partItems = [];
|
|
|
|
|
|
|
|
|
|
// Loop through the Part mappings
|
|
|
|
|
foreach (var partMapping in partMappings)
|
|
|
|
|
{
|
|
|
|
|
// Get the mapping pieces
|
|
|
|
|
var partItem = partMapping.Key;
|
|
|
|
|
var datItem = partMapping.Value;
|
|
|
|
|
|
|
|
|
|
// Get the part name and skip if there's none
|
|
|
|
|
string? partName = partItem.ReadString(Models.Metadata.Part.NameKey);
|
|
|
|
|
if (partName == null)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// Create the part in the dictionary, if needed
|
|
|
|
|
if (!partItems.ContainsKey(partName))
|
|
|
|
|
partItems[partName] = [];
|
|
|
|
|
|
|
|
|
|
// Copy over string values
|
|
|
|
|
partItems[partName][Models.Metadata.Part.NameKey] = partName;
|
|
|
|
|
if (!partItems[partName].ContainsKey(Models.Metadata.Part.InterfaceKey))
|
|
|
|
|
partItems[partName][Models.Metadata.Part.InterfaceKey] = partItem.ReadString(Models.Metadata.Part.InterfaceKey);
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(partItems[partName]);
|
|
|
|
|
|
|
|
|
|
// If the item has a DataArea mapping
|
|
|
|
|
if (dataAreaMappings.ContainsKey(partItem))
|
|
|
|
|
{
|
|
|
|
|
// Get the mapped items
|
|
|
|
|
var (dataArea, romItem) = dataAreaMappings[partItem];
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(romItem);
|
|
|
|
|
|
|
|
|
|
// Get the data area name and skip if there's none
|
|
|
|
|
string? dataAreaName = dataArea.ReadString(Models.Metadata.DataArea.NameKey);
|
|
|
|
|
if (dataAreaName != null)
|
|
|
|
|
{
|
|
|
|
|
// Get existing data areas as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var dataAreasArr = partItems[partName].Read<Models.Metadata.DataArea[]>(Models.Metadata.Part.DataAreaKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DataArea> dataAreas = [.. dataAreasArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Find the existing disk area to append to, otherwise create a new disk area
|
|
|
|
|
int dataAreaIndex = dataAreas.FindIndex(da => da.ReadString(Models.Metadata.DataArea.NameKey) == dataAreaName);
|
|
|
|
|
Models.Metadata.DataArea aggregateDataArea;
|
|
|
|
|
if (dataAreaIndex > -1)
|
|
|
|
|
{
|
|
|
|
|
aggregateDataArea = dataAreas[dataAreaIndex];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
aggregateDataArea = [];
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.EndiannessKey] = dataArea.ReadString(Models.Metadata.DataArea.EndiannessKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.NameKey] = dataArea.ReadString(Models.Metadata.DataArea.NameKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.SizeKey] = dataArea.ReadString(Models.Metadata.DataArea.SizeKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.WidthKey] = dataArea.ReadString(Models.Metadata.DataArea.WidthKey);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(aggregateDataArea);
|
|
|
|
|
|
|
|
|
|
// Get existing roms as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var romsArr = aggregateDataArea.Read<Models.Metadata.Rom[]>(Models.Metadata.DataArea.RomKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Rom> roms = [.. romsArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Add the rom to the data area
|
|
|
|
|
roms.Add(romItem);
|
|
|
|
|
|
|
|
|
|
// Assign back the roms
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.RomKey] = roms.ToArray();
|
|
|
|
|
|
|
|
|
|
// Assign back the data area
|
|
|
|
|
if (dataAreaIndex > -1)
|
|
|
|
|
dataAreas[dataAreaIndex] = aggregateDataArea;
|
|
|
|
|
else
|
|
|
|
|
dataAreas.Add(aggregateDataArea);
|
|
|
|
|
|
|
|
|
|
// Assign back the data areas array
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DataAreaKey] = dataAreas.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the item has a DiskArea mapping
|
|
|
|
|
if (diskAreaMappings.ContainsKey(partItem))
|
|
|
|
|
{
|
|
|
|
|
// Get the mapped items
|
|
|
|
|
var (diskArea, diskItem) = diskAreaMappings[partItem];
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(diskItem);
|
|
|
|
|
|
|
|
|
|
// Get the disk area name and skip if there's none
|
|
|
|
|
string? diskAreaName = diskArea.ReadString(Models.Metadata.DiskArea.NameKey);
|
|
|
|
|
if (diskAreaName != null)
|
|
|
|
|
{
|
2024-10-19 21:41:08 -04:00
|
|
|
// Get existing disk areas as a list
|
|
|
|
|
var diskAreasArr = partItems[partName].Read<Models.Metadata.DiskArea[]>(Models.Metadata.Part.DiskAreaKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DiskArea> diskAreas = [.. diskAreasArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Find the existing disk area to append to, otherwise create a new disk area
|
|
|
|
|
int diskAreaIndex = diskAreas.FindIndex(da => da.ReadString(Models.Metadata.DiskArea.NameKey) == diskAreaName);
|
|
|
|
|
Models.Metadata.DiskArea aggregateDiskArea;
|
|
|
|
|
if (diskAreaIndex > -1)
|
|
|
|
|
{
|
|
|
|
|
aggregateDiskArea = diskAreas[diskAreaIndex];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
aggregateDiskArea = [];
|
|
|
|
|
aggregateDiskArea[Models.Metadata.DiskArea.NameKey] = diskArea.ReadString(Models.Metadata.DiskArea.NameKey);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(aggregateDiskArea);
|
|
|
|
|
|
|
|
|
|
// Get existing disks as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var disksArr = aggregateDiskArea.Read<Models.Metadata.Disk[]>(Models.Metadata.DiskArea.DiskKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Disk> disks = [.. disksArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Add the disk to the data area
|
|
|
|
|
disks.Add(diskItem);
|
|
|
|
|
|
|
|
|
|
// Assign back the disks
|
|
|
|
|
aggregateDiskArea[Models.Metadata.DiskArea.DiskKey] = disks.ToArray();
|
|
|
|
|
|
|
|
|
|
// Assign back the disk area
|
|
|
|
|
if (diskAreaIndex > -1)
|
|
|
|
|
diskAreas[diskAreaIndex] = aggregateDiskArea;
|
|
|
|
|
else
|
|
|
|
|
diskAreas.Add(aggregateDiskArea);
|
|
|
|
|
|
|
|
|
|
// Assign back the disk areas array
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DiskAreaKey] = diskAreas.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the item is a DipSwitch
|
|
|
|
|
if (datItem is Models.Metadata.DipSwitch dipSwitchItem)
|
|
|
|
|
{
|
|
|
|
|
// Get existing dipswitches as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var dipSwitchesArr = partItems[partName].Read<Models.Metadata.DipSwitch[]>(Models.Metadata.Part.DipSwitchKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DipSwitch> dipSwitches = [.. dipSwitchesArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(dipSwitchItem);
|
|
|
|
|
|
|
|
|
|
// Add the dipswitch
|
|
|
|
|
dipSwitches.Add(dipSwitchItem);
|
|
|
|
|
|
|
|
|
|
// Assign back the dipswitches
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DipSwitchKey] = dipSwitches.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the item is a Feature
|
|
|
|
|
else if (datItem is Models.Metadata.Feature featureItem)
|
|
|
|
|
{
|
|
|
|
|
// Get existing features as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var featuresArr = partItems[partName].Read<Models.Metadata.Feature[]>(Models.Metadata.Part.FeatureKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Feature> features = [.. featuresArr];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(featureItem);
|
|
|
|
|
|
|
|
|
|
// Add the feature
|
|
|
|
|
features.Add(featureItem);
|
|
|
|
|
|
|
|
|
|
// Assign back the features
|
|
|
|
|
partItems[partName][Models.Metadata.Part.FeatureKey] = features.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Assign the part array to the machine
|
|
|
|
|
machine[Models.Metadata.Machine.PartKey] = partItems.Values.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add the machine to the list
|
|
|
|
|
machines.Add(machine);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Return the list of machines
|
|
|
|
|
return [.. machines];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Convert machines information
|
|
|
|
|
/// </summary>
|
|
|
|
|
private Models.Metadata.Machine[]? ConvertMachinesDB(bool ignoreblanks = false)
|
|
|
|
|
{
|
|
|
|
|
// Create a machine list to hold all outputs
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Machine> machines = [];
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Loop through the sorted items and create games for them
|
|
|
|
|
foreach (string key in ItemsDB.SortedKeys)
|
|
|
|
|
{
|
2025-01-12 23:15:30 -05:00
|
|
|
var items = GetItemsForBucketDB(key, filter: true);
|
2024-12-06 23:16:09 -05:00
|
|
|
if (items == null || items.Count == 0)
|
2024-03-20 11:36:14 -04:00
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// Create a machine to hold everything
|
2025-02-24 09:20:46 -05:00
|
|
|
var machine = GetMachineForItemDB(items.First().Key).Value!.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Handle Trurip object, if it exists
|
|
|
|
|
if (machine.ContainsKey(Models.Metadata.Machine.TruripKey))
|
|
|
|
|
{
|
|
|
|
|
var trurip = machine.Read<DatItems.Trurip>(Models.Metadata.Machine.TruripKey);
|
|
|
|
|
if (trurip != null)
|
|
|
|
|
{
|
|
|
|
|
var truripItem = trurip.ConvertToLogiqx();
|
|
|
|
|
truripItem.Publisher = machine.ReadString(Models.Metadata.Machine.PublisherKey);
|
|
|
|
|
truripItem.Year = machine.ReadString(Models.Metadata.Machine.YearKey);
|
|
|
|
|
truripItem.Players = machine.ReadString(Models.Metadata.Machine.PlayersKey);
|
|
|
|
|
truripItem.Source = machine.ReadString(Models.Metadata.Machine.SourceFileKey);
|
|
|
|
|
truripItem.CloneOf = machine.ReadString(Models.Metadata.Machine.CloneOfKey);
|
|
|
|
|
machine[Models.Metadata.Machine.TruripKey] = truripItem;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create mapping dictionaries for the Parts, DataAreas, and DiskAreas associated with this machine
|
|
|
|
|
Dictionary<Models.Metadata.Part, Models.Metadata.DatItem> partMappings = [];
|
|
|
|
|
Dictionary<Models.Metadata.Part, (Models.Metadata.DataArea, Models.Metadata.Rom)> dataAreaMappings = [];
|
|
|
|
|
Dictionary<Models.Metadata.Part, (Models.Metadata.DiskArea, Models.Metadata.Disk)> diskAreaMappings = [];
|
|
|
|
|
|
|
|
|
|
// Loop through and convert the items to respective lists
|
2024-12-06 23:16:09 -05:00
|
|
|
foreach (var kvp in items)
|
2024-03-20 11:36:14 -04:00
|
|
|
{
|
|
|
|
|
// Check for a "null" item
|
2025-01-09 10:01:56 -05:00
|
|
|
var item = new KeyValuePair<long, DatItem>(kvp.Key, ProcessNullifiedItem(kvp.Value));
|
2024-03-20 11:36:14 -04:00
|
|
|
|
|
|
|
|
// Skip if we're ignoring the item
|
2024-12-06 23:16:09 -05:00
|
|
|
if (ShouldIgnore(item.Value, ignoreblanks))
|
2024-03-20 11:36:14 -04:00
|
|
|
continue;
|
|
|
|
|
|
2024-12-06 23:16:09 -05:00
|
|
|
switch (item.Value)
|
2024-03-20 11:36:14 -04:00
|
|
|
{
|
|
|
|
|
case DatItems.Formats.Adjuster adjuster:
|
2025-01-11 23:34:26 -05:00
|
|
|
var adjusterItem = adjuster.GetInternalClone();
|
2024-03-20 11:36:14 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Adjuster?>(machine, Models.Metadata.Machine.AdjusterKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.AdjusterKey, adjusterItem);
|
2024-03-11 11:07:21 -04:00
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Archive archive:
|
|
|
|
|
var archiveItem = archive.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Archive?>(machine, Models.Metadata.Machine.ArchiveKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ArchiveKey, archiveItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.BiosSet biosSet:
|
|
|
|
|
var biosSetItem = biosSet.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.BiosSet?>(machine, Models.Metadata.Machine.BiosSetKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.BiosSetKey, biosSetItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Chip chip:
|
|
|
|
|
var chipItem = chip.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Chip?>(machine, Models.Metadata.Machine.ChipKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ChipKey, chipItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Configuration configuration:
|
2025-01-11 23:34:26 -05:00
|
|
|
var configurationItem = configuration.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Configuration?>(machine, Models.Metadata.Machine.ConfigurationKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ConfigurationKey, configurationItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Device device:
|
2025-01-11 23:34:26 -05:00
|
|
|
var deviceItem = device.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Device?>(machine, Models.Metadata.Machine.DeviceKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DeviceKey, deviceItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.DeviceRef deviceRef:
|
|
|
|
|
var deviceRefItem = deviceRef.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.DeviceRef?>(machine, Models.Metadata.Machine.DeviceRefKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DeviceRefKey, deviceRefItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.DipSwitch dipSwitch:
|
2025-01-11 23:34:26 -05:00
|
|
|
var dipSwitchItem = dipSwitch.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.DipSwitch?>(machine, Models.Metadata.Machine.DipSwitchKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DipSwitchKey, dipSwitchItem);
|
2024-03-12 14:16:40 -04:00
|
|
|
|
|
|
|
|
// Add Part mapping
|
|
|
|
|
bool dipSwitchContainsPart = dipSwitchItem.ContainsKey(DatItems.Formats.DipSwitch.PartKey);
|
|
|
|
|
if (dipSwitchContainsPart)
|
|
|
|
|
{
|
|
|
|
|
var partItem = dipSwitchItem.Read<DatItems.Formats.Part>(DatItems.Formats.DipSwitch.PartKey);
|
|
|
|
|
if (partItem != null)
|
2024-03-12 14:34:52 -04:00
|
|
|
partMappings[partItem.GetInternalClone()] = dipSwitchItem;
|
2024-03-12 14:16:40 -04:00
|
|
|
}
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Disk disk:
|
2024-03-12 14:16:40 -04:00
|
|
|
var diskItem = disk.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Disk?>(machine, Models.Metadata.Machine.DiskKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DiskKey, diskItem);
|
2024-03-12 14:16:40 -04:00
|
|
|
|
|
|
|
|
// Add Part and DiskArea mappings
|
|
|
|
|
bool diskContainsPart = diskItem.ContainsKey(DatItems.Formats.Disk.PartKey);
|
|
|
|
|
bool diskContainsDiskArea = diskItem.ContainsKey(DatItems.Formats.Disk.DiskAreaKey);
|
|
|
|
|
if (diskContainsPart && diskContainsDiskArea)
|
|
|
|
|
{
|
|
|
|
|
var partItem = diskItem.Read<DatItems.Formats.Part>(DatItems.Formats.Disk.PartKey);
|
|
|
|
|
if (partItem != null)
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
|
|
|
|
var partItemInternal = partItem.GetInternalClone();
|
|
|
|
|
partMappings[partItemInternal] = diskItem;
|
|
|
|
|
|
|
|
|
|
var diskAreaItem = diskItem.Read<DatItems.Formats.DiskArea>(DatItems.Formats.Disk.DiskAreaKey);
|
|
|
|
|
if (diskAreaItem != null)
|
|
|
|
|
diskAreaMappings[partItemInternal] = (diskAreaItem.GetInternalClone(), diskItem);
|
|
|
|
|
}
|
2024-03-12 14:16:40 -04:00
|
|
|
}
|
2024-03-11 11:07:21 -04:00
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Display display:
|
2024-03-12 12:10:36 -04:00
|
|
|
var displayItem = ProcessItem(display, machine);
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Display?>(machine, Models.Metadata.Machine.DisplayKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DisplayKey, displayItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Driver driver:
|
|
|
|
|
var driverItem = driver.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Driver?>(machine, Models.Metadata.Machine.DriverKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DriverKey, driverItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Feature feature:
|
|
|
|
|
var featureItem = feature.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Feature?>(machine, Models.Metadata.Machine.FeatureKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.FeatureKey, featureItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Info info:
|
|
|
|
|
var infoItem = info.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Info?>(machine, Models.Metadata.Machine.InfoKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.InfoKey, infoItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Input input:
|
2025-01-11 23:34:26 -05:00
|
|
|
var inputItem = input.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Input?>(machine, Models.Metadata.Machine.InputKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.InputKey, inputItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Media media:
|
|
|
|
|
var mediaItem = media.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Media?>(machine, Models.Metadata.Machine.MediaKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.MediaKey, mediaItem);
|
|
|
|
|
break;
|
2024-03-12 14:34:52 -04:00
|
|
|
case DatItems.Formats.PartFeature partFeature:
|
|
|
|
|
var partFeatureItem = partFeature.GetInternalClone();
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Feature?>(machine, Models.Metadata.Machine.FeatureKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.FeatureKey, partFeatureItem);
|
|
|
|
|
|
|
|
|
|
// Add Part mapping
|
|
|
|
|
bool partFeatureContainsPart = partFeatureItem.ContainsKey(DatItems.Formats.PartFeature.PartKey);
|
|
|
|
|
if (partFeatureContainsPart)
|
|
|
|
|
{
|
|
|
|
|
var partItem = partFeatureItem.Read<DatItems.Formats.Part>(DatItems.Formats.PartFeature.PartKey);
|
|
|
|
|
if (partItem != null)
|
|
|
|
|
partMappings[partItem.GetInternalClone()] = partFeatureItem;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2024-03-11 11:07:21 -04:00
|
|
|
case DatItems.Formats.Port port:
|
2025-01-11 23:34:26 -05:00
|
|
|
var portItem = port.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Port?>(machine, Models.Metadata.Machine.PortKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.PortKey, portItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.RamOption ramOption:
|
|
|
|
|
var ramOptionItem = ramOption.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.RamOption?>(machine, Models.Metadata.Machine.RamOptionKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.RamOptionKey, ramOptionItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Release release:
|
|
|
|
|
var releaseItem = release.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Release?>(machine, Models.Metadata.Machine.ReleaseKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.ReleaseKey, releaseItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Rom rom:
|
2024-03-11 13:19:50 -04:00
|
|
|
var romItem = ProcessItem(rom, machine);
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Rom?>(machine, Models.Metadata.Machine.RomKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.RomKey, romItem);
|
2024-03-12 14:16:40 -04:00
|
|
|
|
|
|
|
|
// Add Part and DataArea mappings
|
|
|
|
|
bool romContainsPart = romItem.ContainsKey(DatItems.Formats.Rom.PartKey);
|
|
|
|
|
bool romContainsDataArea = romItem.ContainsKey(DatItems.Formats.Rom.DataAreaKey);
|
|
|
|
|
if (romContainsPart && romContainsDataArea)
|
|
|
|
|
{
|
|
|
|
|
var partItem = romItem.Read<DatItems.Formats.Part>(DatItems.Formats.Rom.PartKey);
|
|
|
|
|
if (partItem != null)
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
|
|
|
|
var partItemInternal = partItem.GetInternalClone();
|
|
|
|
|
partMappings[partItemInternal] = romItem;
|
|
|
|
|
|
|
|
|
|
var dataAreaItem = romItem.Read<DatItems.Formats.DataArea>(DatItems.Formats.Rom.DataAreaKey);
|
|
|
|
|
if (dataAreaItem != null)
|
|
|
|
|
dataAreaMappings[partItemInternal] = (dataAreaItem.GetInternalClone(), romItem);
|
|
|
|
|
}
|
2024-03-12 14:16:40 -04:00
|
|
|
}
|
2024-03-11 11:07:21 -04:00
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Sample sample:
|
|
|
|
|
var sampleItem = sample.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Sample?>(machine, Models.Metadata.Machine.SampleKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SampleKey, sampleItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.SharedFeat sharedFeat:
|
|
|
|
|
var sharedFeatItem = sharedFeat.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.SharedFeat?>(machine, Models.Metadata.Machine.SharedFeatKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SharedFeatKey, sharedFeatItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Slot slot:
|
2025-01-11 23:34:26 -05:00
|
|
|
var slotItem = slot.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Slot?>(machine, Models.Metadata.Machine.SlotKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SlotKey, slotItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.SoftwareList softwareList:
|
|
|
|
|
var softwareListItem = softwareList.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.SoftwareList?>(machine, Models.Metadata.Machine.SoftwareListKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SoftwareListKey, softwareListItem);
|
|
|
|
|
break;
|
|
|
|
|
case DatItems.Formats.Sound sound:
|
|
|
|
|
var soundItem = sound.GetInternalClone();
|
2024-03-11 14:31:02 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Sound?>(machine, Models.Metadata.Machine.SoundKey);
|
2024-03-11 11:07:21 -04:00
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.SoundKey, soundItem);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 14:34:52 -04:00
|
|
|
// Handle Part, DiskItem, and DatItem mappings, if they exist
|
2024-03-19 14:35:43 -04:00
|
|
|
if (partMappings.Count != 0)
|
2024-03-12 14:34:52 -04:00
|
|
|
{
|
|
|
|
|
// Create a collection to hold the inverted Parts
|
2024-03-12 14:54:19 -04:00
|
|
|
Dictionary<string, Models.Metadata.Part> partItems = [];
|
2024-03-12 14:34:52 -04:00
|
|
|
|
|
|
|
|
// Loop through the Part mappings
|
|
|
|
|
foreach (var partMapping in partMappings)
|
|
|
|
|
{
|
2024-03-12 14:54:19 -04:00
|
|
|
// Get the mapping pieces
|
|
|
|
|
var partItem = partMapping.Key;
|
|
|
|
|
var datItem = partMapping.Value;
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// Get the part name and skip if there's none
|
|
|
|
|
string? partName = partItem.ReadString(Models.Metadata.Part.NameKey);
|
|
|
|
|
if (partName == null)
|
|
|
|
|
continue;
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// Create the part in the dictionary, if needed
|
|
|
|
|
if (!partItems.ContainsKey(partName))
|
|
|
|
|
partItems[partName] = [];
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// Copy over string values
|
|
|
|
|
partItems[partName][Models.Metadata.Part.NameKey] = partName;
|
|
|
|
|
if (!partItems[partName].ContainsKey(Models.Metadata.Part.InterfaceKey))
|
|
|
|
|
partItems[partName][Models.Metadata.Part.InterfaceKey] = partItem.ReadString(Models.Metadata.Part.InterfaceKey);
|
|
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(partItems[partName]);
|
|
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// If the item has a DataArea mapping
|
|
|
|
|
if (dataAreaMappings.ContainsKey(partItem))
|
|
|
|
|
{
|
|
|
|
|
// Get the mapped items
|
|
|
|
|
var (dataArea, romItem) = dataAreaMappings[partItem];
|
|
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(romItem);
|
|
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// Get the data area name and skip if there's none
|
|
|
|
|
string? dataAreaName = dataArea.ReadString(Models.Metadata.DataArea.NameKey);
|
|
|
|
|
if (dataAreaName != null)
|
|
|
|
|
{
|
|
|
|
|
// Get existing data areas as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var dataAreasArr = partItems[partName].Read<Models.Metadata.DataArea[]>(Models.Metadata.Part.DataAreaKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DataArea> dataAreas = [.. dataAreasArr];
|
2024-03-12 14:54:19 -04:00
|
|
|
|
|
|
|
|
// Find the existing disk area to append to, otherwise create a new disk area
|
|
|
|
|
int dataAreaIndex = dataAreas.FindIndex(da => da.ReadString(Models.Metadata.DataArea.NameKey) == dataAreaName);
|
|
|
|
|
Models.Metadata.DataArea aggregateDataArea;
|
|
|
|
|
if (dataAreaIndex > -1)
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
2024-03-12 14:54:19 -04:00
|
|
|
aggregateDataArea = dataAreas[dataAreaIndex];
|
2024-03-12 16:17:05 -04:00
|
|
|
}
|
2024-03-12 14:54:19 -04:00
|
|
|
else
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
2024-03-12 14:54:19 -04:00
|
|
|
aggregateDataArea = [];
|
2024-03-12 16:17:05 -04:00
|
|
|
aggregateDataArea[Models.Metadata.DataArea.EndiannessKey] = dataArea.ReadString(Models.Metadata.DataArea.EndiannessKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.NameKey] = dataArea.ReadString(Models.Metadata.DataArea.NameKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.SizeKey] = dataArea.ReadString(Models.Metadata.DataArea.SizeKey);
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.WidthKey] = dataArea.ReadString(Models.Metadata.DataArea.WidthKey);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(aggregateDataArea);
|
2024-03-12 14:54:19 -04:00
|
|
|
|
2024-03-12 15:03:31 -04:00
|
|
|
// Get existing roms as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var romsArr = aggregateDataArea.Read<Models.Metadata.Rom[]>(Models.Metadata.DataArea.RomKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Rom> roms = [.. romsArr];
|
2024-03-12 15:03:31 -04:00
|
|
|
|
|
|
|
|
// Add the rom to the data area
|
|
|
|
|
roms.Add(romItem);
|
|
|
|
|
|
|
|
|
|
// Assign back the roms
|
|
|
|
|
aggregateDataArea[Models.Metadata.DataArea.RomKey] = roms.ToArray();
|
|
|
|
|
|
|
|
|
|
// Assign back the data area
|
|
|
|
|
if (dataAreaIndex > -1)
|
|
|
|
|
dataAreas[dataAreaIndex] = aggregateDataArea;
|
|
|
|
|
else
|
|
|
|
|
dataAreas.Add(aggregateDataArea);
|
|
|
|
|
|
|
|
|
|
// Assign back the data areas array
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DataAreaKey] = dataAreas.ToArray();
|
2024-03-12 14:54:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// If the item has a DiskArea mapping
|
|
|
|
|
if (diskAreaMappings.ContainsKey(partItem))
|
|
|
|
|
{
|
|
|
|
|
// Get the mapped items
|
|
|
|
|
var (diskArea, diskItem) = diskAreaMappings[partItem];
|
|
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(diskItem);
|
|
|
|
|
|
2024-03-12 14:54:19 -04:00
|
|
|
// Get the disk area name and skip if there's none
|
|
|
|
|
string? diskAreaName = diskArea.ReadString(Models.Metadata.DiskArea.NameKey);
|
|
|
|
|
if (diskAreaName != null)
|
|
|
|
|
{
|
2024-10-19 21:41:08 -04:00
|
|
|
// Get existing disk areas as a list
|
|
|
|
|
var diskAreasArr = partItems[partName].Read<Models.Metadata.DiskArea[]>(Models.Metadata.Part.DiskAreaKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DiskArea> diskAreas = [.. diskAreasArr];
|
2024-03-12 14:54:19 -04:00
|
|
|
|
|
|
|
|
// Find the existing disk area to append to, otherwise create a new disk area
|
|
|
|
|
int diskAreaIndex = diskAreas.FindIndex(da => da.ReadString(Models.Metadata.DiskArea.NameKey) == diskAreaName);
|
|
|
|
|
Models.Metadata.DiskArea aggregateDiskArea;
|
|
|
|
|
if (diskAreaIndex > -1)
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
2024-03-12 14:54:19 -04:00
|
|
|
aggregateDiskArea = diskAreas[diskAreaIndex];
|
2024-03-12 16:17:05 -04:00
|
|
|
}
|
2024-03-12 14:54:19 -04:00
|
|
|
else
|
2024-03-12 16:17:05 -04:00
|
|
|
{
|
2024-03-12 14:54:19 -04:00
|
|
|
aggregateDiskArea = [];
|
2024-03-12 16:17:05 -04:00
|
|
|
aggregateDiskArea[Models.Metadata.DiskArea.NameKey] = diskArea.ReadString(Models.Metadata.DiskArea.NameKey);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(aggregateDiskArea);
|
2024-03-12 14:54:19 -04:00
|
|
|
|
2024-03-12 15:03:31 -04:00
|
|
|
// Get existing disks as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var disksArr = aggregateDiskArea.Read<Models.Metadata.Disk[]>(Models.Metadata.DiskArea.DiskKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Disk> disks = [.. disksArr];
|
2024-03-12 15:03:31 -04:00
|
|
|
|
|
|
|
|
// Add the disk to the data area
|
2024-03-12 16:17:05 -04:00
|
|
|
disks.Add(diskItem);
|
2024-03-12 15:03:31 -04:00
|
|
|
|
|
|
|
|
// Assign back the disks
|
2024-03-12 16:17:05 -04:00
|
|
|
aggregateDiskArea[Models.Metadata.DiskArea.DiskKey] = disks.ToArray();
|
2024-03-12 15:03:31 -04:00
|
|
|
|
|
|
|
|
// Assign back the disk area
|
|
|
|
|
if (diskAreaIndex > -1)
|
|
|
|
|
diskAreas[diskAreaIndex] = aggregateDiskArea;
|
|
|
|
|
else
|
|
|
|
|
diskAreas.Add(aggregateDiskArea);
|
|
|
|
|
|
|
|
|
|
// Assign back the disk areas array
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DiskAreaKey] = diskAreas.ToArray();
|
2024-03-12 14:54:19 -04:00
|
|
|
}
|
|
|
|
|
}
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// If the item is a DipSwitch
|
|
|
|
|
if (datItem is Models.Metadata.DipSwitch dipSwitchItem)
|
|
|
|
|
{
|
|
|
|
|
// Get existing dipswitches as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var dipSwitchesArr = partItems[partName].Read<Models.Metadata.DipSwitch[]>(Models.Metadata.Part.DipSwitchKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.DipSwitch> dipSwitches = [.. dipSwitchesArr];
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(dipSwitchItem);
|
|
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// Add the dipswitch
|
|
|
|
|
dipSwitches.Add(dipSwitchItem);
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// Assign back the dipswitches
|
|
|
|
|
partItems[partName][Models.Metadata.Part.DipSwitchKey] = dipSwitches.ToArray();
|
|
|
|
|
}
|
2024-03-12 16:17:05 -04:00
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// If the item is a Feature
|
|
|
|
|
else if (datItem is Models.Metadata.Feature featureItem)
|
|
|
|
|
{
|
|
|
|
|
// Get existing features as a list
|
2024-10-19 21:41:08 -04:00
|
|
|
var featuresArr = partItems[partName].Read<Models.Metadata.Feature[]>(Models.Metadata.Part.FeatureKey) ?? [];
|
2024-12-06 23:16:09 -05:00
|
|
|
List<Models.Metadata.Feature> features = [.. featuresArr];
|
2024-03-12 15:09:53 -04:00
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Clear any empty fields
|
|
|
|
|
ClearEmptyKeys(featureItem);
|
|
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// Add the feature
|
|
|
|
|
features.Add(featureItem);
|
2024-03-12 14:34:52 -04:00
|
|
|
|
2024-03-12 15:09:53 -04:00
|
|
|
// Assign back the features
|
|
|
|
|
partItems[partName][Models.Metadata.Part.FeatureKey] = features.ToArray();
|
|
|
|
|
}
|
2024-03-12 14:34:52 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Assign the part array to the machine
|
|
|
|
|
machine[Models.Metadata.Machine.PartKey] = partItems.Values.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
// Add the machine to the list
|
|
|
|
|
machines.Add(machine);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Return the list of machines
|
|
|
|
|
return [.. machines];
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 11:53:58 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Convert Display information
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="item">Item to convert</param>
|
2024-03-12 12:10:36 -04:00
|
|
|
/// <param name="machine">Machine to use for Video</param>
|
2025-01-11 23:34:26 -05:00
|
|
|
/// <remarks>
|
|
|
|
|
/// This method is required because both a Display and a Video
|
|
|
|
|
/// item might be created and added for a given Display input.
|
|
|
|
|
/// </remarks>
|
2024-03-12 12:10:36 -04:00
|
|
|
private static Models.Metadata.Display ProcessItem(DatItems.Formats.Display item, Models.Metadata.Machine machine)
|
2024-03-12 11:53:58 -04:00
|
|
|
{
|
|
|
|
|
var displayItem = item.GetInternalClone();
|
|
|
|
|
|
2024-03-12 12:10:36 -04:00
|
|
|
// Create a Video for any item that has specific fields
|
|
|
|
|
if (displayItem.ContainsKey(Models.Metadata.Video.AspectXKey))
|
|
|
|
|
{
|
|
|
|
|
var videoItem = new Models.Metadata.Video();
|
|
|
|
|
videoItem[Models.Metadata.Video.AspectXKey] = displayItem.ReadLong(Models.Metadata.Video.AspectXKey).ToString();
|
|
|
|
|
videoItem[Models.Metadata.Video.AspectYKey] = displayItem.ReadLong(Models.Metadata.Video.AspectYKey).ToString();
|
|
|
|
|
videoItem[Models.Metadata.Video.HeightKey] = displayItem.ReadLong(Models.Metadata.Display.HeightKey).ToString();
|
|
|
|
|
videoItem[Models.Metadata.Video.RefreshKey] = displayItem.ReadDouble(Models.Metadata.Display.RefreshKey).ToString();
|
2025-05-11 22:55:38 -04:00
|
|
|
videoItem[Models.Metadata.Video.ScreenKey] = displayItem.ReadString(Models.Metadata.Display.DisplayTypeKey).AsDisplayType().AsStringValue();
|
2024-03-12 12:10:36 -04:00
|
|
|
videoItem[Models.Metadata.Video.WidthKey] = displayItem.ReadLong(Models.Metadata.Display.WidthKey).ToString();
|
|
|
|
|
|
2025-01-11 23:03:32 -05:00
|
|
|
switch (displayItem.ReadLong(Models.Metadata.Display.RotateKey))
|
|
|
|
|
{
|
|
|
|
|
case 0:
|
|
|
|
|
case 180:
|
|
|
|
|
videoItem[Models.Metadata.Video.OrientationKey] = "horizontal";
|
|
|
|
|
break;
|
|
|
|
|
case 90:
|
|
|
|
|
case 270:
|
|
|
|
|
videoItem[Models.Metadata.Video.OrientationKey] = "vertical";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 12:10:36 -04:00
|
|
|
EnsureMachineKey<Models.Metadata.Video?>(machine, Models.Metadata.Machine.VideoKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.VideoKey, videoItem);
|
|
|
|
|
}
|
2024-03-12 11:53:58 -04:00
|
|
|
|
|
|
|
|
return displayItem;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Convert Rom information
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="item">Item to convert</param>
|
2024-03-11 13:19:50 -04:00
|
|
|
/// <param name="machine">Machine to use for Part and DataArea</param>
|
2025-01-11 23:34:26 -05:00
|
|
|
/// <remarks>
|
|
|
|
|
/// This method is required because both a Rom and a Dump
|
|
|
|
|
/// item might be created and added for a given Rom input.
|
|
|
|
|
/// </remarks>
|
2024-03-11 13:19:50 -04:00
|
|
|
private static Models.Metadata.Rom ProcessItem(DatItems.Formats.Rom item, Models.Metadata.Machine machine)
|
2024-03-11 11:07:21 -04:00
|
|
|
{
|
|
|
|
|
var romItem = item.GetInternalClone();
|
|
|
|
|
|
2024-03-12 01:02:36 -04:00
|
|
|
// Create a Dump for every Rom that has a subtype
|
2025-05-11 22:55:38 -04:00
|
|
|
switch (romItem.ReadString(Models.Metadata.Rom.OpenMSXMediaType).AsOpenMSXSubType())
|
2024-03-12 01:02:36 -04:00
|
|
|
{
|
|
|
|
|
case OpenMSXSubType.Rom:
|
|
|
|
|
var dumpRom = new Models.Metadata.Dump();
|
|
|
|
|
var rom = new Models.Metadata.Rom();
|
|
|
|
|
|
|
|
|
|
rom[Models.Metadata.Rom.NameKey] = romItem.ReadString(Models.Metadata.Rom.NameKey);
|
|
|
|
|
rom[Models.Metadata.Rom.OffsetKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
rom[Models.Metadata.Rom.OpenMSXType] = romItem.ReadString(Models.Metadata.Rom.OpenMSXType);
|
|
|
|
|
rom[Models.Metadata.Rom.RemarkKey] = romItem.ReadString(Models.Metadata.Rom.RemarkKey);
|
|
|
|
|
rom[Models.Metadata.Rom.SHA1Key] = romItem.ReadString(Models.Metadata.Rom.SHA1Key);
|
|
|
|
|
rom[Models.Metadata.Rom.StartKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
|
|
|
|
|
dumpRom[Models.Metadata.Dump.RomKey] = rom;
|
|
|
|
|
|
|
|
|
|
var romOriginal = romItem.Read<DatItems.Formats.Original>("ORIGINAL");
|
|
|
|
|
if (romOriginal != null)
|
|
|
|
|
{
|
|
|
|
|
var newOriginal = new Models.Metadata.Original
|
|
|
|
|
{
|
|
|
|
|
[Models.Metadata.Original.ValueKey] = romOriginal.Value.FromYesNo(),
|
|
|
|
|
[Models.Metadata.Original.ContentKey] = romOriginal.Content,
|
|
|
|
|
};
|
|
|
|
|
dumpRom[Models.Metadata.Dump.OriginalKey] = newOriginal;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Dump?>(machine, Models.Metadata.Machine.DumpKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DumpKey, dumpRom);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OpenMSXSubType.MegaRom:
|
|
|
|
|
var dumpMegaRom = new Models.Metadata.Dump();
|
|
|
|
|
var megaRom = new Models.Metadata.Rom();
|
|
|
|
|
|
|
|
|
|
megaRom[Models.Metadata.Rom.NameKey] = romItem.ReadString(Models.Metadata.Rom.NameKey);
|
|
|
|
|
megaRom[Models.Metadata.Rom.OffsetKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
megaRom[Models.Metadata.Rom.OpenMSXType] = romItem.ReadString(Models.Metadata.Rom.OpenMSXType);
|
|
|
|
|
megaRom[Models.Metadata.Rom.RemarkKey] = romItem.ReadString(Models.Metadata.Rom.RemarkKey);
|
|
|
|
|
megaRom[Models.Metadata.Rom.SHA1Key] = romItem.ReadString(Models.Metadata.Rom.SHA1Key);
|
|
|
|
|
megaRom[Models.Metadata.Rom.StartKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
|
|
|
|
|
dumpMegaRom[Models.Metadata.Dump.MegaRomKey] = megaRom;
|
|
|
|
|
|
|
|
|
|
var megaRomOriginal = romItem.Read<DatItems.Formats.Original>("ORIGINAL");
|
|
|
|
|
if (megaRomOriginal != null)
|
|
|
|
|
{
|
|
|
|
|
var newOriginal = new Models.Metadata.Original
|
|
|
|
|
{
|
|
|
|
|
[Models.Metadata.Original.ValueKey] = megaRomOriginal.Value.FromYesNo(),
|
|
|
|
|
[Models.Metadata.Original.ContentKey] = megaRomOriginal.Content,
|
|
|
|
|
};
|
|
|
|
|
dumpMegaRom[Models.Metadata.Dump.OriginalKey] = newOriginal;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Dump?>(machine, Models.Metadata.Machine.DumpKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DumpKey, dumpMegaRom);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OpenMSXSubType.SCCPlusCart:
|
|
|
|
|
var dumpSccPlusCart = new Models.Metadata.Dump();
|
|
|
|
|
var sccPlusCart = new Models.Metadata.Rom();
|
|
|
|
|
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.NameKey] = romItem.ReadString(Models.Metadata.Rom.NameKey);
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.OffsetKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.OpenMSXType] = romItem.ReadString(Models.Metadata.Rom.OpenMSXType);
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.RemarkKey] = romItem.ReadString(Models.Metadata.Rom.RemarkKey);
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.SHA1Key] = romItem.ReadString(Models.Metadata.Rom.SHA1Key);
|
|
|
|
|
sccPlusCart[Models.Metadata.Rom.StartKey] = romItem.ReadString(Models.Metadata.Rom.StartKey) ?? romItem.ReadString(Models.Metadata.Rom.OffsetKey);
|
|
|
|
|
|
|
|
|
|
dumpSccPlusCart[Models.Metadata.Dump.RomKey] = sccPlusCart;
|
|
|
|
|
|
|
|
|
|
var sccPlusCartOriginal = romItem.Read<DatItems.Formats.Original>("ORIGINAL");
|
|
|
|
|
if (sccPlusCartOriginal != null)
|
|
|
|
|
{
|
|
|
|
|
var newOriginal = new Models.Metadata.Original
|
|
|
|
|
{
|
|
|
|
|
[Models.Metadata.Original.ValueKey] = sccPlusCartOriginal.Value.FromYesNo(),
|
|
|
|
|
[Models.Metadata.Original.ContentKey] = sccPlusCartOriginal.Content,
|
|
|
|
|
};
|
|
|
|
|
dumpSccPlusCart[Models.Metadata.Dump.OriginalKey] = newOriginal;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EnsureMachineKey<Models.Metadata.Dump?>(machine, Models.Metadata.Machine.DumpKey);
|
|
|
|
|
AppendToMachineKey(machine, Models.Metadata.Machine.DumpKey, dumpSccPlusCart);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-11 11:07:21 -04:00
|
|
|
return romItem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Ensure a key in a machine
|
|
|
|
|
/// </summary>
|
|
|
|
|
private static void EnsureMachineKey<T>(Models.Metadata.Machine machine, string key)
|
|
|
|
|
{
|
2024-03-11 14:31:02 -04:00
|
|
|
if (machine.Read<T[]?>(key) == null)
|
2024-03-19 14:35:43 -04:00
|
|
|
#if NET20 || NET35 || NET40 || NET452
|
2024-03-11 14:31:02 -04:00
|
|
|
machine[key] = new T[0];
|
2024-03-19 14:35:43 -04:00
|
|
|
#else
|
2024-03-20 11:45:36 -04:00
|
|
|
machine[key] = System.Array.Empty<T>();
|
2024-03-19 14:35:43 -04:00
|
|
|
#endif
|
2024-03-11 11:07:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Append to a machine key as if its an array
|
|
|
|
|
/// </summary>
|
2024-03-12 16:17:05 -04:00
|
|
|
private static void AppendToMachineKey<T>(Models.Metadata.Machine machine, string key, T value) where T : Models.Metadata.DatItem
|
2024-03-11 11:07:21 -04:00
|
|
|
{
|
2024-03-12 16:17:05 -04:00
|
|
|
// Get the existing array
|
2024-03-11 11:07:21 -04:00
|
|
|
var arr = machine.Read<T[]>(key);
|
|
|
|
|
if (arr == null)
|
|
|
|
|
return;
|
|
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
// Trim all null fields
|
|
|
|
|
ClearEmptyKeys(value);
|
|
|
|
|
|
|
|
|
|
// Add to the array
|
2024-03-11 11:07:21 -04:00
|
|
|
List<T> list = [.. arr];
|
|
|
|
|
list.Add(value);
|
|
|
|
|
machine[key] = list.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 16:17:05 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Clear empty keys from a DictionaryBase object
|
|
|
|
|
/// </summary>
|
|
|
|
|
private static void ClearEmptyKeys(Models.Metadata.DictionaryBase obj)
|
|
|
|
|
{
|
2024-10-19 21:41:08 -04:00
|
|
|
string[] fieldNames = [.. obj.Keys];
|
2024-03-12 16:17:05 -04:00
|
|
|
foreach (string fieldName in fieldNames)
|
|
|
|
|
{
|
|
|
|
|
if (obj[fieldName] == null)
|
|
|
|
|
obj.Remove(fieldName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-17 15:46:42 -04:00
|
|
|
#endregion
|
2024-03-11 01:39:55 -04:00
|
|
|
}
|
|
|
|
|
}
|