Add RigidMultiPredicateSort to replace some .ThenBy() boilerplate

As documented, the class "Sorts using a single primary predicate, with subsorts
using the remaining predicates in order." So only the most recent
column-header-click is taken into account. I've got a WIP class in #if false for
providing the "remember which column headers I clicked and in which order"
behaviour, but it doesn't look like that behaviour actually exists in the
codebase?
This commit is contained in:
YoshiRulz 2020-07-07 17:48:57 +10:00
parent 2dc28ecc4c
commit 33d8f4a62c
No known key found for this signature in database
GPG Key ID: C4DE31C245353FB7
4 changed files with 169 additions and 77 deletions

View File

@ -6,7 +6,7 @@ using System.IO;
using System.Linq;
using System.Text;
using BizHawk.Common.CollectionExtensions;
using BizHawk.Common;
using BizHawk.Emulation.Common;
namespace BizHawk.Client.Common
@ -414,51 +414,22 @@ namespace BizHawk.Client.Common
return true;
}
public void Sort(string column, bool reverse)
{
_cheatList = column switch
private static readonly RigidMultiPredicateSort<Cheat> ColumnSorts
= new RigidMultiPredicateSort<Cheat>(new Dictionary<string, Func<Cheat, IComparable>>
{
NameColumn => _cheatList.OrderBy(c => c.Name, reverse)
.ThenBy(c => c.Address ?? 0)
.ToList(),
AddressColumn => _cheatList.OrderBy(c => c.Address ?? 0, reverse)
.ThenBy(c => c.Name)
.ToList(),
ValueColumn => _cheatList.OrderBy(c => c.Value ?? 0, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
CompareColumn => _cheatList.OrderBy(c => c.Compare ?? 0, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
OnColumn => _cheatList.OrderBy(c => c.Enabled, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
DomainColumn => _cheatList.OrderBy(c => c.Domain, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
SizeColumn => _cheatList.OrderBy(c => (int) c.Size, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
EndianColumn => _cheatList.OrderBy(c => c.BigEndian, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
TypeColumn => _cheatList.OrderBy(c => c.Type, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
ComparisonType => _cheatList.OrderBy(c => c.ComparisonType, reverse)
.ThenBy(c => c.Name)
.ThenBy(c => c.Address ?? 0)
.ToList(),
_ => _cheatList
};
}
[NameColumn] = c => c.Name,
[AddressColumn] = c => c.Address ?? 0L,
[ValueColumn] = c => c.Value ?? 0,
[CompareColumn] = c => c.Compare ?? 0,
[OnColumn] = c => c.Enabled,
[DomainColumn] = c => c.Domain.Name,
[SizeColumn] = c => (int) c.Size,
[EndianColumn] = c => c.BigEndian,
[TypeColumn] = c => c.Type,
[ComparisonType] = c => c.ComparisonType
});
public void Sort(string column, bool reverse) => _cheatList = ColumnSorts.AppliedTo(_cheatList, column, firstIsDesc: reverse);
public void SetDefaultFileName(string defaultFileName)
{

View File

@ -342,41 +342,19 @@ namespace BizHawk.Client.EmuHawk
Close();
}
private static readonly RigidMultiPredicateSort<IMovie> ColumnSorts
= new RigidMultiPredicateSort<IMovie>(new Dictionary<string, Func<IMovie, IComparable>>
{
["File"] = x => Path.GetFileName(x.Filename),
["SysID"] = x => x.SystemID,
["Game"] = x => x.GameName,
["Length (est.)"] = x => x.FrameCount
});
private void MovieView_ColumnClick(object sender, ColumnClickEventArgs e)
{
var columnName = MovieView.Columns[e.Column].Text;
switch (columnName)
{
case "File":
default:
_movieList = _movieList.OrderBy(x => Path.GetFileName(x.Filename))
.ThenBy(x => x.SystemID)
.ThenBy(x => x.GameName)
.ThenBy(x => x.FrameCount)
.ToList();
break;
case "SysID":
_movieList = _movieList.OrderBy(x => x.SystemID)
.ThenBy(x => Path.GetFileName(x.Filename))
.ThenBy(x => x.GameName)
.ThenBy(x => x.FrameCount)
.ToList();
break;
case "Game":
_movieList = _movieList.OrderBy(x => x.GameName)
.ThenBy(x => Path.GetFileName(x.Filename))
.ThenBy(x => x.SystemID)
.ThenBy(x => x.FrameCount)
.ToList();
break;
case "Length (est.)":
_movieList = _movieList.OrderBy(x => x.FrameCount)
.ThenBy(x => Path.GetFileName(x.Filename))
.ThenBy(x => x.SystemID)
.ThenBy(x => x.GameName)
.ToList();
break;
}
_movieList = ColumnSorts.AppliedTo(_movieList, columnName);
if (_sortedCol == columnName && _sortReverse)
{
_movieList.Reverse();

View File

@ -0,0 +1,84 @@
using System;
using System.Collections.Generic;
using System.Linq;
namespace BizHawk.Common
{
#if false
/// <summary>Sorts using a reorderable list of predicates.</summary>
/// <seealso cref="RigidMultiPredicateSort{T}"/>
public sealed class MultiPredicateSort<T>
{
private readonly int _count;
/// <remarks>TODO would an array be faster?</remarks>
private readonly List<(string ID, bool IsDesc)> _order;
private readonly IReadOnlyDictionary<string, Func<T, IComparable>> _predicates;
public MultiPredicateSort(IReadOnlyDictionary<string, Func<T, IComparable>> predicates)
{
_count = predicates.Count;
if (_count == 0) throw new ArgumentException("must have at least 1 predicate", nameof(predicates));
_order = predicates.Select(kvp => (kvp.Key, false)).ToList();
_predicates = predicates;
}
public List<T> AppliedTo(IReadOnlyCollection<T> list)
{
var temp = _order[0].IsDesc
? list.OrderByDescending(_predicates[_order[0].ID])
: list.OrderBy(_predicates[_order[0].ID]);
for (var i = 1; i < _count; i++)
{
temp = _order[i].IsDesc
? temp.ThenByDescending(_predicates[_order[i].ID])
: temp.ThenBy(_predicates[_order[i].ID]);
}
return temp.ToList();
}
}
#endif
/// <summary>Sorts using a single primary predicate, with subsorts using the remaining predicates in order.</summary>
#if false
/// <seealso cref="MultiPredicateSort{T}"/>
#endif
public sealed class RigidMultiPredicateSort<T>
{
private readonly IReadOnlyDictionary<string, Func<T, IComparable>> _predicates;
public RigidMultiPredicateSort(IReadOnlyDictionary<string, Func<T, IComparable>> predicates)
{
if (predicates.Count == 0) throw new ArgumentException("must have at least 1 predicate", nameof(predicates));
_predicates = predicates;
}
/// <remarks>sorts using <paramref name="idOfFirst"/> asc/desc (by <paramref name="firstIsDesc"/>), then by the remaining predicates, all asc</remarks>
public List<T> AppliedTo(IReadOnlyCollection<T> list, string idOfFirst, bool firstIsDesc = false)
{
var temp = firstIsDesc
? list.OrderByDescending(_predicates[idOfFirst])
: list.OrderBy(_predicates[idOfFirst]);
foreach (var kvp in _predicates)
{
if (kvp.Key == idOfFirst) continue;
temp = temp.ThenBy(kvp.Value);
}
return temp.ToList();
}
public List<T> AppliedTo(IReadOnlyCollection<T> list, string idOfFirst, IReadOnlyDictionary<string, bool> isDescMap)
{
var temp = isDescMap[idOfFirst]
? list.OrderByDescending(_predicates[idOfFirst])
: list.OrderBy(_predicates[idOfFirst]);
foreach (var kvp in _predicates)
{
if (kvp.Key == idOfFirst) continue;
temp = isDescMap[kvp.Key] ? temp.ThenByDescending(kvp.Value) : temp.ThenBy(kvp.Value);
}
return temp.ToList();
}
}
}

View File

@ -0,0 +1,59 @@
using System;
using System.Collections.Generic;
using System.Linq;
using Microsoft.VisualStudio.TestTools.UnitTesting;
namespace BizHawk.Common.Tests.Common.MultiPredicateSort
{
[TestClass]
public class MultiPredicateSortTests
{
private static readonly (int X, string Y)[] Unsorted = { (1, "b"), (2, "a"), (2, "b"), (1, "a") };
private static readonly (int X, string Y)[] SortedByXThenYDesc = { (1, "b"), (1, "a"), (2, "b"), (2, "a") };
private static readonly (int X, string Y)[] SortedByYDescThenXDesc = { (2, "b"), (1, "b"), (2, "a"), (1, "a") };
private static void AssertSequenceEqual<T>(IEnumerable<T> expected, IEnumerable<T> actual) => Assert.IsTrue(expected.SequenceEqual(actual));
[TestMethod]
public void SanityCheck()
{
AssertSequenceEqual(
SortedByYDescThenXDesc,
Unsorted.OrderByDescending(t => t.Y).ThenByDescending(t => t.X)
);
AssertSequenceEqual(
SortedByYDescThenXDesc,
Unsorted.OrderByDescending(t => t.X).OrderByDescending(t => t.Y)
);
}
[TestMethod]
public void TestRigidSort()
{
var sorts = new RigidMultiPredicateSort<(int X, string Y)>(new Dictionary<string, Func<(int X, string Y), IComparable>>
{
["by_x"] = t => t.X,
["by_y"] = t => t.Y
});
AssertSequenceEqual(
SortedByXThenYDesc,
sorts.AppliedTo(
Unsorted,
"by_x",
new Dictionary<string, bool> { ["by_x"] = false, ["by_y"] = true }
)
);
AssertSequenceEqual(
SortedByYDescThenXDesc,
sorts.AppliedTo(
Unsorted,
"by_y",
new Dictionary<string, bool> { ["by_x"] = true, ["by_y"] = true }
)
);
}
}
}