Convert OfflineList writer to serializer, fix bugs

This commit is contained in:
Matt Nadareski
2024-03-12 00:09:48 -04:00
parent fa26420982
commit 1b9c3a4dc9
4 changed files with 34 additions and 357 deletions

View File

@@ -136,6 +136,17 @@ namespace SabreTools.DatFiles
Default = infos.ReleaseNumber.Default.AsYesNo(), Default = infos.ReleaseNumber.Default.AsYesNo(),
}); });
} }
// TODO: Uncomment this when the change to both Models and Serialization is done
//if (infos.ImageNumber != null)
//{
// offlineListInfos.Add(new Formats.OfflineListInfo
// {
// Name = "imageNumber",
// Visible = infos.ImageNumber.Visible.AsYesNo(),
// InNamingOption = infos.ImageNumber.InNamingOption.AsYesNo(),
// Default = infos.ImageNumber.Default.AsYesNo(),
// });
//}
if (infos.LanguageNumber != null) if (infos.LanguageNumber != null)
{ {
offlineListInfos.Add(new Formats.OfflineListInfo offlineListInfos.Add(new Formats.OfflineListInfo
@@ -206,13 +217,15 @@ namespace SabreTools.DatFiles
if (newDat != null) if (newDat != null)
{ {
Header.SetFieldValue<string?>("DATVERSIONURL", newDat.DatVersionUrl); Header.SetFieldValue<string?>("DATVERSIONURL", newDat.DatVersionUrl);
//Header.SetFieldValue<Models.OfflineList.DatUrl?>("DATURL", newDat.DatUrl); // TODO: Add to internal model Header.SetFieldValue<Models.OfflineList.DatUrl?>("DATURL", newDat.DatUrl);
Header.SetFieldValue<string?>("IMURL", newDat.ImUrl); Header.SetFieldValue<string?>("IMURL", newDat.ImUrl);
} }
} }
if (item.ContainsKey(Models.Metadata.Header.SearchKey)) if (item.ContainsKey(Models.Metadata.Header.SearchKey))
{ {
// TODO: Add to internal model var search = item.Read<Models.OfflineList.Search>(Models.Metadata.Header.SearchKey);
if (search != null)
Header.SetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey, search);
} }
// Selectively set all possible fields -- TODO: Figure out how to make this less manual // Selectively set all possible fields -- TODO: Figure out how to make this less manual

View File

@@ -55,7 +55,7 @@ namespace SabreTools.DatFiles
// Convert subheader values // Convert subheader values
if (Header.CanOpenSpecified) if (Header.CanOpenSpecified)
header[Models.Metadata.Header.CanOpenKey] = Header.GetFieldValue<Models.OfflineList.CanOpen[]?>(Models.Metadata.Header.CanOpenKey); header[Models.Metadata.Header.CanOpenKey] = new Models.OfflineList.CanOpen { Extension = Header.GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey) };
// if (Header.ImagesSpecified) // if (Header.ImagesSpecified)
// // TODO: Add to internal model // // TODO: Add to internal model
if (Header.InfosSpecified) if (Header.InfosSpecified)
@@ -180,13 +180,13 @@ namespace SabreTools.DatFiles
var newDat = new Models.OfflineList.NewDat var newDat = new Models.OfflineList.NewDat
{ {
DatVersionUrl = Header.GetStringFieldValue("DATVERSIONURL"), DatVersionUrl = Header.GetStringFieldValue("DATVERSIONURL"),
//DatUrl = Header.GetFieldValue<Models.OfflineList.DatUrl?>("DATURL"), // TODO: Add to internal model DatUrl = Header.GetFieldValue<Models.OfflineList.DatUrl?>("DATURL"),
ImUrl = Header.GetStringFieldValue("IMURL"), ImUrl = Header.GetStringFieldValue("IMURL"),
}; };
header[Models.Metadata.Header.NewDatKey] = newDat; header[Models.Metadata.Header.NewDatKey] = newDat;
} }
// if (Header.SearchSpecified) if (Header.SearchSpecified)
// // TODO: Add to internal model header[Models.Metadata.Header.SearchKey] = Header.GetFieldValue<Models.OfflineList.Search>(Models.Metadata.Header.SearchKey);
return header; return header;
} }

View File

