1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 14:23:14 +08:00
osu-lazer/osu.Game.Rulesets.Mania/UI/PlayfieldCoveringWrapper.cs

172 lines
6.2 KiB
C#
Raw Normal View History

2023-06-23 00:37:25 +08:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
2020-07-16 16:26:18 +08:00
// See the LICENCE file in the repository root for full licence text.
using System;
2024-02-15 23:40:58 +08:00
using System.ComponentModel;
2020-07-16 16:26:18 +08:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Utils;
2020-07-16 16:26:18 +08:00
using osu.Game.Rulesets.UI.Scrolling;
using osuTK;
2020-07-16 16:26:18 +08:00
using osuTK.Graphics;
2024-02-15 23:40:58 +08:00
using Container = osu.Framework.Graphics.Containers.Container;
2020-07-16 16:26:18 +08:00
namespace osu.Game.Rulesets.Mania.UI
{
/// <summary>
2024-02-15 23:40:58 +08:00
/// A <see cref="Framework.Graphics.Containers.Container"/> that has its contents partially hidden by an adjustable "cover". This is intended to be used in a playfield.
2020-07-16 16:26:18 +08:00
/// </summary>
2022-11-24 13:32:20 +08:00
public partial class PlayfieldCoveringWrapper : CompositeDrawable
2020-07-16 16:26:18 +08:00
{
2024-02-06 22:32:54 +08:00
/// <summary>
/// The relative area that should be completely covered. This does not include the fade.
/// </summary>
public readonly BindableFloat Coverage = new BindableFloat();
2020-07-16 16:26:18 +08:00
/// <summary>
/// The complete cover, including gradient and fill.
/// </summary>
private readonly Drawable cover;
2020-07-16 16:26:18 +08:00
/// <summary>
/// The gradient portion of the cover.
/// </summary>
private readonly Box gradient;
/// <summary>
/// The fully-opaque portion of the cover.
/// </summary>
private readonly Box filled;
private readonly IBindable<ScrollingDirection> scrollDirection = new Bindable<ScrollingDirection>();
2024-02-15 21:05:25 +08:00
private float currentCoverageHeight;
2024-02-07 23:20:32 +08:00
2020-07-16 20:18:24 +08:00
public PlayfieldCoveringWrapper(Drawable content)
2020-07-16 16:26:18 +08:00
{
InternalChild = new BufferedContainer
{
RelativeSizeAxes = Axes.Both,
Children = new[]
{
2020-07-16 20:17:51 +08:00
content,
cover = new Container
2020-07-16 16:26:18 +08:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Both,
Blending = new BlendingParameters
{
// Don't change the destination colour.
RGBEquation = BlendingEquation.Add,
Source = BlendingType.Zero,
Destination = BlendingType.One,
// Subtract the cover's alpha from the destination (points with alpha 1 should make the destination completely transparent).
AlphaEquation = BlendingEquation.Add,
SourceAlpha = BlendingType.Zero,
DestinationAlpha = BlendingType.OneMinusSrcAlpha
},
Children = new Drawable[]
{
gradient = new Box
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
RelativeSizeAxes = Axes.Both,
RelativePositionAxes = Axes.Both,
Height = 0.25f,
Colour = ColourInfo.GradientVertical(
Color4.White.Opacity(0f),
Color4.White.Opacity(1f)
)
},
filled = new Box
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
2020-07-16 16:35:00 +08:00
RelativeSizeAxes = Axes.Both,
Height = 0
2020-07-16 16:26:18 +08:00
}
}
}
}
};
}
[BackgroundDependencyLoader]
private void load(IScrollingInfo scrollingInfo)
{
scrollDirection.BindTo(scrollingInfo.Direction);
scrollDirection.BindValueChanged(onScrollDirectionChanged, true);
}
2024-02-06 22:32:54 +08:00
protected override void LoadComplete()
2020-07-16 16:26:18 +08:00
{
2024-02-06 22:32:54 +08:00
base.LoadComplete();
2024-02-15 21:05:25 +08:00
updateCoverSize(true);
}
protected override void Update()
{
base.Update();
2024-02-15 21:05:25 +08:00
updateCoverSize(false);
}
2024-02-15 21:05:25 +08:00
private void updateCoverSize(bool instant)
{
2024-02-15 21:05:25 +08:00
float targetCoverage;
float targetAlpha;
if (instant)
{
targetCoverage = Coverage.Value;
targetAlpha = Coverage.Value > 0 ? 1 : 0;
}
else
{
targetCoverage = (float)Interpolation.DampContinuously(currentCoverageHeight, Coverage.Value, 25, Math.Abs(Time.Elapsed));
targetAlpha = (float)Interpolation.DampContinuously(gradient.Alpha, Coverage.Value > 0 ? 1 : 0, 25, Math.Abs(Time.Elapsed));
}
filled.Height = GetHeight(targetCoverage);
gradient.Y = -GetHeight(targetCoverage);
gradient.Alpha = targetAlpha;
2024-02-07 23:20:32 +08:00
2024-02-15 21:05:25 +08:00
currentCoverageHeight = targetCoverage;
2020-07-16 16:26:18 +08:00
}
2024-02-07 23:20:32 +08:00
protected virtual float GetHeight(float coverage) => coverage;
2024-02-06 22:32:54 +08:00
private void onScrollDirectionChanged(ValueChangedEvent<ScrollingDirection> direction)
=> cover.Rotation = direction.NewValue == ScrollingDirection.Up ? 0 : 180f;
/// <summary>
/// The direction in which the cover expands.
/// </summary>
public CoverExpandDirection Direction
{
set => cover.Scale = value == CoverExpandDirection.AlongScroll ? Vector2.One : new Vector2(1, -1);
}
}
public enum CoverExpandDirection
{
/// <summary>
/// The cover expands along the scrolling direction.
/// </summary>
2024-02-15 23:40:58 +08:00
[Description("Along scroll")]
AlongScroll,
/// <summary>
/// The cover expands against the scrolling direction.
/// </summary>
2024-02-15 23:40:58 +08:00
[Description("Against scroll")]
AgainstScroll
2020-07-16 16:26:18 +08:00
}
}