1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-24 09:02:57 +08:00
osu-lazer/osu.Game/Storyboards/Drawables/DrawableStoryboardSample.cs

85 lines
3.1 KiB
C#
Raw Normal View History

// 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-06-28 12:43:56 +08:00
2020-06-16 21:54:50 +08:00
using System.Collections.Generic;
using System.Linq;
2018-06-28 12:43:56 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio.Sample;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
2018-06-28 12:43:56 +08:00
using osu.Framework.Graphics;
using osu.Game.Beatmaps;
2020-06-16 21:54:50 +08:00
using osu.Game.Rulesets.Mods;
2018-06-28 12:43:56 +08:00
namespace osu.Game.Storyboards.Drawables
{
public class DrawableStoryboardSample : Component
{
/// <summary>
/// The amount of time allowable beyond the start time of the sample, for the sample to start.
/// </summary>
private const double allowable_late_start = 100;
private readonly StoryboardSampleInfo sampleInfo;
2020-06-19 04:46:32 +08:00
2020-06-20 02:13:43 +08:00
protected SampleChannel Channel { get; private set; }
2018-06-28 12:43:56 +08:00
public override bool RemoveWhenNotAlive => false;
public DrawableStoryboardSample(StoryboardSampleInfo sampleInfo)
2018-06-28 12:43:56 +08:00
{
this.sampleInfo = sampleInfo;
LifetimeStart = sampleInfo.StartTime;
2018-06-28 12:43:56 +08:00
}
[BackgroundDependencyLoader]
2020-06-24 23:17:18 +08:00
private void load(IBindable<WorkingBeatmap> beatmap, IBindable<IReadOnlyList<Mod>> mods)
2018-06-28 12:43:56 +08:00
{
2020-06-19 04:46:32 +08:00
Channel = beatmap.Value.Skin.GetSample(sampleInfo);
2020-06-20 02:15:14 +08:00
if (Channel == null)
return;
2018-06-28 12:43:56 +08:00
2020-06-20 02:15:14 +08:00
Channel.Volume.Value = sampleInfo.Volume / 100.0;
2020-06-16 21:54:50 +08:00
2020-06-20 02:15:14 +08:00
foreach (var mod in mods.Value.OfType<IApplicableToSample>())
mod.ApplyToSample(Channel);
2018-06-28 12:43:56 +08:00
}
protected override void Update()
{
base.Update();
2018-09-15 22:30:11 +08:00
// TODO: this logic will need to be consolidated with other game samples like hit sounds.
if (Time.Current < sampleInfo.StartTime)
2018-06-28 12:43:56 +08:00
{
// We've rewound before the start time of the sample
2020-06-19 04:46:32 +08:00
Channel?.Stop();
2018-06-28 12:43:56 +08:00
// In the case that the user fast-forwards to a point far beyond the start time of the sample,
// we want to be able to fall into the if-conditional below (therefore we must not have a life time end)
LifetimeStart = sampleInfo.StartTime;
2018-06-28 12:43:56 +08:00
LifetimeEnd = double.MaxValue;
}
else if (Time.Current - Time.Elapsed <= sampleInfo.StartTime)
2018-06-28 12:43:56 +08:00
{
// We've passed the start time of the sample. We only play the sample if we're within an allowable range
// from the sample's start, to reduce layering if we've been fast-forwarded far into the future
if (Time.Current - sampleInfo.StartTime < allowable_late_start)
2020-06-19 04:46:32 +08:00
Channel?.Play();
2018-06-28 12:43:56 +08:00
// In the case that the user rewinds to a point far behind the start time of the sample,
// we want to be able to fall into the if-conditional above (therefore we must not have a life time start)
LifetimeStart = double.MinValue;
LifetimeEnd = sampleInfo.StartTime;
2018-06-28 12:43:56 +08:00
}
}
protected override void Dispose(bool isDisposing)
{
2020-06-19 04:46:32 +08:00
Channel?.Stop();
Channel = null;
base.Dispose(isDisposing);
}
2018-06-28 12:43:56 +08:00
}
}