2020-03-23 10:07:09 +08:00
|
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
2019-01-24 16:43:03 +08:00
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-03-12 21:13:43 +08:00
|
|
|
|
using System;
|
2020-03-23 06:49:45 +08:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Reflection;
|
2018-11-28 12:12:29 +08:00
|
|
|
|
using Newtonsoft.Json;
|
2020-03-23 06:49:45 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
2021-04-16 14:10:53 +08:00
|
|
|
|
using osu.Framework.Extensions.TypeExtensions;
|
2019-03-27 18:29:27 +08:00
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
2022-08-11 03:51:11 +08:00
|
|
|
|
using osu.Framework.Localisation;
|
2020-03-23 06:49:45 +08:00
|
|
|
|
using osu.Game.Configuration;
|
2023-02-22 02:05:10 +08:00
|
|
|
|
using osu.Game.Extensions;
|
2020-03-19 11:43:26 +08:00
|
|
|
|
using osu.Game.Rulesets.UI;
|
2021-04-13 01:10:22 +08:00
|
|
|
|
using osu.Game.Utils;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-04-18 15:05:58 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Mods
|
2017-02-17 04:05:03 +08:00
|
|
|
|
{
|
2017-02-23 19:59:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The base class for gameplay modifiers.
|
|
|
|
|
/// </summary>
|
2021-08-31 13:38:35 +08:00
|
|
|
|
public abstract class Mod : IMod, IEquatable<Mod>, IDeepCloneable<Mod>
|
2017-02-17 04:05:03 +08:00
|
|
|
|
{
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2017-03-06 17:28:30 +08:00
|
|
|
|
public abstract string Name { get; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-11-30 16:16:00 +08:00
|
|
|
|
public abstract string Acronym { get; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2023-09-27 17:54:47 +08:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
public virtual string ExtendedIconInformation => string.Empty;
|
|
|
|
|
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2020-01-14 21:22:00 +08:00
|
|
|
|
public virtual IconUsage? Icon => null;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2018-07-31 17:00:42 +08:00
|
|
|
|
public virtual ModType Type => ModType.Fun;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2022-08-11 03:51:11 +08:00
|
|
|
|
public abstract LocalisableString Description { get; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-03-19 11:43:26 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The tooltip to display for this mod when used in a <see cref="ModIcon"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Differs from <see cref="Name"/>, as the value of attributes (AR, CS, etc) changeable via the mod
|
|
|
|
|
/// are displayed in the tooltip.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
[JsonIgnore]
|
2020-03-21 04:34:36 +08:00
|
|
|
|
public string IconTooltip
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-03-23 14:20:56 +08:00
|
|
|
|
string description = SettingDescription;
|
|
|
|
|
|
|
|
|
|
return string.IsNullOrEmpty(description) ? Name : $"{Name} ({description})";
|
2020-03-21 04:34:36 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The description of editable settings of a mod to use in the <see cref="IconTooltip"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Parentheses are added to the tooltip, surrounding the value of this property. If this property is <c>string.Empty</c>,
|
|
|
|
|
/// the tooltip will not have parentheses.
|
|
|
|
|
/// </remarks>
|
2020-03-23 06:49:45 +08:00
|
|
|
|
public virtual string SettingDescription
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
var tooltipTexts = new List<string>();
|
|
|
|
|
|
|
|
|
|
foreach ((SettingSourceAttribute attr, PropertyInfo property) in this.GetOrderedSettingsSourceProperties())
|
|
|
|
|
{
|
2022-12-16 17:16:26 +08:00
|
|
|
|
var bindable = (IBindable)property.GetValue(this)!;
|
2020-03-23 14:20:56 +08:00
|
|
|
|
|
2023-08-23 19:21:43 +08:00
|
|
|
|
string valueText;
|
|
|
|
|
|
|
|
|
|
switch (bindable)
|
|
|
|
|
{
|
|
|
|
|
case Bindable<bool> b:
|
|
|
|
|
valueText = b.Value ? "on" : "off";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
valueText = bindable.ToString() ?? string.Empty;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-03 14:47:15 +08:00
|
|
|
|
if (!bindable.IsDefault)
|
2023-08-23 19:21:43 +08:00
|
|
|
|
tooltipTexts.Add($"{attr.Label}: {valueText}");
|
2020-03-23 06:49:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-23 14:20:56 +08:00
|
|
|
|
return string.Join(", ", tooltipTexts.Where(s => !string.IsNullOrEmpty(s)));
|
2020-03-23 06:49:45 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-03-19 11:43:26 +08:00
|
|
|
|
|
2017-02-23 19:57:58 +08:00
|
|
|
|
/// <summary>
|
2022-02-17 12:26:12 +08:00
|
|
|
|
/// The score multiplier of this mod.
|
2017-02-23 19:57:58 +08:00
|
|
|
|
/// </summary>
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2022-02-17 12:26:12 +08:00
|
|
|
|
public abstract double ScoreMultiplier { get; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-12-31 03:09:51 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns true if this mod is implemented (and playable).
|
|
|
|
|
/// </summary>
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2017-12-31 03:09:51 +08:00
|
|
|
|
public virtual bool HasImplementation => this is IApplicableMod;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-03-17 08:40:15 +08:00
|
|
|
|
[JsonIgnore]
|
2022-03-18 07:08:11 +08:00
|
|
|
|
public virtual bool UserPlayable => true;
|
2022-03-17 08:40:15 +08:00
|
|
|
|
|
2022-05-05 19:37:38 +08:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
public virtual bool ValidForMultiplayer => true;
|
|
|
|
|
|
|
|
|
|
[JsonIgnore]
|
|
|
|
|
public virtual bool ValidForMultiplayerAsFreeMod => true;
|
|
|
|
|
|
2020-01-21 00:06:36 +08:00
|
|
|
|
/// <summary>
|
2020-01-21 11:44:22 +08:00
|
|
|
|
/// Whether this mod requires configuration to apply changes to the game.
|
2020-01-21 00:06:36 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
[JsonIgnore]
|
|
|
|
|
public virtual bool RequiresConfiguration => false;
|
|
|
|
|
|
2017-02-23 19:57:58 +08:00
|
|
|
|
/// <summary>
|
2017-03-02 08:57:33 +08:00
|
|
|
|
/// The mods this mod cannot be enabled with.
|
2017-02-23 19:57:58 +08:00
|
|
|
|
/// </summary>
|
2018-11-28 12:12:29 +08:00
|
|
|
|
[JsonIgnore]
|
2019-11-28 21:41:29 +08:00
|
|
|
|
public virtual Type[] IncompatibleMods => Array.Empty<Type>();
|
2019-03-14 22:39:45 +08:00
|
|
|
|
|
2023-04-26 03:05:40 +08:00
|
|
|
|
private IReadOnlyDictionary<string, IBindable>? settingsBacking;
|
2021-08-18 14:17:42 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
2023-04-26 03:05:40 +08:00
|
|
|
|
/// All <see cref="IBindable"/> settings within this mod.
|
2021-08-18 14:17:42 +08:00
|
|
|
|
/// </summary>
|
2023-05-05 00:26:45 +08:00
|
|
|
|
/// <remarks>
|
2023-05-05 00:30:41 +08:00
|
|
|
|
/// The settings are returned in ascending key order as per <see cref="SettingsMap"/>.
|
2023-05-05 00:26:45 +08:00
|
|
|
|
/// The ordering is intentionally enforced manually, as ordering of <see cref="Dictionary{TKey,TValue}.Values"/> is unspecified.
|
|
|
|
|
/// </remarks>
|
2023-05-05 00:30:41 +08:00
|
|
|
|
internal IEnumerable<IBindable> SettingsBindables => SettingsMap.OrderBy(pair => pair.Key).Select(pair => pair.Value);
|
2023-04-26 03:05:40 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Provides mapping of names to <see cref="IBindable"/>s of all settings within this mod.
|
|
|
|
|
/// </summary>
|
2023-05-05 00:30:41 +08:00
|
|
|
|
internal IReadOnlyDictionary<string, IBindable> SettingsMap =>
|
2021-08-18 14:17:42 +08:00
|
|
|
|
settingsBacking ??= this.GetSettingsSourceProperties()
|
2023-04-26 03:05:40 +08:00
|
|
|
|
.Select(p => p.Item2)
|
|
|
|
|
.ToDictionary(property => property.Name.ToSnakeCase(), property => (IBindable)property.GetValue(this)!);
|
2021-08-18 14:17:42 +08:00
|
|
|
|
|
2022-07-18 12:21:10 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether all settings in this mod are set to their default state.
|
|
|
|
|
/// </summary>
|
2023-04-26 03:05:40 +08:00
|
|
|
|
protected virtual bool UsesDefaultConfiguration => SettingsBindables.All(s => s.IsDefault);
|
2022-07-18 12:21:10 +08:00
|
|
|
|
|
2019-03-14 22:39:45 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a copy of this <see cref="Mod"/> initialised to a default state.
|
|
|
|
|
/// </summary>
|
2021-07-19 11:38:22 +08:00
|
|
|
|
public virtual Mod DeepClone()
|
2020-08-13 18:48:31 +08:00
|
|
|
|
{
|
2022-12-16 17:16:26 +08:00
|
|
|
|
var result = (Mod)Activator.CreateInstance(GetType())!;
|
2021-01-01 08:46:09 +08:00
|
|
|
|
result.CopyFrom(this);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2023-02-22 02:05:10 +08:00
|
|
|
|
/// <summary>
|
2023-02-22 03:59:36 +08:00
|
|
|
|
/// Copies mod setting values from <paramref name="source"/> into this instance, overwriting all existing settings.
|
2023-02-22 02:05:10 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="source">The mod to copy properties from.</param>
|
2023-02-22 03:59:36 +08:00
|
|
|
|
public void CopyFrom(Mod source)
|
2023-02-22 02:05:10 +08:00
|
|
|
|
{
|
2023-02-22 03:59:36 +08:00
|
|
|
|
if (source.GetType() != GetType())
|
|
|
|
|
throw new ArgumentException($"Expected mod of type {GetType()}, got {source.GetType()}.", nameof(source));
|
2023-02-22 02:05:10 +08:00
|
|
|
|
|
|
|
|
|
foreach (var (_, property) in this.GetSettingsSourceProperties())
|
|
|
|
|
{
|
|
|
|
|
var targetBindable = (IBindable)property.GetValue(this)!;
|
2023-02-22 03:59:36 +08:00
|
|
|
|
var sourceBindable = (IBindable)property.GetValue(source)!;
|
2023-02-22 02:05:10 +08:00
|
|
|
|
|
2023-02-22 03:59:36 +08:00
|
|
|
|
CopyAdjustedSetting(targetBindable, sourceBindable);
|
2023-02-22 02:05:10 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-01 08:46:09 +08:00
|
|
|
|
/// <summary>
|
2023-05-05 00:33:49 +08:00
|
|
|
|
/// This method copies the values of all settings from <paramref name="source"/> that share the same names with this mod instance.
|
|
|
|
|
/// The most frequent use of this is when switching rulesets, in order to preserve values of common settings during the switch.
|
2021-01-01 08:46:09 +08:00
|
|
|
|
/// </summary>
|
2023-05-05 00:33:49 +08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// The values are copied directly, without adjusting for possibly different allowed ranges of values.
|
|
|
|
|
/// If the value of a setting is not valid for this instance due to not falling inside of the allowed range, it will be clamped accordingly.
|
|
|
|
|
/// </remarks>
|
2023-04-26 03:05:40 +08:00
|
|
|
|
/// <param name="source">The mod to extract settings from.</param>
|
2023-05-05 00:15:12 +08:00
|
|
|
|
public void CopyCommonSettingsFrom(Mod source)
|
2021-01-01 08:46:09 +08:00
|
|
|
|
{
|
2023-04-26 03:05:40 +08:00
|
|
|
|
if (source.UsesDefaultConfiguration)
|
|
|
|
|
return;
|
2023-02-22 03:59:36 +08:00
|
|
|
|
|
2023-05-05 00:30:41 +08:00
|
|
|
|
foreach (var (name, targetSetting) in SettingsMap)
|
2023-02-22 03:59:36 +08:00
|
|
|
|
{
|
2023-05-05 00:30:41 +08:00
|
|
|
|
if (!source.SettingsMap.TryGetValue(name, out IBindable? sourceSetting))
|
2023-02-22 03:59:36 +08:00
|
|
|
|
continue;
|
|
|
|
|
|
2023-04-26 03:05:40 +08:00
|
|
|
|
if (sourceSetting.IsDefault)
|
2023-02-24 22:11:22 +08:00
|
|
|
|
continue;
|
2023-03-12 06:29:36 +08:00
|
|
|
|
|
2023-05-05 01:06:07 +08:00
|
|
|
|
var targetBindableType = targetSetting.GetType();
|
|
|
|
|
var sourceBindableType = sourceSetting.GetType();
|
2023-04-26 16:46:29 +08:00
|
|
|
|
|
2023-05-05 01:06:07 +08:00
|
|
|
|
// if either the target is assignable to the source or the source is assignable to the target,
|
|
|
|
|
// then we presume that the data types contained in both bindables are compatible and we can proceed with the copy.
|
|
|
|
|
// this handles cases like `Bindable<int>` and `BindableInt`.
|
|
|
|
|
if (!targetBindableType.IsAssignableFrom(sourceBindableType) && !sourceBindableType.IsAssignableFrom(targetBindableType))
|
2023-03-12 06:29:36 +08:00
|
|
|
|
continue;
|
|
|
|
|
|
2023-04-26 03:05:40 +08:00
|
|
|
|
// TODO: special case for handling number types
|
2023-02-24 22:11:22 +08:00
|
|
|
|
|
2023-04-26 03:05:40 +08:00
|
|
|
|
PropertyInfo property = targetSetting.GetType().GetProperty(nameof(Bindable<bool>.Value))!;
|
|
|
|
|
property.SetValue(targetSetting, property.GetValue(sourceSetting));
|
2020-08-13 18:48:31 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-03 14:48:28 +08:00
|
|
|
|
/// <summary>
|
2021-01-03 20:30:21 +08:00
|
|
|
|
/// When creating copies or clones of a Mod, this method will be called
|
|
|
|
|
/// to copy explicitly adjusted user settings from <paramref name="target"/>.
|
|
|
|
|
/// The base implementation will transfer the value via <see cref="Bindable{T}.Parse"/>
|
|
|
|
|
/// or by binding and unbinding (if <paramref name="source"/> is an <see cref="IBindable"/>)
|
|
|
|
|
/// and should be called unless replaced with custom logic.
|
2021-01-03 14:48:28 +08:00
|
|
|
|
/// </summary>
|
2021-01-03 20:30:21 +08:00
|
|
|
|
/// <param name="target">The target bindable to apply the adjustment to.</param>
|
2021-01-03 20:25:44 +08:00
|
|
|
|
/// <param name="source">The adjustment to apply.</param>
|
|
|
|
|
internal virtual void CopyAdjustedSetting(IBindable target, object source)
|
2021-01-03 19:44:57 +08:00
|
|
|
|
{
|
2021-01-03 20:25:44 +08:00
|
|
|
|
if (source is IBindable sourceBindable)
|
2021-01-03 19:44:57 +08:00
|
|
|
|
{
|
2021-01-03 20:30:21 +08:00
|
|
|
|
// copy including transfer of default values.
|
2021-01-03 20:25:44 +08:00
|
|
|
|
target.BindTo(sourceBindable);
|
|
|
|
|
target.UnbindFrom(sourceBindable);
|
2020-08-13 18:48:31 +08:00
|
|
|
|
}
|
2021-01-03 19:44:57 +08:00
|
|
|
|
else
|
2021-04-16 14:10:53 +08:00
|
|
|
|
{
|
|
|
|
|
if (!(target is IParseable parseable))
|
2021-04-16 14:40:06 +08:00
|
|
|
|
throw new InvalidOperationException($"Bindable type {target.GetType().ReadableName()} is not {nameof(IParseable)}.");
|
2021-04-16 14:10:53 +08:00
|
|
|
|
|
|
|
|
|
parseable.Parse(source);
|
|
|
|
|
}
|
2020-08-13 18:48:31 +08:00
|
|
|
|
}
|
2019-05-03 09:05:45 +08:00
|
|
|
|
|
2022-12-16 17:16:26 +08:00
|
|
|
|
public bool Equals(IMod? other) => other is Mod them && Equals(them);
|
2021-04-13 01:10:22 +08:00
|
|
|
|
|
2022-12-16 17:16:26 +08:00
|
|
|
|
public bool Equals(Mod? other)
|
2021-04-13 01:10:22 +08:00
|
|
|
|
{
|
|
|
|
|
if (ReferenceEquals(null, other)) return false;
|
|
|
|
|
if (ReferenceEquals(this, other)) return true;
|
|
|
|
|
|
|
|
|
|
return GetType() == other.GetType() &&
|
2023-04-26 03:05:40 +08:00
|
|
|
|
SettingsBindables.SequenceEqual(other.SettingsBindables, ModSettingsEqualityComparer.Default);
|
2021-04-13 01:10:22 +08:00
|
|
|
|
}
|
2021-02-09 12:44:42 +08:00
|
|
|
|
|
2021-08-17 09:27:04 +08:00
|
|
|
|
public override int GetHashCode()
|
|
|
|
|
{
|
|
|
|
|
var hashCode = new HashCode();
|
|
|
|
|
|
|
|
|
|
hashCode.Add(GetType());
|
|
|
|
|
|
2023-04-26 03:05:40 +08:00
|
|
|
|
foreach (var setting in SettingsBindables)
|
2022-03-15 13:53:50 +08:00
|
|
|
|
hashCode.Add(setting.GetUnderlyingSettingValue());
|
2021-08-17 09:27:04 +08:00
|
|
|
|
|
|
|
|
|
return hashCode.ToHashCode();
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-09 12:44:42 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Reset all custom settings for this mod back to their defaults.
|
|
|
|
|
/// </summary>
|
2022-12-16 17:16:26 +08:00
|
|
|
|
public virtual void ResetSettingsToDefaults() => CopyFrom((Mod)Activator.CreateInstance(GetType())!);
|
2021-08-18 14:17:42 +08:00
|
|
|
|
|
|
|
|
|
private class ModSettingsEqualityComparer : IEqualityComparer<IBindable>
|
|
|
|
|
{
|
|
|
|
|
public static ModSettingsEqualityComparer Default { get; } = new ModSettingsEqualityComparer();
|
|
|
|
|
|
2022-12-16 17:16:26 +08:00
|
|
|
|
public bool Equals(IBindable? x, IBindable? y)
|
2021-08-18 14:17:42 +08:00
|
|
|
|
{
|
2022-12-16 17:16:26 +08:00
|
|
|
|
object? xValue = x?.GetUnderlyingSettingValue();
|
|
|
|
|
object? yValue = y?.GetUnderlyingSettingValue();
|
2021-08-18 14:17:42 +08:00
|
|
|
|
|
|
|
|
|
return EqualityComparer<object>.Default.Equals(xValue, yValue);
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-15 13:53:50 +08:00
|
|
|
|
public int GetHashCode(IBindable obj) => obj.GetUnderlyingSettingValue().GetHashCode();
|
2021-08-18 14:17:42 +08:00
|
|
|
|
}
|
2017-02-18 19:44:46 +08:00
|
|
|
|
}
|
2017-03-02 08:57:33 +08:00
|
|
|
|
}
|