@@ -7,7 +7,6 @@ using SabreTools.Core;
using SabreTools.Core.Tools; using SabreTools.Core.Tools;
using SabreTools.DatFiles.Formats; using SabreTools.DatFiles.Formats;
using SabreTools.Filter; using SabreTools.Filter;
using SabreTools.Serialization;
namespace SabreTools.DatFiles namespace SabreTools.DatFiles
{ {
@@ -98,7 +97,7 @@ namespace SabreTools.DatFiles
{ {
get get
{ {
var canOpen = GetFieldValue<Models.OfflineList.CanOpen[]?>(Models.Metadata.Header.CanOpenKey); var canOpen = GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey);
return canOpen != null && canOpen.Length > 0; return canOpen != null && canOpen.Length > 0;
} }
} }
@@ -109,11 +108,20 @@ namespace SabreTools.DatFiles
get get
{ {
return GetStringFieldValue("DATVERSIONURL") != null return GetStringFieldValue("DATVERSIONURL") != null
//&& GetFieldValue<Models.OfflineList.DatUrl?>("DATURL") != null // TODO: Add to internal model && GetFieldValue<Models.OfflineList.DatUrl?>("DATURL") != null
&& GetStringFieldValue("IMURL") != null; && GetStringFieldValue("IMURL") != null;
} }
} }
[JsonIgnore]
public bool SearchSpecified
{
get
{
return GetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey) != null;
}
}
/// <summary> /// <summary>
/// Internal Header model /// Internal Header model
/// </summary> /// </summary>
@@ -312,6 +320,7 @@ namespace SabreTools.DatFiles
header.SetFieldValue<string?>(Models.Metadata.Header.SampleModeKey, GetStringFieldValue(Models.Metadata.Header.SampleModeKey).AsEnumValue<MergingFlag>().AsStringValue()); header.SetFieldValue<string?>(Models.Metadata.Header.SampleModeKey, GetStringFieldValue(Models.Metadata.Header.SampleModeKey).AsEnumValue<MergingFlag>().AsStringValue());
header.SetFieldValue<string?>(Models.Metadata.Header.ScreenshotsHeightKey, GetStringFieldValue(Models.Metadata.Header.ScreenshotsHeightKey)); header.SetFieldValue<string?>(Models.Metadata.Header.ScreenshotsHeightKey, GetStringFieldValue(Models.Metadata.Header.ScreenshotsHeightKey));
header.SetFieldValue<string?>(Models.Metadata.Header.ScreenshotsWidthKey, GetStringFieldValue(Models.Metadata.Header.ScreenshotsWidthKey)); header.SetFieldValue<string?>(Models.Metadata.Header.ScreenshotsWidthKey, GetStringFieldValue(Models.Metadata.Header.ScreenshotsWidthKey));
header.SetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey, GetFieldValue<Models.OfflineList.Search?>(Models.Metadata.Header.SearchKey)); // TODO: Perform a deep clone
header.SetFieldValue<string?>(Models.Metadata.Header.SystemKey, GetStringFieldValue(Models.Metadata.Header.SystemKey)); header.SetFieldValue<string?>(Models.Metadata.Header.SystemKey, GetStringFieldValue(Models.Metadata.Header.SystemKey));
header.SetFieldValue<string?>(Models.Metadata.Header.TypeKey, GetStringFieldValue(Models.Metadata.Header.TypeKey)); header.SetFieldValue<string?>(Models.Metadata.Header.TypeKey, GetStringFieldValue(Models.Metadata.Header.TypeKey));
header.SetFieldValue<string?>(Models.Metadata.Header.UrlKey, GetStringFieldValue(Models.Metadata.Header.UrlKey)); header.SetFieldValue<string?>(Models.Metadata.Header.UrlKey, GetStringFieldValue(Models.Metadata.Header.UrlKey));

View File

