2019-01-24 08:43:03 +00:00
|
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using osu.Framework.Graphics;
|
2019-03-27 10:29:27 +00:00
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Framework.Input.Bindings;
|
2019-09-04 11:28:21 +00:00
|
|
|
|
using osu.Framework.IO.Stores;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Overlays.Settings;
|
|
|
|
|
using osu.Game.Rulesets.Edit;
|
|
|
|
|
using osu.Game.Rulesets.Mods;
|
|
|
|
|
using osu.Game.Rulesets.Replays.Types;
|
|
|
|
|
using osu.Game.Rulesets.UI;
|
2018-04-13 13:41:35 +00:00
|
|
|
|
using osu.Game.Beatmaps.Legacy;
|
2018-06-11 04:17:08 +00:00
|
|
|
|
using osu.Game.Configuration;
|
|
|
|
|
using osu.Game.Rulesets.Configuration;
|
2018-05-15 08:38:04 +00:00
|
|
|
|
using osu.Game.Rulesets.Difficulty;
|
2019-12-17 11:08:13 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2018-11-28 07:12:57 +00:00
|
|
|
|
using osu.Game.Scoring;
|
2019-08-26 03:21:49 +00:00
|
|
|
|
using osu.Game.Skinning;
|
2020-01-03 11:39:15 +00:00
|
|
|
|
using osu.Game.Users;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets
|
|
|
|
|
{
|
|
|
|
|
public abstract class Ruleset
|
|
|
|
|
{
|
2019-12-24 04:48:27 +00:00
|
|
|
|
public RulesetInfo RulesetInfo { get; internal set; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
public IEnumerable<Mod> GetAllMods() => Enum.GetValues(typeof(ModType)).Cast<ModType>()
|
|
|
|
|
// Confine all mods of each mod type into a single IEnumerable<Mod>
|
|
|
|
|
.SelectMany(GetModsFor)
|
|
|
|
|
// Filter out all null mods
|
|
|
|
|
.Where(mod => mod != null)
|
|
|
|
|
// Resolve MultiMods as their .Mods property
|
|
|
|
|
.SelectMany(mod => (mod as MultiMod)?.Mods ?? new[] { mod });
|
|
|
|
|
|
|
|
|
|
public abstract IEnumerable<Mod> GetModsFor(ModType type);
|
|
|
|
|
|
2018-04-16 12:14:40 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Converts mods from legacy enum values. Do not override if you're not a legacy ruleset.
|
|
|
|
|
/// </summary>
|
2020-03-24 03:06:24 +00:00
|
|
|
|
/// <param name="mods">The legacy enum which will be converted.</param>
|
|
|
|
|
/// <returns>An enumerable of constructed <see cref="Mod"/>s.</returns>
|
|
|
|
|
public virtual IEnumerable<Mod> ConvertFromLegacyMods(LegacyMods mods) => Array.Empty<Mod>();
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Converts mods to legacy enum values. Do not override if you're not a legacy ruleset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="mods">The mods which will be converted.</param>
|
|
|
|
|
/// <returns>A single bitwise enumerable value representing (to the best of our ability) the mods.</returns>
|
|
|
|
|
public virtual LegacyMods ConvertToLegacyMods(Mod[] mods)
|
|
|
|
|
{
|
|
|
|
|
var value = LegacyMods.None;
|
|
|
|
|
|
|
|
|
|
foreach (var mod in mods)
|
|
|
|
|
{
|
|
|
|
|
switch (mod)
|
|
|
|
|
{
|
|
|
|
|
case ModNoFail _:
|
|
|
|
|
value |= LegacyMods.NoFail;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModEasy _:
|
|
|
|
|
value |= LegacyMods.Easy;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModHidden _:
|
|
|
|
|
value |= LegacyMods.Hidden;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModHardRock _:
|
|
|
|
|
value |= LegacyMods.HardRock;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModSuddenDeath _:
|
|
|
|
|
value |= LegacyMods.SuddenDeath;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModDoubleTime _:
|
|
|
|
|
value |= LegacyMods.DoubleTime;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModRelax _:
|
|
|
|
|
value |= LegacyMods.Relax;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModHalfTime _:
|
|
|
|
|
value |= LegacyMods.HalfTime;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ModFlashlight _:
|
|
|
|
|
value |= LegacyMods.Flashlight;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
}
|
2018-04-13 13:41:35 +00:00
|
|
|
|
|
2019-03-07 09:30:31 +00:00
|
|
|
|
public ModAutoplay GetAutoplayMod() => GetAllMods().OfType<ModAutoplay>().First();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-04-02 09:39:49 +00:00
|
|
|
|
public virtual ISkin CreateLegacySkinProvider(ISkinSource source, IBeatmap beatmap) => null;
|
2019-08-26 03:21:49 +00:00
|
|
|
|
|
2019-12-18 05:49:09 +00:00
|
|
|
|
protected Ruleset()
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2019-12-24 07:02:35 +00:00
|
|
|
|
RulesetInfo = new RulesetInfo
|
|
|
|
|
{
|
|
|
|
|
Name = Description,
|
|
|
|
|
ShortName = ShortName,
|
|
|
|
|
ID = (this as ILegacyRuleset)?.LegacyID,
|
|
|
|
|
InstantiationInfo = GetType().AssemblyQualifiedName,
|
2020-01-03 11:39:15 +00:00
|
|
|
|
Available = true,
|
2019-12-24 07:02:35 +00:00
|
|
|
|
};
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Attempt to create a hit renderer for a beatmap
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="beatmap">The beatmap to create the hit renderer for.</param>
|
2019-04-25 08:36:17 +00:00
|
|
|
|
/// <param name="mods">The <see cref="Mod"/>s to apply.</param>
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <exception cref="BeatmapInvalidForRulesetException">Unable to successfully load the beatmap to be usable with this ruleset.</exception>
|
|
|
|
|
/// <returns></returns>
|
2019-12-12 06:58:11 +00:00
|
|
|
|
public abstract DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod> mods = null);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-12-17 11:08:13 +00:00
|
|
|
|
/// <summary>
|
2019-12-24 08:01:17 +00:00
|
|
|
|
/// Creates a <see cref="ScoreProcessor"/> for this <see cref="Ruleset"/>.
|
2019-12-17 11:08:13 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>The score processor.</returns>
|
2019-12-24 08:01:17 +00:00
|
|
|
|
public virtual ScoreProcessor CreateScoreProcessor() => new ScoreProcessor();
|
2019-12-17 11:08:13 +00:00
|
|
|
|
|
2019-12-19 11:03:14 +00:00
|
|
|
|
/// <summary>
|
2019-12-24 08:01:17 +00:00
|
|
|
|
/// Creates a <see cref="HealthProcessor"/> for this <see cref="Ruleset"/>.
|
2019-12-19 11:03:14 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>The health processor.</returns>
|
2019-12-27 07:14:49 +00:00
|
|
|
|
public virtual HealthProcessor CreateHealthProcessor(double drainStartTime) => new DrainingHealthProcessor(drainStartTime);
|
2019-12-19 11:03:14 +00:00
|
|
|
|
|
2018-06-29 04:07:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a <see cref="IBeatmapConverter"/> to convert a <see cref="IBeatmap"/> to one that is applicable for this <see cref="Ruleset"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="beatmap">The <see cref="IBeatmap"/> to be converted.</param>
|
|
|
|
|
/// <returns>The <see cref="IBeatmapConverter"/>.</returns>
|
2018-04-19 13:04:12 +00:00
|
|
|
|
public abstract IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap);
|
|
|
|
|
|
2018-06-29 04:07:00 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Optionally creates a <see cref="IBeatmapProcessor"/> to alter a <see cref="IBeatmap"/> after it has been converted.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="beatmap">The <see cref="IBeatmap"/> to be processed.</param>
|
|
|
|
|
/// <returns>The <see cref="IBeatmapProcessor"/>.</returns>
|
2018-04-19 13:04:12 +00:00
|
|
|
|
public virtual IBeatmapProcessor CreateBeatmapProcessor(IBeatmap beatmap) => null;
|
|
|
|
|
|
2019-02-21 04:12:37 +00:00
|
|
|
|
public abstract DifficultyCalculator CreateDifficultyCalculator(WorkingBeatmap beatmap);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-11-30 06:18:52 +00:00
|
|
|
|
public virtual PerformanceCalculator CreatePerformanceCalculator(WorkingBeatmap beatmap, ScoreInfo score) => null;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
public virtual HitObjectComposer CreateHitObjectComposer() => null;
|
|
|
|
|
|
2019-04-02 10:55:24 +00:00
|
|
|
|
public virtual Drawable CreateIcon() => new SpriteIcon { Icon = FontAwesome.Solid.QuestionCircle };
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-12-28 13:13:18 +00:00
|
|
|
|
public virtual IResourceStore<byte[]> CreateResourceStore() => new NamespacedResourceStore<byte[]>(new DllResourceStore(GetType().Assembly), @"Resources");
|
2019-09-04 11:28:21 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
public abstract string Description { get; }
|
|
|
|
|
|
2018-06-11 04:28:50 +00:00
|
|
|
|
public virtual RulesetSettingsSubsection CreateSettings() => null;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-06-11 04:17:08 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates the <see cref="IRulesetConfigManager"/> for this <see cref="Ruleset"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="settings">The <see cref="SettingsStore"/> to store the settings.</param>
|
|
|
|
|
public virtual IRulesetConfigManager CreateConfig(SettingsStore settings) => null;
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// A unique short name to reference this ruleset in online requests.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public abstract string ShortName { get; }
|
|
|
|
|
|
2020-01-03 11:39:15 +00:00
|
|
|
|
/// <summary>
|
2020-01-03 13:00:57 +00:00
|
|
|
|
/// The playing verb to be shown in the <see cref="UserActivity.SoloGame.Status"/>.
|
2020-01-03 11:39:15 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual string PlayingVerb => "Playing solo";
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// A list of available variant ids.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual IEnumerable<int> AvailableVariants => new[] { 0 };
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Get a list of default keys for the specified variant.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="variant">A variant.</param>
|
|
|
|
|
/// <returns>A list of valid <see cref="KeyBinding"/>s.</returns>
|
2019-11-28 13:41:29 +00:00
|
|
|
|
public virtual IEnumerable<KeyBinding> GetDefaultKeyBindings(int variant = 0) => Array.Empty<KeyBinding>();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the name for a key binding variant. This is used for display in the settings overlay.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="variant">The variant.</param>
|
|
|
|
|
/// <returns>A descriptive name of the variant.</returns>
|
|
|
|
|
public virtual string GetVariantName(int variant) => string.Empty;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// For rulesets which support legacy (osu-stable) replay conversion, this method will create an empty replay frame
|
|
|
|
|
/// for conversion use.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>An empty frame for the current ruleset, or null if unsupported.</returns>
|
|
|
|
|
public virtual IConvertibleReplayFrame CreateConvertibleReplayFrame() => null;
|
|
|
|
|
}
|
|
|
|
|
}
|