1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-11 22:27:25 +08:00
osu-lazer/osu.Game/Rulesets/UI/RulesetContainer.cs

405 lines
16 KiB
C#
Raw Normal View History

2018-01-05 19:21:19 +08:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2016-09-02 18:25:13 +08:00
2016-11-14 16:23:33 +08:00
using osu.Framework.Allocation;
using osu.Framework.Graphics;
2016-09-02 18:25:13 +08:00
using osu.Framework.Graphics.Containers;
using osu.Game.Beatmaps;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Judgements;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Drawables;
using System;
using System.Collections.Generic;
2017-03-14 16:14:11 +08:00
using System.Diagnostics;
using System.Linq;
using osu.Framework.Configuration;
using osu.Framework.Graphics.Cursor;
using osu.Framework.Input;
using osu.Game.Configuration;
using osu.Game.Input.Handlers;
using osu.Game.Overlays;
using osu.Game.Rulesets.Configuration;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Replays;
using osu.Game.Rulesets.Scoring;
using OpenTK;
2016-09-02 18:25:13 +08:00
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.UI
2016-09-02 18:25:13 +08:00
{
/// <summary>
2017-08-09 12:28:29 +08:00
/// Base RulesetContainer. Doesn't hold objects.
2017-03-15 20:58:00 +08:00
/// <para>
2017-09-06 17:05:51 +08:00
/// Should not be derived - derive <see cref="RulesetContainer{TObject}"/> instead.
2017-03-15 20:58:00 +08:00
/// </para>
/// </summary>
2017-08-09 12:28:29 +08:00
public abstract class RulesetContainer : Container
{
/// <summary>
/// The selected variant.
/// </summary>
public virtual int Variant => 0;
/// <summary>
2017-08-09 12:28:29 +08:00
/// The input manager for this RulesetContainer.
/// </summary>
2017-08-24 19:31:57 +08:00
internal IHasReplayHandler ReplayInputManager => KeyBindingInputManager as IHasReplayHandler;
/// <summary>
2017-08-09 12:28:29 +08:00
/// The key conversion input manager for this RulesetContainer.
/// </summary>
public PassThroughInputManager KeyBindingInputManager;
/// <summary>
/// Whether a replay is currently loaded.
/// </summary>
public readonly BindableBool HasReplayLoaded = new BindableBool();
public abstract IEnumerable<HitObject> Objects { get; }
2017-12-14 19:27:51 +08:00
private readonly Lazy<Playfield> playfield;
2017-11-30 20:56:12 +08:00
/// <summary>
/// The playfield.
/// </summary>
2017-12-14 19:27:51 +08:00
public Playfield Playfield => playfield.Value;
2017-11-30 20:56:12 +08:00
/// <summary>
/// The cursor provided by this <see cref="RulesetContainer"/>. May be null if no cursor is provided.
/// </summary>
public readonly CursorContainer Cursor;
protected readonly Ruleset Ruleset;
private IRulesetConfigManager rulesetConfig;
private OnScreenDisplay onScreenDisplay;
private DependencyContainer dependencies;
protected override IReadOnlyDependencyContainer CreateLocalDependencies(IReadOnlyDependencyContainer parent)
=> dependencies = new DependencyContainer(base.CreateLocalDependencies(parent));
/// <summary>
/// A visual representation of a <see cref="Rulesets.Ruleset"/>.
/// </summary>
/// <param name="ruleset">The ruleset being repesented.</param>
2017-11-21 11:11:29 +08:00
protected RulesetContainer(Ruleset ruleset)
{
Ruleset = ruleset;
2017-12-14 19:27:51 +08:00
playfield = new Lazy<Playfield>(CreatePlayfield);
Cursor = CreateCursor();
}
2018-03-12 05:09:16 +08:00
[BackgroundDependencyLoader(true)]
private void load(OnScreenDisplay onScreenDisplay, SettingsStore settings)
{
2018-03-12 04:19:34 +08:00
this.onScreenDisplay = onScreenDisplay;
rulesetConfig = CreateConfig(Ruleset, settings);
if (rulesetConfig != null)
{
dependencies.Cache(rulesetConfig);
onScreenDisplay?.BeginTracking(this, rulesetConfig);
}
}
public abstract ScoreProcessor CreateScoreProcessor();
/// <summary>
/// Creates a key conversion input manager. An exception will be thrown if a valid <see cref="RulesetInputManager{T}"/> is not returned.
/// </summary>
/// <returns>The input manager.</returns>
public abstract PassThroughInputManager CreateInputManager();
protected virtual ReplayInputHandler CreateReplayInputHandler(Replay replay) => null;
public Replay Replay { get; private set; }
2017-03-31 15:01:48 +08:00
/// <summary>
/// Sets a replay to be used, overriding local input.
/// </summary>
/// <param name="replay">The replay, null for local input.</param>
2017-08-23 15:48:13 +08:00
public virtual void SetReplay(Replay replay)
{
2017-08-24 19:31:57 +08:00
if (ReplayInputManager == null)
throw new InvalidOperationException($"A {nameof(KeyBindingInputManager)} which supports replay loading is not available");
Replay = replay;
ReplayInputManager.ReplayInputHandler = replay != null ? CreateReplayInputHandler(replay) : null;
HasReplayLoaded.Value = ReplayInputManager.ReplayInputHandler != null;
}
/// <summary>
/// Creates the cursor. May be null if the <see cref="RulesetContainer"/> doesn't provide a custom cursor.
/// </summary>
protected virtual CursorContainer CreateCursor() => null;
protected virtual IRulesetConfigManager CreateConfig(Ruleset ruleset, SettingsStore settings) => null;
/// <summary>
/// Creates a Playfield.
/// </summary>
/// <returns>The Playfield.</returns>
protected abstract Playfield CreatePlayfield();
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
if (rulesetConfig != null)
{
onScreenDisplay?.StopTracking(this, rulesetConfig);
rulesetConfig = null;
}
}
2017-03-06 12:59:11 +08:00
}
2016-11-02 13:07:20 +08:00
/// <summary>
2017-08-09 12:28:29 +08:00
/// RulesetContainer that applies conversion to Beatmaps. Does not contain a Playfield
/// and does not load drawable hit objects.
2017-03-15 20:58:00 +08:00
/// <para>
2017-09-06 17:05:51 +08:00
/// Should not be derived - derive <see cref="RulesetContainer{TObject}"/> instead.
2017-03-15 20:58:00 +08:00
/// </para>
/// </summary>
2017-08-09 12:28:29 +08:00
/// <typeparam name="TObject">The type of HitObject contained by this RulesetContainer.</typeparam>
public abstract class RulesetContainer<TObject> : RulesetContainer
2017-03-06 12:59:11 +08:00
where TObject : HitObject
{
2017-09-06 17:05:51 +08:00
public event Action<Judgement> OnJudgement;
public event Action<Judgement> OnJudgementRemoved;
2017-09-06 17:05:51 +08:00
/// <summary>
/// The Beatmap
/// </summary>
public Beatmap<TObject> Beatmap;
/// <summary>
/// All the converted hit objects contained by this hit renderer.
/// </summary>
public override IEnumerable<HitObject> Objects => Beatmap.HitObjects;
/// <summary>
/// The mods which are to be applied.
/// </summary>
protected IEnumerable<Mod> Mods;
/// <summary>
/// The <see cref="WorkingBeatmap"/> this <see cref="RulesetContainer{TObject}"/> was created with.
/// </summary>
protected readonly WorkingBeatmap WorkingBeatmap;
2017-08-22 13:18:17 +08:00
/// <summary>
/// Whether the specified beatmap is assumed to be specific to the current ruleset.
/// </summary>
public readonly bool IsForCurrentRuleset;
2017-08-22 13:18:17 +08:00
public override ScoreProcessor CreateScoreProcessor() => new ScoreProcessor<TObject>(this);
2017-09-06 17:05:51 +08:00
protected override Container<Drawable> Content => content;
private Container content;
private IEnumerable<Mod> mods;
[BackgroundDependencyLoader]
2018-03-15 06:14:03 +08:00
private void load(OsuConfigManager osuConfig)
{
2018-03-15 06:14:03 +08:00
// Apply mods
applyMods(Mods, osuConfig);
KeyBindingInputManager.Add(content = new Container
{
RelativeSizeAxes = Axes.Both,
});
AddInternal(KeyBindingInputManager);
KeyBindingInputManager.Add(Playfield);
if (Cursor != null)
KeyBindingInputManager.Add(Cursor);
loadObjects();
}
2017-09-06 17:05:51 +08:00
/// <summary>
/// Whether to assume the beatmap passed into this <see cref="RulesetContainer{TObject}"/> is for the current ruleset.
/// Creates a hit renderer for a beatmap.
/// </summary>
/// <param name="ruleset">The ruleset being repesented.</param>
/// <param name="workingBeatmap">The beatmap to create the hit renderer for.</param>
/// <param name="isForCurrentRuleset">Whether to assume the beatmap is for the current ruleset.</param>
2017-09-06 17:05:51 +08:00
protected RulesetContainer(Ruleset ruleset, WorkingBeatmap workingBeatmap, bool isForCurrentRuleset)
: base(ruleset)
{
Debug.Assert(workingBeatmap != null, "RulesetContainer initialized with a null beatmap.");
2017-03-14 16:14:11 +08:00
WorkingBeatmap = workingBeatmap;
2017-08-22 13:18:17 +08:00
IsForCurrentRuleset = isForCurrentRuleset;
Mods = workingBeatmap.Mods.Value;
2017-03-16 15:55:08 +08:00
RelativeSizeAxes = Axes.Both;
2017-03-11 23:34:21 +08:00
BeatmapConverter<TObject> converter = CreateBeatmapConverter();
BeatmapProcessor<TObject> processor = CreateBeatmapProcessor();
2017-03-11 23:34:21 +08:00
2017-04-17 14:44:46 +08:00
// Check if the beatmap can be converted
if (!converter.CanConvert(workingBeatmap.Beatmap))
throw new BeatmapInvalidForRulesetException($"{nameof(Beatmap)} can not be converted for the current ruleset (converter: {converter}).");
2017-04-17 14:44:46 +08:00
// Apply conversion adjustments before converting
foreach (var mod in Mods.OfType<IApplicableToBeatmapConverter<TObject>>())
mod.ApplyToBeatmapConverter(converter);
2017-03-16 15:55:08 +08:00
// Convert the beatmap
Beatmap = converter.Convert(workingBeatmap.Beatmap);
2017-03-16 15:55:08 +08:00
// Apply difficulty adjustments from mods before using Difficulty.
foreach (var mod in Mods.OfType<IApplicableToDifficulty>())
2017-10-19 13:05:11 +08:00
mod.ApplyToDifficulty(Beatmap.BeatmapInfo.BaseDifficulty);
// Post-process the beatmap
processor.PostProcess(Beatmap);
2017-03-16 15:55:08 +08:00
// Apply defaults
foreach (var h in Beatmap.HitObjects)
2017-10-19 13:05:11 +08:00
h.ApplyDefaults(Beatmap.ControlPointInfo, Beatmap.BeatmapInfo.BaseDifficulty);
2017-03-16 15:55:08 +08:00
KeyBindingInputManager = CreateInputManager();
KeyBindingInputManager.RelativeSizeAxes = Axes.Both;
2017-03-16 15:55:08 +08:00
// Add mods, should always be the last thing applied to give full control to mods
2018-03-12 04:19:34 +08:00
// Mods are now added in the load() method because we need the OsuConfigManager
2018-03-12 05:40:49 +08:00
// for the IReadFromConfig implementations. This method is still executed after the constructor,
2018-03-12 04:19:34 +08:00
// so the mods are still added in last
}
2017-05-03 14:56:54 +08:00
2017-08-23 15:48:13 +08:00
2017-09-06 17:05:51 +08:00
/// <summary>
/// Applies the active mods to this RulesetContainer.
/// </summary>
/// <param name="mods"></param>
private void applyMods(IEnumerable<Mod> mods, OsuConfigManager config)
2017-09-06 17:05:51 +08:00
{
2018-03-12 05:40:49 +08:00
if (mods == null)
return;
2017-09-06 17:05:51 +08:00
2018-03-15 06:14:03 +08:00
foreach (var mod in mods.OfType<IReadFromConfig>())
mod.ReadFromConfig(config);
foreach (var mod in mods.OfType<IApplicableToHitObject<TObject>>())
foreach (var obj in Beatmap.HitObjects)
mod.ApplyToHitObject(obj);
foreach (var mod in mods.OfType<IApplicableToRulesetContainer<TObject>>())
2017-09-06 17:05:51 +08:00
mod.ApplyToRulesetContainer(this);
}
2017-08-23 15:48:13 +08:00
public override void SetReplay(Replay replay)
{
base.SetReplay(replay);
if (ReplayInputManager?.ReplayInputHandler != null)
ReplayInputManager.ReplayInputHandler.ToScreenSpace = input => Playfield.ScaledContent.ToScreenSpace(input);
}
/// <summary>
/// Creates and adds drawable representations of hit objects to the play field.
/// </summary>
private void loadObjects()
{
2017-03-11 23:34:21 +08:00
foreach (TObject h in Beatmap.HitObjects)
{
var drawableObject = GetVisualRepresentation(h);
if (drawableObject == null)
continue;
drawableObject.OnJudgement += (d, j) => OnJudgement?.Invoke(j);
drawableObject.OnJudgementRemoved += (d, j) => OnJudgementRemoved?.Invoke(j);
2017-09-12 22:30:15 +08:00
Playfield.Add(drawableObject);
}
2017-02-10 13:16:23 +08:00
Playfield.PostProcess();
foreach (var mod in Mods.OfType<IApplicableToDrawableHitObjects>())
mod.ApplyToDrawableHitObjects(Playfield.HitObjects.Objects);
}
protected override void Update()
{
base.Update();
Playfield.Size = GetAspectAdjustedSize() * PlayfieldArea;
}
/// <summary>
2017-09-06 17:05:51 +08:00
/// Creates a processor to perform post-processing operations
/// on HitObjects in converted Beatmaps.
/// </summary>
/// <returns>The Beatmap processor.</returns>
protected virtual BeatmapProcessor<TObject> CreateBeatmapProcessor() => new BeatmapProcessor<TObject>();
/// <summary>
/// Computes the size of the <see cref="Playfield"/> in relative coordinate space after aspect adjustments.
/// </summary>
/// <returns>The aspect-adjusted size.</returns>
protected virtual Vector2 GetAspectAdjustedSize() => Vector2.One;
/// <summary>
/// The area of this <see cref="RulesetContainer"/> that is available for the <see cref="Playfield"/> to use.
/// Must be specified in relative coordinate space to this <see cref="RulesetContainer"/>.
/// This affects the final size of the <see cref="Playfield"/> but does not affect the <see cref="Playfield"/>'s scale.
/// </summary>
protected virtual Vector2 PlayfieldArea => new Vector2(0.75f); // A sane default
2017-09-06 17:05:51 +08:00
/// <summary>
/// Creates a converter to convert Beatmap to a specific mode.
/// </summary>
/// <returns>The Beatmap converter.</returns>
protected abstract BeatmapConverter<TObject> CreateBeatmapConverter();
/// <summary>
/// Creates a DrawableHitObject from a HitObject.
/// </summary>
/// <param name="h">The HitObject to make drawable.</param>
/// <returns>The DrawableHitObject.</returns>
2017-09-06 17:05:51 +08:00
protected abstract DrawableHitObject<TObject> GetVisualRepresentation(TObject h);
2016-09-02 18:25:13 +08:00
}
2017-04-17 14:44:46 +08:00
/// <summary>
2017-08-09 12:28:29 +08:00
/// A derivable RulesetContainer that manages the Playfield and HitObjects.
/// </summary>
2017-08-09 12:28:29 +08:00
/// <typeparam name="TPlayfield">The type of Playfield contained by this RulesetContainer.</typeparam>
/// <typeparam name="TObject">The type of HitObject contained by this RulesetContainer.</typeparam>
2017-09-06 17:05:51 +08:00
public abstract class RulesetContainer<TPlayfield, TObject> : RulesetContainer<TObject>
where TObject : HitObject
where TPlayfield : Playfield
{
/// <summary>
/// The playfield.
/// </summary>
protected new TPlayfield Playfield => (TPlayfield)base.Playfield;
/// <summary>
/// Creates a hit renderer for a beatmap.
/// </summary>
/// <param name="ruleset">The ruleset being repesented.</param>
/// <param name="beatmap">The beatmap to create the hit renderer for.</param>
/// <param name="isForCurrentRuleset">Whether to assume the beatmap is for the current ruleset.</param>
2017-08-09 12:28:29 +08:00
protected RulesetContainer(Ruleset ruleset, WorkingBeatmap beatmap, bool isForCurrentRuleset)
: base(ruleset, beatmap, isForCurrentRuleset)
{
}
}
2017-05-07 00:38:17 +08:00
public class BeatmapInvalidForRulesetException : ArgumentException
2017-04-17 14:44:46 +08:00
{
2017-04-20 11:11:03 +08:00
public BeatmapInvalidForRulesetException(string text)
2017-04-17 14:44:46 +08:00
: base(text)
{
}
}
2016-09-02 18:25:13 +08:00
}