Move ClearEmpty to DatFile

This commit is contained in:
Matt Nadareski
2025-01-14 22:07:05 -05:00
parent 7e2d094ba5
commit 3e839e1249
5 changed files with 136 additions and 102 deletions

View File

@@ -1,5 +1,6 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.IO; using System.IO;
using SabreTools.DatFiles.Formats;
using SabreTools.DatItems; using SabreTools.DatItems;
using SabreTools.DatItems.Formats; using SabreTools.DatItems.Formats;
using SabreTools.Hashing; using SabreTools.Hashing;
@@ -64,6 +65,48 @@ namespace SabreTools.DatFiles.Test
#endregion #endregion
#region ClearEmpty
[Fact]
public void ClearEmpty_Items()
{
Source source = new Source(0, source: null);
Machine machine = new Machine();
machine.SetFieldValue<string?>(Models.Metadata.Machine.NameKey, "game-1");
DatItem datItem = new Rom();
datItem.SetFieldValue<Source?>(DatItem.SourceKey, source);
datItem.SetFieldValue<Machine?>(DatItem.MachineKey, machine);
DatFile datFile = new Logiqx(datFile: null, deprecated: false);
datFile.AddItem(datItem, statsOnly: false);
datFile.ClearEmpty();
Assert.Single(datFile.Items.SortedKeys);
}
[Fact]
public void ClearEmpty_ItemsDB()
{
Source source = new Source(0, source: null);
Machine machine = new Machine();
machine.SetFieldValue<string?>(Models.Metadata.Machine.NameKey, "game-1");
DatItem datItem = new Rom();
DatFile datFile = new Logiqx(datFile: null, deprecated: false);
long sourceIndex = datFile.AddSourceDB(source);
long machineIndex = datFile.AddMachineDB(machine);
_ = datFile.AddItemDB(datItem, machineIndex, sourceIndex, statsOnly: false);
datFile.ClearEmpty();
Assert.Single(datFile.ItemsDB.SortedKeys);
}
#endregion
#region FillHeaderFromPath #region FillHeaderFromPath
[Fact] [Fact]
@@ -188,6 +231,24 @@ namespace SabreTools.DatFiles.Test
#endregion #endregion
#region SetHeader
[Fact]
public void SetHeaderTest()
{
DatHeader datHeader = new DatHeader();
datHeader.SetFieldValue(Models.Metadata.Header.NameKey, "name");
DatFile? datFile = new Formats.Logiqx(datFile: null, deprecated: false);
datFile.Header.SetFieldValue(Models.Metadata.Header.NameKey, "notname");
datFile.SetHeader(datHeader);
Assert.NotNull(datFile.Header);
Assert.Equal("name", datFile.Header.GetStringFieldValue(Models.Metadata.Header.NameKey));
}
#endregion
#region ResetDictionary #region ResetDictionary
[Fact] [Fact]
@@ -209,24 +270,6 @@ namespace SabreTools.DatFiles.Test
#endregion #endregion
#region SetHeader
[Fact]
public void SetHeaderTest()
{
DatHeader datHeader = new DatHeader();
datHeader.SetFieldValue(Models.Metadata.Header.NameKey, "name");
DatFile? datFile = new Formats.Logiqx(datFile: null, deprecated: false);
datFile.Header.SetFieldValue(Models.Metadata.Header.NameKey, "notname");
datFile.SetHeader(datHeader);
Assert.NotNull(datFile.Header);
Assert.Equal("name", datFile.Header.GetStringFieldValue(Models.Metadata.Header.NameKey));
}
#endregion
#region ProcessItemName #region ProcessItemName
[Theory] [Theory]
@@ -1952,7 +1995,7 @@ namespace SabreTools.DatFiles.Test
[Fact] [Fact]
public void ShouldIgnore_UnsupportedType_True() public void ShouldIgnore_UnsupportedType_True()
{ {
DatItem? datItem = new SoftwareList(); DatItem? datItem = new DatItems.Formats.SoftwareList();
DatFile datFile = new Formats.Logiqx(null, deprecated: false); DatFile datFile = new Formats.Logiqx(null, deprecated: false);
bool actual = datFile.ShouldIgnore(datItem, ignoreBlanks: true); bool actual = datFile.ShouldIgnore(datItem, ignoreBlanks: true);

View File

@@ -128,6 +128,15 @@ namespace SabreTools.DatFiles
#region Accessors #region Accessors
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
public void ClearEmpty()
{
ClearEmptyImpl();
ClearEmptyImplDB();
}
/// <summary> /// <summary>
/// Set the internal header /// Set the internal header
/// </summary> /// </summary>
@@ -138,6 +147,42 @@ namespace SabreTools.DatFiles
Header = datHeader; Header = datHeader;
} }
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
private void ClearEmptyImpl()
{
foreach (string key in Items.SortedKeys)
{
// If the value is empty, remove
List<DatItem> value = GetItemsForBucket(key);
if (value.Count == 0)
RemoveBucket(key);
// If there are no non-blank items, remove
else if (value.FindIndex(i => i != null && i is not Blank) == -1)
RemoveBucket(key);
}
}
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
private void ClearEmptyImplDB()
{
foreach (string key in ItemsDB.SortedKeys)
{
// If the value is empty, remove
List<DatItem> value = [.. GetItemsForBucketDB(key).Values];
if (value.Count == 0)
RemoveBucketDB(key);
// If there are no non-blank items, remove
else if (value.FindIndex(i => i != null && i is not Blank) == -1)
RemoveBucketDB(key);
}
}
#endregion #endregion
#region Item Dictionary Passthrough - Accessors #region Item Dictionary Passthrough - Accessors
@@ -182,15 +227,6 @@ namespace SabreTools.DatFiles
return ItemsDB.AddSource(source); return ItemsDB.AddSource(source);
} }
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
public void ClearEmpty()
{
Items.ClearEmpty();
ItemsDB.ClearEmpty();
}
/// <summary> /// <summary>
/// Remove all items marked for removal /// Remove all items marked for removal
/// </summary> /// </summary>
@@ -227,6 +263,15 @@ namespace SabreTools.DatFiles
return Items.RemoveBucket(key); return Items.RemoveBucket(key);
} }
/// <summary>
/// Remove a key from the file dictionary if it exists
/// </summary>
/// <param name="key">Key in the dictionary to remove</param>
public bool RemoveBucketDB(string key)
{
return ItemsDB.RemoveBucket(key);
}
/// <summary> /// <summary>
/// Remove the first instance of a value from the file dictionary if it exists /// Remove the first instance of a value from the file dictionary if it exists
/// </summary> /// </summary>

