1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-07 23:03:21 +08:00

Improved readability and sounds

This commit is contained in:
Terochi 2023-04-25 21:05:40 +02:00
parent 8e8dda3ac0
commit 0841e73a39
3 changed files with 69 additions and 90 deletions

View File

@ -43,18 +43,18 @@ namespace osu.Game.Tests.Mods
var modDouble = new TestNonMatchingSettingTypeModDouble { TestSetting = { Value = setting_change } }; var modDouble = new TestNonMatchingSettingTypeModDouble { TestSetting = { Value = setting_change } };
var modBool = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = false, Value = true } }; var modBool = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = false, Value = true } };
var modInt = new TestNonMatchingSettingTypeModInt { TestSetting = { Value = (int)setting_change } }; var modInt = new TestNonMatchingSettingTypeModInt { TestSetting = { Value = (int)setting_change / 2 } };
modDouble.CopySharedSettings(modBool); modDouble.CopyCommonSettings(modBool);
modDouble.CopySharedSettings(modInt); modDouble.CopyCommonSettings(modInt);
modBool.CopySharedSettings(modDouble); modBool.CopyCommonSettings(modDouble);
modBool.CopySharedSettings(modInt); modBool.CopyCommonSettings(modInt);
modInt.CopySharedSettings(modDouble); modInt.CopyCommonSettings(modDouble);
modInt.CopySharedSettings(modBool); modInt.CopyCommonSettings(modBool);
Assert.That(modDouble.TestSetting.Value, Is.EqualTo(setting_change)); Assert.That(modDouble.TestSetting.Value, Is.EqualTo(setting_change));
Assert.That(modBool.TestSetting.Value, Is.EqualTo(!modBool.TestSetting.Default)); Assert.That(modBool.TestSetting.Value, Is.EqualTo(!modBool.TestSetting.Default));
Assert.That(modInt.TestSetting.Value, Is.EqualTo((int)setting_change)); Assert.That(modInt.TestSetting.Value, Is.EqualTo((int)setting_change / 2));
} }
[Test] [Test]
@ -63,23 +63,12 @@ namespace osu.Game.Tests.Mods
var modBoolTrue = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = true, Value = false } }; var modBoolTrue = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = true, Value = false } };
var modBoolFalse = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = false, Value = true } }; var modBoolFalse = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = false, Value = true } };
modBoolFalse.CopySharedSettings(modBoolTrue); modBoolFalse.CopyCommonSettings(modBoolTrue);
Assert.That(modBoolFalse.TestSetting.Default, Is.EqualTo(false)); Assert.That(modBoolFalse.TestSetting.Default, Is.EqualTo(false));
Assert.That(modBoolFalse.TestSetting.Value, Is.EqualTo(modBoolTrue.TestSetting.Value)); Assert.That(modBoolFalse.TestSetting.Value, Is.EqualTo(modBoolTrue.TestSetting.Value));
} }
[Test]
public void TestValueResetsToDefaultWhenCopied()
{
var modDouble = new TestNonMatchingSettingTypeModDouble();
var modBool = new TestNonMatchingSettingTypeModBool { TestSetting = { Default = false, Value = true } };
modBool.CopySharedSettings(modDouble);
Assert.That(modBool.TestSetting.Value, Is.EqualTo(modBool.TestSetting.Default));
}
private class TestNonMatchingSettingTypeModDouble : TestNonMatchingSettingTypeMod private class TestNonMatchingSettingTypeModDouble : TestNonMatchingSettingTypeMod
{ {
public override string Acronym => "NMD"; public override string Acronym => "NMD";

View File

@ -58,7 +58,6 @@ using osu.Game.Rulesets.Mods;
using osu.Game.Scoring; using osu.Game.Scoring;
using osu.Game.Skinning; using osu.Game.Skinning;
using osu.Game.Utils; using osu.Game.Utils;
using File = System.IO.File;
using RuntimeInfo = osu.Framework.RuntimeInfo; using RuntimeInfo = osu.Framework.RuntimeInfo;
namespace osu.Game namespace osu.Game
@ -71,7 +70,7 @@ namespace osu.Game
[Cached(typeof(OsuGameBase))] [Cached(typeof(OsuGameBase))]
public partial class OsuGameBase : Framework.Game, ICanAcceptFiles, IBeatSyncProvider public partial class OsuGameBase : Framework.Game, ICanAcceptFiles, IBeatSyncProvider
{ {
public static readonly string[] VIDEO_EXTENSIONS = { ".mp4", ".mov", ".avi", ".flv" }; public static readonly string[] VIDEO_EXTENSIONS = { ".mp4", ".mov", ".avi", ".flv", ".mpg", ".wmv", ".m4v" };
public const string OSU_PROTOCOL = "osu://"; public const string OSU_PROTOCOL = "osu://";
@ -626,20 +625,30 @@ namespace osu.Game
return; return;
} }
var previouslySelectedCommonMods = new List<Mod>(SelectedMods.Value.Count);
var convertedCommonMods = new List<Mod>(SelectedMods.Value.Count);
foreach (var mod in SelectedMods.Value)
{
var convertedMod = instance.CreateModFromAcronym(mod.Acronym);
if (convertedMod == null)
continue;
previouslySelectedCommonMods.Add(mod);
convertedMod.CopyCommonSettings(mod);
convertedCommonMods.Add(convertedMod);
}
if (!SelectedMods.Disabled)
// Select common mods to play the deselect samples for other mods
SelectedMods.Value = previouslySelectedCommonMods;
AvailableMods.Value = dict; AvailableMods.Value = dict;
if (!SelectedMods.Disabled) if (!SelectedMods.Disabled)
{ SelectedMods.Value = convertedCommonMods;
//converting mods from one ruleset to the other, while also keeping their shared settings unchanged
SelectedMods.Value = SelectedMods.Value.Select(oldMod =>
{
Mod newMod = instance.CreateModFromAcronym(oldMod.Acronym);
newMod?.CopySharedSettings(oldMod);
return newMod;
}).Where(m => m != null).ToArray();
}
void revertRulesetChange() => Ruleset.Value = r.OldValue?.Available == true ? r.OldValue : RulesetStore.AvailableRulesets.First(); void revertRulesetChange() => Ruleset.Value = r.OldValue?.Available == true ? r.OldValue : RulesetStore.AvailableRulesets.First();
} }

