1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-07 20:42:54 +08:00
osu-lazer/osu.Game/Beatmaps/BeatmapConverter.cs

113 lines
4.5 KiB
C#
Raw Normal View History

2018-01-05 19:21:19 +08:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
2017-03-11 23:34:21 +08:00
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2017-04-17 14:44:46 +08:00
using System;
using System.Collections.Generic;
using System.Linq;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Objects;
2017-03-11 23:34:21 +08:00
2017-09-18 21:32:49 +08:00
namespace osu.Game.Beatmaps
2017-03-11 23:34:21 +08:00
{
/// <summary>
/// Converts a Beatmap for another mode.
/// </summary>
/// <typeparam name="T">The type of HitObject stored in the Beatmap.</typeparam>
public abstract class BeatmapConverter<T> : IBeatmapConverter
where T : HitObject
2017-03-11 23:34:21 +08:00
{
private event Action<HitObject, IEnumerable<HitObject>> ObjectConverted;
event Action<HitObject, IEnumerable<HitObject>> IBeatmapConverter.ObjectConverted
{
add => ObjectConverted += value;
remove => ObjectConverted -= value;
}
2017-04-17 14:44:46 +08:00
/// <summary>
2017-04-18 13:24:16 +08:00
/// Checks if a Beatmap can be converted using this Beatmap Converter.
2017-04-17 14:44:46 +08:00
/// </summary>
2017-04-18 13:24:16 +08:00
/// <param name="beatmap">The Beatmap to check.</param>
/// <returns>Whether the Beatmap can be converted using this Beatmap Converter.</returns>
public bool CanConvert(Beatmap beatmap) => ValidConversionTypes.All(t => beatmap.HitObjects.Any(t.IsInstanceOfType));
2017-04-17 14:44:46 +08:00
/// <summary>
2017-04-18 13:24:16 +08:00
/// Converts a Beatmap using this Beatmap Converter.
/// </summary>
2017-04-18 13:24:16 +08:00
/// <param name="original">The un-converted Beatmap.</param>
/// <returns>The converted Beatmap.</returns>
2017-08-22 13:18:17 +08:00
public Beatmap<T> Convert(Beatmap original)
2017-04-18 13:24:16 +08:00
{
// We always operate on a clone of the original beatmap, to not modify it game-wide
2017-08-22 13:18:17 +08:00
return ConvertBeatmap(new Beatmap(original));
2017-04-18 13:24:16 +08:00
}
2017-04-17 14:44:46 +08:00
void IBeatmapConverter.Convert(Beatmap original) => Convert(original);
2017-04-17 14:44:46 +08:00
/// <summary>
2017-04-18 13:24:16 +08:00
/// Performs the conversion of a Beatmap using this Beatmap Converter.
2017-04-17 14:44:46 +08:00
/// </summary>
2017-04-18 13:24:16 +08:00
/// <param name="original">The un-converted Beatmap.</param>
/// <returns>The converted Beatmap.</returns>
2017-08-22 13:18:17 +08:00
protected virtual Beatmap<T> ConvertBeatmap(Beatmap original)
2017-04-18 13:24:16 +08:00
{
var beatmap = CreateBeatmap();
// todo: this *must* share logic (or directly use) Beatmap<T>'s constructor.
// right now this isn't easily possible due to generic entanglement.
beatmap.BeatmapInfo = original.BeatmapInfo;
beatmap.ControlPointInfo = original.ControlPointInfo;
beatmap.HitObjects = original.HitObjects.SelectMany(h => convert(h, original)).ToList();
beatmap.Breaks = original.Breaks;
return beatmap;
2017-04-18 13:24:16 +08:00
}
/// <summary>
/// Converts a hit object.
/// </summary>
/// <param name="original">The hit object to convert.</param>
/// <param name="beatmap">The un-converted Beatmap.</param>
/// <returns>The converted hit object.</returns>
private IEnumerable<T> convert(HitObject original, Beatmap beatmap)
{
// Check if the hitobject is already the converted type
T tObject = original as T;
if (tObject != null)
{
2017-04-18 13:24:16 +08:00
yield return tObject;
yield break;
}
2017-04-18 13:34:39 +08:00
var converted = ConvertHitObject(original, beatmap).ToList();
ObjectConverted?.Invoke(original, converted);
2017-04-18 13:24:16 +08:00
// Convert the hit object
foreach (var obj in converted)
{
if (obj == null)
continue;
2017-04-18 13:24:16 +08:00
yield return obj;
}
2017-04-18 13:24:16 +08:00
}
/// <summary>
/// The types of HitObjects that can be converted to be used for this Beatmap.
/// </summary>
protected abstract IEnumerable<Type> ValidConversionTypes { get; }
/// <summary>
/// Creates the <see cref="Beatmap{T}"/> that will be returned by this <see cref="BeatmapProcessor{T}"/>.
/// </summary>
protected virtual Beatmap<T> CreateBeatmap() => new Beatmap<T>();
2017-04-18 13:24:16 +08:00
/// <summary>
/// Performs the conversion of a hit object.
2017-11-28 20:30:03 +08:00
/// This method is generally executed sequentially for all objects in a beatmap.
2017-04-18 13:24:16 +08:00
/// </summary>
/// <param name="original">The hit object to convert.</param>
/// <param name="beatmap">The un-converted Beatmap.</param>
/// <returns>The converted hit object.</returns>
protected abstract IEnumerable<T> ConvertHitObject(HitObject original, Beatmap beatmap);
2017-04-17 14:44:46 +08:00
}
2017-03-11 23:34:21 +08:00
}