View File

@@ -190,45 +190,10 @@ namespace SabreTools.DatFiles
return key; return key;
} }
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
internal void ClearEmpty()
{
foreach (string key in SortedKeys)
{
#if NET40_OR_GREATER || NETCOREAPP
// If the key doesn't exist, skip
if (!_items.TryGetValue(key, out var value))
continue;
// If the value is null, remove
else if (value == null)
_items.TryRemove(key, out _);
// If there are no non-blank items, remove
else if (value!.FindIndex(i => i != null && i is not Blank) == -1)
_items.TryRemove(key, out _);
#else
// If the key doesn't exist, skip
if (!_items.ContainsKey(key))
continue;
// If the value is null, remove
else if (_items[key] == null)
_items.Remove(key);
// If there are no non-blank items, remove
else if (_items[key]!.FindIndex(i => i != null && i is not Blank) == -1)
_items.Remove(key);
#endif
}
}
/// <summary> /// <summary>
/// Remove all items marked for removal /// Remove all items marked for removal
/// </summary> /// </summary>
internal void ClearMarked() public void ClearMarked()
{ {
foreach (string key in SortedKeys) foreach (string key in SortedKeys)
{ {

View File

@@ -300,36 +300,10 @@ namespace SabreTools.DatFiles
#endif #endif
} }
/// <summary>
/// Remove any keys that have null or empty values
/// </summary>
internal void ClearEmpty()
{
var keys = Array.FindAll(SortedKeys, k => k != null);
foreach (string key in keys)
{
// Get items for the bucket
var items = GetItemsForBucket(key);
if (items == null || items.Count == 0)
continue;
// Convert to list of indices for ease of access
List<DatItem> itemsList = [.. items.Values];
// If there are no non-blank items, remove
if (!itemsList.Exists(i => i != null && i is not Blank))
#if NET40_OR_GREATER || NETCOREAPP
_buckets.TryRemove(key, out _);
#else
_buckets.Remove(key);
#endif
}
}
/// <summary> /// <summary>
/// Remove all items marked for removal /// Remove all items marked for removal
/// </summary> /// </summary>
internal void ClearMarked() public void ClearMarked()
{ {
var itemIndices = _items.Keys; var itemIndices = _items.Keys;
foreach (long itemIndex in itemIndices) foreach (long itemIndex in itemIndices)
@@ -565,6 +539,24 @@ namespace SabreTools.DatFiles
} }
} }
/// <summary>
/// Remove a key from the file dictionary if it exists
/// </summary>
/// <param name="key">Key in the dictionary to remove</param>
public bool RemoveBucket(string key)
{
#if NET40_OR_GREATER || NETCOREAPP
return _buckets.TryRemove(key, out var list);
#else
if (!_buckets.ContainsKey(key))
return false;
var list = _buckets[key];
_buckets.Remove(key);
return true;
#endif
}
/// <summary> /// <summary>
/// Remove an item, returning if it could be removed /// Remove an item, returning if it could be removed
/// </summary> /// </summary>

View File

@@ -61,17 +61,6 @@ namespace SabreTools.Test.DatFiles
Assert.Equal(expected, dict.SortedKeys.Length); Assert.Equal(expected, dict.SortedKeys.Length);
} }
[Fact]
public void ClearEmptyTest()
{
// Setup the dictionary
var dict = new ItemDictionary();
dict.AddItem("game-1", new Rom());
dict.ClearEmpty();
Assert.Single(dict.SortedKeys);
}
[Fact] [Fact]
public void ClearMarkedTest() public void ClearMarkedTest()
{ {