1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-28 04:42:58 +08:00

Add a decoupled clock to allow for lead-in and lead-out time.

This commit is contained in:
Dean Herbert 2017-04-26 18:16:55 +09:00
parent e003d9fc3c
commit 01caaf44f3

View File

@ -38,7 +38,7 @@ namespace osu.Game.Screens.Play
public Action RestartRequested; public Action RestartRequested;
public bool IsPaused => !interpolatedSourceClock.IsRunning; public bool IsPaused => !decoupledClock.IsRunning;
internal override bool AllowRulesetChange => false; internal override bool AllowRulesetChange => false;
@ -51,9 +51,9 @@ namespace osu.Game.Screens.Play
private bool canPause => ValidForResume && !HasFailed && Time.Current >= lastPauseActionTime + pause_cooldown; private bool canPause => ValidForResume && !HasFailed && Time.Current >= lastPauseActionTime + pause_cooldown;
private IAdjustableClock sourceClock; private IAdjustableClock adjustableSourceClock;
private OffsetClock offsetClock; private FramedOffsetClock offsetClock;
private IFrameBasedClock interpolatedSourceClock; private DecoupleableInterpolatingFramedClock decoupledClock;
private RulesetInfo ruleset; private RulesetInfo ruleset;
@ -123,23 +123,31 @@ namespace osu.Game.Screens.Play
if (track != null) if (track != null)
{ {
audio.Track.SetExclusive(track); audio.Track.SetExclusive(track);
sourceClock = track; adjustableSourceClock = track;
} }
sourceClock = (IAdjustableClock)track ?? new StopwatchClock(); adjustableSourceClock = (IAdjustableClock)track ?? new StopwatchClock();
offsetClock = new OffsetClock(sourceClock);
decoupledClock = new DecoupleableInterpolatingFramedClock();
decoupledClock.ChangeSource(adjustableSourceClock);
decoupledClock.IsCoupled = false;
offsetClock = new FramedOffsetClock(decoupledClock);
userAudioOffset = config.GetBindable<double>(OsuConfig.AudioOffset); userAudioOffset = config.GetBindable<double>(OsuConfig.AudioOffset);
userAudioOffset.ValueChanged += v => offsetClock.Offset = v; userAudioOffset.ValueChanged += v => offsetClock.Offset = v;
userAudioOffset.TriggerChange(); userAudioOffset.TriggerChange();
interpolatedSourceClock = new InterpolatingFramedClock(offsetClock);
Schedule(() => Schedule(() =>
{ {
sourceClock.Reset(); adjustableSourceClock.Reset();
var firstObjectTime = HitRenderer.Objects.First().StartTime;
decoupledClock.Seek(Math.Min(0, firstObjectTime - Math.Max(Beatmap.Beatmap.TimingInfo.BeatLengthAt(firstObjectTime) * 2, Beatmap.BeatmapInfo.AudioLeadIn)));
foreach (var mod in Beatmap.Mods.Value.OfType<IApplicableToClock>()) foreach (var mod in Beatmap.Mods.Value.OfType<IApplicableToClock>())
mod.ApplyToClock(sourceClock); mod.ApplyToClock(adjustableSourceClock);
}); });
scoreProcessor = HitRenderer.CreateScoreProcessor(); scoreProcessor = HitRenderer.CreateScoreProcessor();
@ -155,7 +163,7 @@ namespace osu.Game.Screens.Play
hudOverlay.BindHitRenderer(HitRenderer); hudOverlay.BindHitRenderer(HitRenderer);
hudOverlay.Progress.Objects = HitRenderer.Objects; hudOverlay.Progress.Objects = HitRenderer.Objects;
hudOverlay.Progress.AudioClock = interpolatedSourceClock; hudOverlay.Progress.AudioClock = decoupledClock;
hudOverlay.Progress.AllowSeeking = HitRenderer.HasReplayLoaded; hudOverlay.Progress.AllowSeeking = HitRenderer.HasReplayLoaded;
hudOverlay.Progress.OnSeek = progress => hudOverlay.Progress.OnSeek = progress =>
{ {
@ -173,7 +181,7 @@ namespace osu.Game.Screens.Play
new Container new Container
{ {
RelativeSizeAxes = Axes.Both, RelativeSizeAxes = Axes.Both,
Clock = interpolatedSourceClock, Clock = offsetClock,
Children = new Drawable[] Children = new Drawable[]
{ {
HitRenderer, HitRenderer,
@ -229,7 +237,7 @@ namespace osu.Game.Screens.Play
skipButton.Action = () => skipButton.Action = () =>
{ {
sourceClock.Seek(firstHitObject - skip_required_cutoff - fade_time); decoupledClock.Seek(firstHitObject - skip_required_cutoff - fade_time);
skipButton.Action = null; skipButton.Action = null;
}; };
@ -246,7 +254,7 @@ namespace osu.Game.Screens.Play
// we want to wait for the source clock to stop so we can be sure all components are in a stable state. // we want to wait for the source clock to stop so we can be sure all components are in a stable state.
if (!IsPaused) if (!IsPaused)
{ {
sourceClock.Stop(); decoupledClock.Stop();
Schedule(() => Pause(force)); Schedule(() => Pause(force));
return; return;
@ -273,7 +281,7 @@ namespace osu.Game.Screens.Play
hudOverlay.KeyCounter.IsCounting = true; hudOverlay.KeyCounter.IsCounting = true;
hudOverlay.Progress.Hide(); hudOverlay.Progress.Hide();
pauseOverlay.Hide(); pauseOverlay.Hide();
sourceClock.Start(); decoupledClock.Start();
} }
public void Restart() public void Restart()
@ -309,7 +317,7 @@ namespace osu.Game.Screens.Play
private void onFail() private void onFail()
{ {
sourceClock.Stop(); decoupledClock.Stop();
HasFailed = true; HasFailed = true;
failOverlay.Retries = RestartCount; failOverlay.Retries = RestartCount;
@ -337,7 +345,7 @@ namespace osu.Game.Screens.Play
Delay(750); Delay(750);
Schedule(() => Schedule(() =>
{ {
sourceClock.Start(); decoupledClock.Start();
initializeSkipButton(); initializeSkipButton();
}); });