@@ -1,8 +1,6 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq;
using SabreTools.Core; using SabreTools.Core;
using SabreTools.Core.Tools;
using SabreTools.DatItems; using SabreTools.DatItems;
using SabreTools.DatItems.Formats; using SabreTools.DatItems.Formats;
@@ -27,10 +25,6 @@ namespace SabreTools.DatFiles.Formats
{ {
var missingFields = new List<string>(); var missingFields = new List<string>();
// Check item name
if (string.IsNullOrEmpty(datItem.GetName()))
missingFields.Add(Models.Metadata.Rom.NameKey);
switch (datItem) switch (datItem)
{ {
case Rom rom: case Rom rom:
@@ -51,7 +45,9 @@ namespace SabreTools.DatFiles.Formats
{ {
logger.User($"Writing to '{outfile}'..."); logger.User($"Writing to '{outfile}'...");
var datafile = CreateDat(ignoreblanks); // Serialize the input file
var metadata = ConvertMetadata(ignoreblanks);
var datafile = new Serialization.CrossModel.OfflineList().Deserialize(metadata);
if (!(new Serialization.Files.OfflineList().Serialize(datafile, outfile))) if (!(new Serialization.Files.OfflineList().Serialize(datafile, outfile)))
{ {
logger.Warning($"File '{outfile}' could not be written! See the log for more details."); logger.Warning($"File '{outfile}' could not be written! See the log for more details.");
@@ -67,346 +63,5 @@ namespace SabreTools.DatFiles.Formats
logger.User($"'{outfile}' written!{Environment.NewLine}"); logger.User($"'{outfile}' written!{Environment.NewLine}");
return true; return true;
} }
#region Converters
/// <summary>
/// Create a Dat from the current internal information
/// <summary>
/// <param name="ignoreblanks">True if blank roms should be skipped on output, false otherwise</param>
private Models.OfflineList.Dat CreateDat(bool ignoreblanks)
{
var dat = new Models.OfflineList.Dat
{
NoNamespaceSchemaLocation = "datas.xsd",
Configuration = CreateConfiguration(),
Games = CreateGames(ignoreblanks),
GUI = CreateGUI(),
};
return dat;
}
/// <summary>
/// Create a Configuration from the current internal information
/// <summary>
private Models.OfflineList.Configuration? CreateConfiguration()
{
// If we don't have a header, we can't do anything
if (this.Header == null)
return null;
var configuration = new Models.OfflineList.Configuration
{
DatName = Header.GetStringFieldValue(Models.Metadata.Header.NameKey),
ImFolder = Header.GetStringFieldValue(Models.Metadata.Header.ImFolderKey),
DatVersion = Header.GetStringFieldValue(Models.Metadata.Header.DatVersionKey),
System = Header.GetStringFieldValue(Models.Metadata.Header.SystemKey),
ScreenshotsWidth = Header.GetStringFieldValue(Models.Metadata.Header.ScreenshotsWidthKey),
ScreenshotsHeight = Header.GetStringFieldValue(Models.Metadata.Header.ScreenshotsHeightKey),
Infos = CreateInfos(),
CanOpen = CreateCanOpen(),
NewDat = CreateNewDat(),
Search = CreateSearch(),
RomTitle = Header.GetStringFieldValue(Models.Metadata.Header.RomTitleKey),
};
return configuration;
}
/// <summary>
/// Create a Infos from the current internal information
/// <summary>
private Models.OfflineList.Infos? CreateInfos()
{
// If we don't have infos, we can't do anything
if (!Header.InfosSpecified)
return null;
var infos = new Models.OfflineList.Infos();
foreach (var info in Header.GetFieldValue<OfflineListInfo[]?>(Models.Metadata.Header.InfosKey)!)
{
switch (info.Name)
{
case "title":
infos.Title = new Models.OfflineList.Title
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "location":
infos.Location = new Models.OfflineList.Location
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "publisher":
infos.Publisher = new Models.OfflineList.Publisher
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "sourceRom":
infos.SourceRom = new Models.OfflineList.SourceRom
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "saveType":
infos.SaveType = new Models.OfflineList.SaveType
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "romSize":
infos.RomSize = new Models.OfflineList.RomSize
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "releaseNumber":
infos.ReleaseNumber = new Models.OfflineList.ReleaseNumber
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "languageNumber":
infos.LanguageNumber = new Models.OfflineList.LanguageNumber
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "comment":
infos.Comment = new Models.OfflineList.Comment
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "romCRC":
infos.RomCRC = new Models.OfflineList.RomCRC
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "im1CRC":
infos.Im1CRC = new Models.OfflineList.Im1CRC
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "im2CRC":
infos.Im2CRC = new Models.OfflineList.Im2CRC
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
case "languages":
infos.Languages = new Models.OfflineList.Languages
{
Visible = info.Visible?.ToString(),
InNamingOption = info.InNamingOption?.ToString(),
Default = info.Default?.ToString(),
};
break;
}
}
return infos;
}
/// <summary>
/// Create a CanOpen from the current internal information
/// <summary>
private Models.OfflineList.CanOpen? CreateCanOpen()
{
// If we don't have a canopen, we can't do anything
if (!Header.CanOpenSpecified || Header.GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey) == null)
return null;
var canOpen = new Models.OfflineList.CanOpen
{
Extension = [.. Header.GetStringArrayFieldValue(Models.Metadata.Header.CanOpenKey)],
};
return canOpen;
}
/// <summary>
/// Create a NewDat from the current internal information
/// <summary>
private Models.OfflineList.NewDat? CreateNewDat()
{
// If we don't have a Header, we can't do anything
if (Header == null)
return null;
var newDat = new Models.OfflineList.NewDat
{
DatVersionUrl = Header.GetStringFieldValue("DATVERSIONURL"),
//DatUrl = Header.GetFieldValue<Models.OfflineList.DatUrl?>("DATURL"); // TODO: Add to internal model
ImUrl = Header.GetStringFieldValue("IMURL"),
};
return newDat;
}
/// <summary>
/// Create a Search from the current internal information
/// <summary>
private Models.OfflineList.Search? CreateSearch()
{
// If we don't have a Header, we can't do anything
if (Header == null)
return null;
// TODO: Add to internal model
return null;
}
/// <summary>
/// Create a Games from the current internal information
/// <summary>
/// <param name="ignoreblanks">True if blank roms should be skipped on output, false otherwise</param>
private Models.OfflineList.Games? CreateGames(bool ignoreblanks)
{
// If we don't have items, we can't do anything
if (this.Items == null || !this.Items.Any())
return null;
// Create a list of hold the games
var games = new List<Models.OfflineList.Game>();
// Loop through the sorted items and create games for them
foreach (string key in Items.SortedKeys)
{
var items = Items.FilteredItems(key);
if (items == null || !items.Any())
continue;
// Get the first item for game information
var machine = items[0].GetFieldValue<Machine>(DatItem.MachineKey);
var game = OfflineList.CreateGame(machine!);
// Create holders for all item types
var romCRCs = new List<Models.OfflineList.FileRomCRC>();
// 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 Rom rom:
romCRCs.Add(CreateRomCRC(rom));
break;
}
}
// Assign the values to the game
game.Files = new Models.OfflineList.Files { RomCRC = [.. romCRCs] };
// Add the game to the list
games.Add(game);
}
return new Models.OfflineList.Games { Game = [.. games] };
}
/// <summary>
/// Create a Machine from the current internal information
/// <summary>
private static Models.OfflineList.Game CreateGame(Machine machine)
{
var game = new Models.OfflineList.Game
{
ImageNumber = machine.GetStringFieldValue(Models.Metadata.Machine.ImageNumberKey),
ReleaseNumber = machine.GetStringFieldValue(Models.Metadata.Machine.ReleaseNumberKey),
Title = machine.GetStringFieldValue(Models.Metadata.Machine.NameKey),
SaveType = machine.GetStringFieldValue(Models.Metadata.Machine.SaveTypeKey),
Publisher = machine.GetStringFieldValue(Models.Metadata.Machine.PublisherKey),
Location = machine.GetStringFieldValue(Models.Metadata.Machine.LocationKey),
SourceRom = machine.GetStringFieldValue(Models.Metadata.Machine.SourceRomKey),
Language = machine.GetStringFieldValue(Models.Metadata.Machine.LanguageKey),
Im1CRC = machine.GetStringFieldValue(Models.Metadata.Machine.Im1CRCKey),
Im2CRC = machine.GetStringFieldValue(Models.Metadata.Machine.Im2CRCKey),
Comment = machine.GetStringFieldValue(Models.Metadata.Machine.CommentKey),
DuplicateID = machine.GetStringFieldValue(Models.Metadata.Machine.CloneOfKey),
};
return game;
}
/// <summary>
/// Create a RomCRC from the current Rom DatItem
/// <summary>
private static Models.OfflineList.FileRomCRC CreateRomCRC(Rom item)
{
var romCRC = new Models.OfflineList.FileRomCRC
{
Content = item.GetStringFieldValue(Models.Metadata.Rom.CRCKey),
};
return romCRC;
}
/// <summary>
/// Create a GUI from the current internal information
/// <summary>
private Models.OfflineList.GUI? CreateGUI()
{
// If we don't have a header, we can't do anything
if (this.Header == null)
return null;
// TODO: Add to internal model
return null;
}
#endregion
} }
} }