mirror of
https://github.com/ppy/osu.git
synced 2025-01-07 18:33:04 +08:00
Merge branch 'master' into custom-game-storage
This commit is contained in:
commit
fc8c345ad7
@ -3,21 +3,26 @@
|
|||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using JetBrains.Annotations;
|
||||||
using osu.Game.Rulesets.Timing;
|
using osu.Game.Rulesets.Timing;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.UI.Scrolling.Algorithms
|
namespace osu.Game.Rulesets.UI.Scrolling.Algorithms
|
||||||
{
|
{
|
||||||
public class SequentialScrollAlgorithm : IScrollAlgorithm
|
public class SequentialScrollAlgorithm : IScrollAlgorithm
|
||||||
{
|
{
|
||||||
private readonly Dictionary<double, double> positionCache;
|
private static readonly IComparer<PositionMapping> by_position_comparer = Comparer<PositionMapping>.Create((c1, c2) => c1.Position.CompareTo(c2.Position));
|
||||||
|
|
||||||
private readonly IReadOnlyList<MultiplierControlPoint> controlPoints;
|
private readonly IReadOnlyList<MultiplierControlPoint> controlPoints;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Stores a mapping of time -> position for each control point.
|
||||||
|
/// </summary>
|
||||||
|
private readonly List<PositionMapping> positionMappings = new List<PositionMapping>();
|
||||||
|
|
||||||
public SequentialScrollAlgorithm(IReadOnlyList<MultiplierControlPoint> controlPoints)
|
public SequentialScrollAlgorithm(IReadOnlyList<MultiplierControlPoint> controlPoints)
|
||||||
{
|
{
|
||||||
this.controlPoints = controlPoints;
|
this.controlPoints = controlPoints;
|
||||||
|
|
||||||
positionCache = new Dictionary<double, double>();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public double GetDisplayStartTime(double originTime, float offset, double timeRange, float scrollLength)
|
public double GetDisplayStartTime(double originTime, float offset, double timeRange, float scrollLength)
|
||||||
@ -27,55 +32,31 @@ namespace osu.Game.Rulesets.UI.Scrolling.Algorithms
|
|||||||
|
|
||||||
public float GetLength(double startTime, double endTime, double timeRange, float scrollLength)
|
public float GetLength(double startTime, double endTime, double timeRange, float scrollLength)
|
||||||
{
|
{
|
||||||
var objectLength = relativePositionAtCached(endTime, timeRange) - relativePositionAtCached(startTime, timeRange);
|
var objectLength = relativePositionAt(endTime, timeRange) - relativePositionAt(startTime, timeRange);
|
||||||
return (float)(objectLength * scrollLength);
|
return (float)(objectLength * scrollLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
public float PositionAt(double time, double currentTime, double timeRange, float scrollLength)
|
public float PositionAt(double time, double currentTime, double timeRange, float scrollLength)
|
||||||
{
|
{
|
||||||
// Caching is not used here as currentTime is unlikely to have been previously cached
|
double timelineLength = relativePositionAt(time, timeRange) - relativePositionAt(currentTime, timeRange);
|
||||||
double timelinePosition = relativePositionAt(currentTime, timeRange);
|
return (float)(timelineLength * scrollLength);
|
||||||
return (float)((relativePositionAtCached(time, timeRange) - timelinePosition) * scrollLength);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public double TimeAt(float position, double currentTime, double timeRange, float scrollLength)
|
public double TimeAt(float position, double currentTime, double timeRange, float scrollLength)
|
||||||
{
|
{
|
||||||
// Convert the position to a length relative to time = 0
|
if (controlPoints.Count == 0)
|
||||||
double length = position / scrollLength + relativePositionAt(currentTime, timeRange);
|
return position * timeRange;
|
||||||
|
|
||||||
// We need to consider all timing points until the specified time and not just the currently-active one,
|
// Find the position at the current time, and the given length.
|
||||||
// since each timing point individually affects the positions of _all_ hitobjects after its start time
|
double relativePosition = relativePositionAt(currentTime, timeRange) + position / scrollLength;
|
||||||
for (int i = 0; i < controlPoints.Count; i++)
|
|
||||||
{
|
|
||||||
var current = controlPoints[i];
|
|
||||||
var next = i < controlPoints.Count - 1 ? controlPoints[i + 1] : null;
|
|
||||||
|
|
||||||
// Duration of the current control point
|
var positionMapping = findControlPointMapping(timeRange, new PositionMapping(0, null, relativePosition), by_position_comparer);
|
||||||
var currentDuration = (next?.StartTime ?? double.PositiveInfinity) - current.StartTime;
|
|
||||||
|
|
||||||
// Figure out the length of control point
|
// Begin at the control point's time and add the remaining time to reach the given position.
|
||||||
var currentLength = currentDuration / timeRange * current.Multiplier;
|
return positionMapping.Time + (relativePosition - positionMapping.Position) * timeRange / positionMapping.ControlPoint.Multiplier;
|
||||||
|
|
||||||
if (currentLength > length)
|
|
||||||
{
|
|
||||||
// The point is within this control point
|
|
||||||
return current.StartTime + length * timeRange / current.Multiplier;
|
|
||||||
}
|
|
||||||
|
|
||||||
length -= currentLength;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0; // Should never occur
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private double relativePositionAtCached(double time, double timeRange)
|
public void Reset() => positionMappings.Clear();
|
||||||
{
|
|
||||||
if (!positionCache.TryGetValue(time, out double existing))
|
|
||||||
positionCache[time] = existing = relativePositionAt(time, timeRange);
|
|
||||||
return existing;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Reset() => positionCache.Clear();
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Finds the position which corresponds to a point in time.
|
/// Finds the position which corresponds to a point in time.
|
||||||
@ -84,37 +65,100 @@ namespace osu.Game.Rulesets.UI.Scrolling.Algorithms
|
|||||||
/// <param name="time">The time to find the position at.</param>
|
/// <param name="time">The time to find the position at.</param>
|
||||||
/// <param name="timeRange">The amount of time visualised by the scrolling area.</param>
|
/// <param name="timeRange">The amount of time visualised by the scrolling area.</param>
|
||||||
/// <returns>A positive value indicating the position at <paramref name="time"/>.</returns>
|
/// <returns>A positive value indicating the position at <paramref name="time"/>.</returns>
|
||||||
private double relativePositionAt(double time, double timeRange)
|
private double relativePositionAt(in double time, in double timeRange)
|
||||||
{
|
{
|
||||||
if (controlPoints.Count == 0)
|
if (controlPoints.Count == 0)
|
||||||
return time / timeRange;
|
return time / timeRange;
|
||||||
|
|
||||||
double length = 0;
|
var mapping = findControlPointMapping(timeRange, new PositionMapping(time));
|
||||||
|
|
||||||
// We need to consider all timing points until the specified time and not just the currently-active one,
|
// Begin at the control point's position and add the remaining distance to reach the given time.
|
||||||
// since each timing point individually affects the positions of _all_ hitobjects after its start time
|
return mapping.Position + (time - mapping.Time) / timeRange * mapping.ControlPoint.Multiplier;
|
||||||
for (int i = 0; i < controlPoints.Count; i++)
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Finds a <see cref="MultiplierControlPoint"/>'s <see cref="PositionMapping"/> that is relevant to a given <see cref="PositionMapping"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// This is used to find the last <see cref="MultiplierControlPoint"/> occuring prior to a time value, or prior to a position value (if <see cref="by_position_comparer"/> is used).
|
||||||
|
/// </remarks>
|
||||||
|
/// <param name="timeRange">The time range.</param>
|
||||||
|
/// <param name="search">The <see cref="PositionMapping"/> to find the closest <see cref="PositionMapping"/> to.</param>
|
||||||
|
/// <param name="comparer">The comparison. If null, the default comparer is used (by time).</param>
|
||||||
|
/// <returns>The <see cref="MultiplierControlPoint"/>'s <see cref="PositionMapping"/> that is relevant for <paramref name="search"/>.</returns>
|
||||||
|
private PositionMapping findControlPointMapping(in double timeRange, in PositionMapping search, IComparer<PositionMapping> comparer = null)
|
||||||
|
{
|
||||||
|
generatePositionMappings(timeRange);
|
||||||
|
|
||||||
|
var mappingIndex = positionMappings.BinarySearch(search, comparer ?? Comparer<PositionMapping>.Default);
|
||||||
|
|
||||||
|
if (mappingIndex < 0)
|
||||||
{
|
{
|
||||||
var current = controlPoints[i];
|
// If the search value isn't found, the _next_ control point is returned, but we actually want the _previous_ control point.
|
||||||
var next = i < controlPoints.Count - 1 ? controlPoints[i + 1] : null;
|
// In doing so, we must make sure to not underflow the position mapping list (i.e. always use the 0th control point for time < first_control_point_time).
|
||||||
|
mappingIndex = Math.Max(0, ~mappingIndex - 1);
|
||||||
|
|
||||||
// We don't need to consider any control points beyond the current time, since it will not yet
|
Debug.Assert(mappingIndex < positionMappings.Count);
|
||||||
// affect any hitobjects
|
|
||||||
if (i > 0 && current.StartTime > time)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
// Duration of the current control point
|
|
||||||
var currentDuration = (next?.StartTime ?? double.PositiveInfinity) - current.StartTime;
|
|
||||||
|
|
||||||
// We want to consider the minimal amount of time that this control point has affected,
|
|
||||||
// which may be either its duration, or the amount of time that has passed within it
|
|
||||||
var durationInCurrent = Math.Min(currentDuration, time - current.StartTime);
|
|
||||||
|
|
||||||
// Figure out how much of the time range the duration represents, and adjust it by the speed multiplier
|
|
||||||
length += durationInCurrent / timeRange * current.Multiplier;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return length;
|
var mapping = positionMappings[mappingIndex];
|
||||||
|
Debug.Assert(mapping.ControlPoint != null);
|
||||||
|
|
||||||
|
return mapping;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generates the mapping of <see cref="MultiplierControlPoint"/> (and their respective start times) to their relative position from 0.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="timeRange">The time range.</param>
|
||||||
|
private void generatePositionMappings(in double timeRange)
|
||||||
|
{
|
||||||
|
if (positionMappings.Count > 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (controlPoints.Count == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
positionMappings.Add(new PositionMapping(controlPoints[0].StartTime, controlPoints[0]));
|
||||||
|
|
||||||
|
for (int i = 0; i < controlPoints.Count - 1; i++)
|
||||||
|
{
|
||||||
|
var current = controlPoints[i];
|
||||||
|
var next = controlPoints[i + 1];
|
||||||
|
|
||||||
|
// Figure out how much of the time range the duration represents, and adjust it by the speed multiplier
|
||||||
|
float length = (float)((next.StartTime - current.StartTime) / timeRange * current.Multiplier);
|
||||||
|
|
||||||
|
positionMappings.Add(new PositionMapping(next.StartTime, next, positionMappings[^1].Position + length));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly struct PositionMapping : IComparable<PositionMapping>
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The time corresponding to this position.
|
||||||
|
/// </summary>
|
||||||
|
public readonly double Time;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The <see cref="MultiplierControlPoint"/> at <see cref="Time"/>.
|
||||||
|
/// </summary>
|
||||||
|
[CanBeNull]
|
||||||
|
public readonly MultiplierControlPoint ControlPoint;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The relative position from 0 of <see cref="ControlPoint"/>.
|
||||||
|
/// </summary>
|
||||||
|
public readonly double Position;
|
||||||
|
|
||||||
|
public PositionMapping(double time, MultiplierControlPoint controlPoint = null, double position = default)
|
||||||
|
{
|
||||||
|
Time = time;
|
||||||
|
ControlPoint = controlPoint;
|
||||||
|
Position = position;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int CompareTo(PositionMapping other) => Time.CompareTo(other.Time);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -70,41 +70,33 @@ namespace osu.Game.Screens.Ranking
|
|||||||
{
|
{
|
||||||
new Drawable[]
|
new Drawable[]
|
||||||
{
|
{
|
||||||
new Container
|
new VerticalScrollContainer
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.Both,
|
RelativeSizeAxes = Axes.Both,
|
||||||
Children = new Drawable[]
|
ScrollbarVisible = false,
|
||||||
|
Child = new Container
|
||||||
{
|
{
|
||||||
new OsuScrollContainer
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
Children = new Drawable[]
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.Both,
|
statisticsPanel = new StatisticsPanel
|
||||||
ScrollbarVisible = false,
|
|
||||||
Child = new Container
|
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.X,
|
RelativeSizeAxes = Axes.Both,
|
||||||
Height = screen_height,
|
Score = { BindTarget = SelectedScore }
|
||||||
Children = new Drawable[]
|
},
|
||||||
{
|
scorePanelList = new ScorePanelList
|
||||||
scorePanelList = new ScorePanelList
|
{
|
||||||
{
|
RelativeSizeAxes = Axes.Both,
|
||||||
RelativeSizeAxes = Axes.Both,
|
SelectedScore = { BindTarget = SelectedScore },
|
||||||
SelectedScore = { BindTarget = SelectedScore },
|
PostExpandAction = () => statisticsPanel.ToggleVisibility()
|
||||||
PostExpandAction = () => statisticsPanel.ToggleVisibility()
|
},
|
||||||
},
|
detachedPanelContainer = new Container<ScorePanel>
|
||||||
detachedPanelContainer = new Container<ScorePanel>
|
{
|
||||||
{
|
RelativeSizeAxes = Axes.Both
|
||||||
RelativeSizeAxes = Axes.Both
|
},
|
||||||
},
|
}
|
||||||
statisticsPanel = new StatisticsPanel
|
|
||||||
{
|
|
||||||
RelativeSizeAxes = Axes.Both,
|
|
||||||
Score = { BindTarget = SelectedScore }
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
new[]
|
new[]
|
||||||
{
|
{
|
||||||
@ -277,5 +269,23 @@ namespace osu.Game.Screens.Ranking
|
|||||||
detachedPanel = null;
|
detachedPanel = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private class VerticalScrollContainer : OsuScrollContainer
|
||||||
|
{
|
||||||
|
protected override Container<Drawable> Content => content;
|
||||||
|
|
||||||
|
private readonly Container content;
|
||||||
|
|
||||||
|
public VerticalScrollContainer()
|
||||||
|
{
|
||||||
|
base.Content.Add(content = new Container { RelativeSizeAxes = Axes.X });
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Update()
|
||||||
|
{
|
||||||
|
base.Update();
|
||||||
|
content.Height = Math.Max(screen_height, DrawHeight);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,8 +7,10 @@ using osu.Framework.Allocation;
|
|||||||
using osu.Framework.Audio.Track;
|
using osu.Framework.Audio.Track;
|
||||||
using osu.Framework.Bindables;
|
using osu.Framework.Bindables;
|
||||||
using osu.Framework.Extensions.IEnumerableExtensions;
|
using osu.Framework.Extensions.IEnumerableExtensions;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Graphics.Audio;
|
using osu.Framework.Graphics.Audio;
|
||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Graphics.Transforms;
|
||||||
using osu.Game.Audio;
|
using osu.Game.Audio;
|
||||||
|
|
||||||
namespace osu.Game.Skinning
|
namespace osu.Game.Skinning
|
||||||
@ -43,6 +45,34 @@ namespace osu.Game.Skinning
|
|||||||
|
|
||||||
public BindableNumber<double> Tempo => samplesContainer.Tempo;
|
public BindableNumber<double> Tempo => samplesContainer.Tempo;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Smoothly adjusts <see cref="Volume"/> over time.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="TransformSequence{T}"/> to which further transforms can be added.</returns>
|
||||||
|
public TransformSequence<DrawableAudioWrapper> VolumeTo(double newVolume, double duration = 0, Easing easing = Easing.None) =>
|
||||||
|
samplesContainer.VolumeTo(newVolume, duration, easing);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Smoothly adjusts <see cref="Balance"/> over time.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="TransformSequence{T}"/> to which further transforms can be added.</returns>
|
||||||
|
public TransformSequence<DrawableAudioWrapper> BalanceTo(double newBalance, double duration = 0, Easing easing = Easing.None) =>
|
||||||
|
samplesContainer.BalanceTo(newBalance, duration, easing);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Smoothly adjusts <see cref="Frequency"/> over time.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="TransformSequence{T}"/> to which further transforms can be added.</returns>
|
||||||
|
public TransformSequence<DrawableAudioWrapper> FrequencyTo(double newFrequency, double duration = 0, Easing easing = Easing.None) =>
|
||||||
|
samplesContainer.FrequencyTo(newFrequency, duration, easing);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Smoothly adjusts <see cref="Tempo"/> over time.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A <see cref="TransformSequence{T}"/> to which further transforms can be added.</returns>
|
||||||
|
public TransformSequence<DrawableAudioWrapper> TempoTo(double newTempo, double duration = 0, Easing easing = Easing.None) =>
|
||||||
|
samplesContainer.TempoTo(newTempo, duration, easing);
|
||||||
|
|
||||||
public bool Looping
|
public bool Looping
|
||||||
{
|
{
|
||||||
get => looping;
|
get => looping;
|
||||||
|
Loading…
Reference in New Issue
Block a user