View File

@ -115,21 +115,25 @@ namespace osu.Game.Rulesets.Mods
[JsonIgnore] [JsonIgnore]
public virtual Type[] IncompatibleMods => Array.Empty<Type>(); public virtual Type[] IncompatibleMods => Array.Empty<Type>();
private IReadOnlyList<IBindable>? settingsBacking; private IReadOnlyDictionary<string, IBindable>? settingsBacking;
/// <summary> /// <summary>
/// A list of the all <see cref="IBindable"/> settings within this mod. /// All <see cref="IBindable"/> settings within this mod.
/// </summary> /// </summary>
internal IReadOnlyList<IBindable> Settings => internal IEnumerable<IBindable> SettingsBindables => Settings.Values;
/// <summary>
/// Provides mapping of names to <see cref="IBindable"/>s of all settings within this mod.
/// </summary>
internal IReadOnlyDictionary<string, IBindable> Settings =>
settingsBacking ??= this.GetSettingsSourceProperties() settingsBacking ??= this.GetSettingsSourceProperties()
.Select(p => p.Item2.GetValue(this)) .Select(p => p.Item2)
.Cast<IBindable>() .ToDictionary(property => property.Name.ToSnakeCase(), property => (IBindable)property.GetValue(this)!);
.ToList();
/// <summary> /// <summary>
/// Whether all settings in this mod are set to their default state. /// Whether all settings in this mod are set to their default state.
/// </summary> /// </summary>
protected virtual bool UsesDefaultConfiguration => Settings.All(s => s.IsDefault); protected virtual bool UsesDefaultConfiguration => SettingsBindables.All(s => s.IsDefault);
/// <summary> /// <summary>
/// Creates a copy of this <see cref="Mod"/> initialised to a default state. /// Creates a copy of this <see cref="Mod"/> initialised to a default state.
@ -160,60 +164,37 @@ namespace osu.Game.Rulesets.Mods
} }
/// <summary> /// <summary>
/// Copies all mod setting values sharing same <see cref="MemberInfo.Name"/> from <paramref name="source"/> into this instance. /// When converting mods from one ruleset to the other, this method makes sure
/// to also copy the values of all settings sharing same <see cref="MemberInfo.Name"/> between the two instances.
/// </summary> /// </summary>
/// <param name="source">The mod to copy properties from.</param> /// <remarks>Copied values are unchanged, even if they have different clamping ranges.</remarks>
internal void CopySharedSettings(Mod source) /// <param name="source">The mod to extract settings from.</param>
public void CopyCommonSettings(Mod source)
{ {
const string value = nameof(Bindable<int>.Value); if (source.UsesDefaultConfiguration)
return;
Dictionary<string, object> sourceSettings = new Dictionary<string, object>(); foreach (var (name, targetSetting) in Settings)
foreach (var (_, sourceProperty) in source.GetSettingsSourceProperties())
{ {
sourceSettings.Add(sourceProperty.Name.ToSnakeCase(), sourceProperty.GetValue(source)!); if (!source.Settings.TryGetValue(name, out IBindable? sourceSetting))
}
foreach (var (_, targetProperty) in this.GetSettingsSourceProperties())
{
object targetSetting = targetProperty.GetValue(this)!;
if (!sourceSettings.TryGetValue(targetProperty.Name.ToSnakeCase(), out object? sourceSetting))
continue; continue;
if (((IBindable)sourceSetting).IsDefault) if (sourceSetting.IsDefault)
{
// reset to default value if the source is default
targetSetting.GetType().GetMethod(nameof(Bindable<int>.SetDefault))!.Invoke(targetSetting, null);
continue;
}
bool hasSameGenericArgument = getGenericBaseType(targetSetting, typeof(Bindable<>))!.GenericTypeArguments.Single() ==
getGenericBaseType(sourceSetting, typeof(Bindable<>))!.GenericTypeArguments.Single();
if (!hasSameGenericArgument)
continue; continue;
Type? targetBindableNumberType = getGenericBaseType(targetSetting, typeof(BindableNumber<>)); if (getBindableGenericType(targetSetting) != getBindableGenericType(sourceSetting))
Type? sourceBindableNumberType = getGenericBaseType(sourceSetting, typeof(BindableNumber<>));
if (targetBindableNumberType == null || sourceBindableNumberType == null)
{
setValue(targetSetting, value, getValue(sourceSetting, value));
continue; continue;
// TODO: special case for handling number types
PropertyInfo property = targetSetting.GetType().GetProperty(nameof(Bindable<bool>.Value))!;
property.SetValue(targetSetting, property.GetValue(sourceSetting));
} }
setValue(targetSetting, value, Convert.ChangeType(getValue(sourceSetting, value), targetBindableNumberType.GenericTypeArguments.Single())); Type getBindableGenericType(IBindable setting) =>
} setting.GetType().GetTypeInheritance()
.First(t => t.IsGenericType && t.GetGenericTypeDefinition() == typeof(Bindable<>))
object? getValue(object setting, string name) => .GenericTypeArguments.Single();
setting.GetType().GetProperty(name)!.GetValue(setting);
void setValue(object setting, string name, object? newValue) =>
setting.GetType().GetProperty(name)!.SetValue(setting, newValue);
Type? getGenericBaseType(object setting, Type genericType) =>
setting.GetType().GetTypeInheritance().FirstOrDefault(t => t.IsGenericType && t.GetGenericTypeDefinition() == genericType);
} }
/// <summary> /// <summary>
@ -250,7 +231,7 @@ namespace osu.Game.Rulesets.Mods
if (ReferenceEquals(this, other)) return true; if (ReferenceEquals(this, other)) return true;
return GetType() == other.GetType() && return GetType() == other.GetType() &&
Settings.SequenceEqual(other.Settings, ModSettingsEqualityComparer.Default); SettingsBindables.SequenceEqual(other.SettingsBindables, ModSettingsEqualityComparer.Default);
} }
public override int GetHashCode() public override int GetHashCode()
@ -259,7 +240,7 @@ namespace osu.Game.Rulesets.Mods
hashCode.Add(GetType()); hashCode.Add(GetType());
foreach (var setting in Settings) foreach (var setting in SettingsBindables)
hashCode.Add(setting.GetUnderlyingSettingValue()); hashCode.Add(setting.GetUnderlyingSettingValue());
return hashCode.ToHashCode(); return hashCode.ToHashCode();