2019-01-24 16:43:03 +08: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 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Objects
|
|
|
|
|
{
|
|
|
|
|
public class HitWindows
|
|
|
|
|
{
|
|
|
|
|
private static readonly IReadOnlyDictionary<HitResult, (double od0, double od5, double od10)> base_ranges = new Dictionary<HitResult, (double, double, double)>
|
|
|
|
|
{
|
|
|
|
|
{ HitResult.Perfect, (44.8, 38.8, 27.8) },
|
2019-02-28 12:31:40 +08:00
|
|
|
|
{ HitResult.Great, (128, 98, 68) },
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{ HitResult.Good, (194, 164, 134) },
|
|
|
|
|
{ HitResult.Ok, (254, 224, 194) },
|
|
|
|
|
{ HitResult.Meh, (302, 272, 242) },
|
|
|
|
|
{ HitResult.Miss, (376, 346, 316) },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for a <see cref="HitResult.Perfect"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Perfect { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for a <see cref="HitResult.Great"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Great { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for a <see cref="HitResult.Good"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Good { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for an <see cref="HitResult.Ok"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Ok { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for a <see cref="HitResult.Meh"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Meh { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Hit window for a <see cref="HitResult.Miss"/> result.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double Miss { get; protected set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2018-12-12 18:15:59 +08:00
|
|
|
|
/// Retrieves the <see cref="HitResult"/> with the largest hit window that produces a successful hit.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// </summary>
|
2018-12-12 18:15:59 +08:00
|
|
|
|
/// <returns>The lowest allowed successful <see cref="HitResult"/>.</returns>
|
|
|
|
|
protected HitResult LowestSuccessfulHitResult()
|
|
|
|
|
{
|
|
|
|
|
for (var result = HitResult.Meh; result <= HitResult.Perfect; ++result)
|
|
|
|
|
{
|
|
|
|
|
if (IsHitResultAllowed(result))
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return HitResult.None;
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-08-30 17:35:06 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves a mapping of <see cref="HitResult"/>s to their half window timing for all allowed <see cref="HitResult"/>s.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public IEnumerable<(HitResult result, double length)> GetAllAvailableHalfWindows()
|
|
|
|
|
{
|
|
|
|
|
for (var result = HitResult.Meh; result <= HitResult.Perfect; ++result)
|
|
|
|
|
{
|
|
|
|
|
if (IsHitResultAllowed(result))
|
|
|
|
|
yield return (result, HalfWindowFor(result));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// <summary>
|
2018-12-12 14:11:03 +08:00
|
|
|
|
/// Check whether it is possible to achieve the provided <see cref="HitResult"/>.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// </summary>
|
2018-12-12 14:10:47 +08:00
|
|
|
|
/// <param name="result">The result type to check.</param>
|
|
|
|
|
/// <returns>Whether the <see cref="HitResult"/> can be achieved.</returns>
|
2018-12-06 20:04:54 +08:00
|
|
|
|
public virtual bool IsHitResultAllowed(HitResult result)
|
|
|
|
|
{
|
2018-12-10 13:06:18 +08:00
|
|
|
|
switch (result)
|
2018-12-06 20:04:54 +08:00
|
|
|
|
{
|
|
|
|
|
case HitResult.Perfect:
|
|
|
|
|
case HitResult.Ok:
|
|
|
|
|
return false;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-12-06 20:04:54 +08:00
|
|
|
|
default:
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
2018-05-11 14:30:26 +08:00
|
|
|
|
/// Sets hit windows with values that correspond to a difficulty parameter.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="difficulty">The parameter.</param>
|
2018-05-11 14:30:26 +08:00
|
|
|
|
public virtual void SetDifficulty(double difficulty)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
Perfect = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Perfect]);
|
|
|
|
|
Great = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Great]);
|
|
|
|
|
Good = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Good]);
|
|
|
|
|
Ok = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Ok]);
|
|
|
|
|
Meh = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Meh]);
|
|
|
|
|
Miss = BeatmapDifficulty.DifficultyRange(difficulty, base_ranges[HitResult.Miss]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves the <see cref="HitResult"/> for a time offset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="timeOffset">The time offset.</param>
|
|
|
|
|
/// <returns>The hit result, or <see cref="HitResult.None"/> if <paramref name="timeOffset"/> doesn't result in a judgement.</returns>
|
|
|
|
|
public HitResult ResultFor(double timeOffset)
|
|
|
|
|
{
|
|
|
|
|
timeOffset = Math.Abs(timeOffset);
|
|
|
|
|
|
2018-12-10 00:38:19 +08:00
|
|
|
|
for (var result = HitResult.Perfect; result >= HitResult.Miss; --result)
|
2018-12-06 20:04:54 +08:00
|
|
|
|
{
|
2018-12-10 00:38:29 +08:00
|
|
|
|
if (IsHitResultAllowed(result) && timeOffset <= HalfWindowFor(result))
|
2018-12-06 20:04:54 +08:00
|
|
|
|
return result;
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
return HitResult.None;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves half the hit window for a <see cref="HitResult"/>.
|
|
|
|
|
/// This is useful if the hit window for one half of the hittable range of a <see cref="HitObject"/> is required.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="result">The expected <see cref="HitResult"/>.</param>
|
|
|
|
|
/// <returns>One half of the hit window for <paramref name="result"/>.</returns>
|
|
|
|
|
public double HalfWindowFor(HitResult result)
|
|
|
|
|
{
|
|
|
|
|
switch (result)
|
|
|
|
|
{
|
|
|
|
|
case HitResult.Perfect:
|
|
|
|
|
return Perfect / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case HitResult.Great:
|
|
|
|
|
return Great / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case HitResult.Good:
|
|
|
|
|
return Good / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case HitResult.Ok:
|
|
|
|
|
return Ok / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case HitResult.Meh:
|
|
|
|
|
return Meh / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case HitResult.Miss:
|
|
|
|
|
return Miss / 2;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
default:
|
|
|
|
|
throw new ArgumentException(nameof(result));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Given a time offset, whether the <see cref="HitObject"/> can ever be hit in the future with a non-<see cref="HitResult.Miss"/> result.
|
2019-04-25 16:36:17 +08:00
|
|
|
|
/// This happens if <paramref name="timeOffset"/> is less than what is required for <see cref="LowestSuccessfulHitResult"/>.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="timeOffset">The time offset.</param>
|
|
|
|
|
/// <returns>Whether the <see cref="HitObject"/> can be hit at any point in the future from this time offset.</returns>
|
2018-12-12 18:15:59 +08:00
|
|
|
|
public bool CanBeHit(double timeOffset) => timeOffset <= HalfWindowFor(LowestSuccessfulHitResult());
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
}
|