1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 06:32:55 +08:00
osu-lazer/osu.Game/Screens/Menu/LogoVisualisation.cs

224 lines
8.8 KiB
C#
Raw Normal View History

2017-06-19 08:33:50 +08:00
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using OpenTK;
using OpenTK.Graphics;
using OpenTK.Graphics.ES30;
using osu.Framework.Allocation;
using osu.Framework.Configuration;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Batches;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.OpenGL.Vertices;
using osu.Framework.Graphics.Primitives;
using osu.Framework.Graphics.Shaders;
using osu.Framework.Graphics.Textures;
using osu.Game.Beatmaps;
using osu.Game.Graphics;
using System;
namespace osu.Game.Screens.Menu
{
internal class LogoVisualisation : Drawable, IHasAccentColour
{
2017-06-19 11:01:07 +08:00
private readonly Bindable<WorkingBeatmap> beatmap = new Bindable<WorkingBeatmap>();
2017-06-19 08:33:50 +08:00
2017-06-19 17:41:11 +08:00
/// <summary>
/// The number of bars to jump each update iteration.
/// </summary>
2017-06-19 08:33:50 +08:00
private const int index_change = 5;
2017-06-19 17:41:11 +08:00
/// <summary>
/// The maximum length of each bar in the visualiser. Will be reduced when kiai is not activated.
/// </summary>
private const float bar_length = 600;
2017-06-19 17:41:11 +08:00
/// <summary>
/// The number of bars in one rotation of the visualiser.
/// </summary>
2017-06-19 17:41:35 +08:00
private const int bars_per_visualiser = 200;
2017-06-19 17:41:11 +08:00
/// <summary>
/// How many times we should stretch around the circumference (overlapping overselves).
/// </summary>
private const float visualiser_rounds = 5;
2017-06-19 08:33:50 +08:00
/// <summary>
/// How much should each bar go down each milisecond (based on a full bar).
2017-06-19 08:33:50 +08:00
/// </summary>
private const float decay_per_milisecond = 0.0024f;
2017-06-19 17:41:11 +08:00
/// <summary>
/// Number of milliseconds between each amplitude update.
/// </summary>
private const float time_between_updates = 50;
/// <summary>
/// The minimum amplitude to show a bar.
/// </summary>
2017-06-22 07:40:35 +08:00
private const float amplitude_dead_zone = 1f / bar_length;
2017-06-19 08:33:50 +08:00
private int indexOffset;
2017-06-19 17:38:39 +08:00
public Color4 AccentColour { get; set; }
2017-06-19 08:33:50 +08:00
2017-06-19 11:01:07 +08:00
private readonly float[] frequencyAmplitudes = new float[256];
2017-06-19 08:33:50 +08:00
public override bool HandleInput => false;
private Shader shader;
private readonly Texture texture;
public LogoVisualisation()
{
texture = Texture.WhitePixel;
AccentColour = new Color4(1, 1, 1, 0.2f);
BlendingMode = BlendingMode.Additive;
2017-06-19 08:33:50 +08:00
}
[BackgroundDependencyLoader(true)]
2017-06-19 08:33:50 +08:00
private void load(ShaderManager shaders, OsuGame game)
{
if (game?.Beatmap != null)
beatmap.BindTo(game.Beatmap);
2017-06-19 08:33:50 +08:00
shader = shaders?.Load(VertexShaderDescriptor.TEXTURE_2, FragmentShaderDescriptor.TEXTURE_ROUNDED);
}
2017-06-19 17:41:11 +08:00
private void updateAmplitudes()
2017-06-19 08:33:50 +08:00
{
2017-06-19 17:29:44 +08:00
float[] temporalAmplitudes = beatmap.Value?.Track?.CurrentAmplitudes.FrequencyAmplitudes ?? new float[256];
var effect = beatmap.Value?.Beatmap.ControlPointInfo.EffectPointAt(beatmap.Value.Track?.CurrentTime ?? Time.Current);
2017-06-19 08:33:50 +08:00
2017-06-19 17:41:35 +08:00
for (int i = 0; i < bars_per_visualiser; i++)
2017-06-19 08:33:50 +08:00
{
if (beatmap?.Value?.Track?.IsRunning ?? false)
{
float targetAmplitude = temporalAmplitudes[(i + indexOffset) % bars_per_visualiser] * (effect?.KiaiMode == true ? 1 : 0.5f);
if (targetAmplitude > frequencyAmplitudes[i])
frequencyAmplitudes[i] = targetAmplitude;
2017-06-19 08:33:50 +08:00
}
else
{
int index = (i + index_change) % bars_per_visualiser;
if (frequencyAmplitudes[index] > frequencyAmplitudes[i])
frequencyAmplitudes[i] = frequencyAmplitudes[index];
2017-06-19 08:33:50 +08:00
}
}
2017-06-19 17:41:11 +08:00
2017-06-19 17:41:35 +08:00
indexOffset = (indexOffset + index_change) % bars_per_visualiser;
2017-06-19 17:41:11 +08:00
Scheduler.AddDelayed(updateAmplitudes, time_between_updates);
2017-06-19 08:33:50 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
2017-06-19 17:41:11 +08:00
updateAmplitudes();
2017-06-19 08:33:50 +08:00
}
protected override void Update()
{
base.Update();
float decayFactor = (float)Time.Elapsed * decay_per_milisecond;
2017-06-19 17:41:35 +08:00
for (int i = 0; i < bars_per_visualiser; i++)
2017-06-19 08:33:50 +08:00
{
//3% of extra bar length to make it a little faster when bar is almost at it's minimum
2017-06-19 11:01:07 +08:00
frequencyAmplitudes[i] -= decayFactor * (frequencyAmplitudes[i] + 0.03f);
if (frequencyAmplitudes[i] < 0)
frequencyAmplitudes[i] = 0;
2017-06-19 08:33:50 +08:00
}
Invalidate(Invalidation.DrawNode, shallPropagate: false);
}
protected override DrawNode CreateDrawNode() => new VisualisationDrawNode();
2017-06-19 17:41:35 +08:00
private readonly VisualiserSharedData sharedData = new VisualiserSharedData();
2017-06-19 08:33:50 +08:00
protected override void ApplyDrawNode(DrawNode node)
{
base.ApplyDrawNode(node);
var visNode = (VisualisationDrawNode)node;
visNode.Shader = shader;
visNode.Texture = texture;
visNode.Size = DrawSize.X;
visNode.Shared = sharedData;
2017-06-19 17:38:39 +08:00
visNode.Colour = AccentColour;
2017-06-19 11:01:07 +08:00
visNode.AudioData = frequencyAmplitudes;
2017-06-19 08:33:50 +08:00
}
2017-06-19 17:41:35 +08:00
private class VisualiserSharedData
2017-06-19 08:33:50 +08:00
{
public readonly LinearBatch<TexturedVertex2D> VertexBatch = new LinearBatch<TexturedVertex2D>(100 * 4, 10, PrimitiveType.Quads);
}
private class VisualisationDrawNode : DrawNode
{
public Shader Shader;
public Texture Texture;
2017-06-19 17:41:35 +08:00
public VisualiserSharedData Shared;
2017-06-19 08:33:50 +08:00
//Asuming the logo is a circle, we don't need a second dimension.
public float Size;
public Color4 Colour;
public float[] AudioData;
public override void Draw(Action<TexturedVertex2D> vertexAction)
{
base.Draw(vertexAction);
Shader.Bind();
Texture.TextureGL.Bind();
Vector2 inflation = DrawInfo.MatrixInverse.ExtractScale().Xy;
ColourInfo colourInfo = DrawInfo.Colour;
colourInfo.ApplyChild(Colour);
if (AudioData != null)
{
2017-06-19 17:41:11 +08:00
for (int j = 0; j < visualiser_rounds; j++)
2017-06-19 08:33:50 +08:00
{
2017-06-19 17:41:35 +08:00
for (int i = 0; i < bars_per_visualiser; i++)
2017-06-19 12:52:42 +08:00
{
2017-06-22 07:40:35 +08:00
if (AudioData[i] < amplitude_dead_zone)
continue;
2017-06-19 17:41:35 +08:00
float rotation = MathHelper.DegreesToRadians(i / (float)bars_per_visualiser * 360 + j * 360 / visualiser_rounds);
2017-06-19 12:52:42 +08:00
float rotationCos = (float)Math.Cos(rotation);
float rotationSin = (float)Math.Sin(rotation);
//taking the cos and sin to the 0..1 range
var barPosition = new Vector2(rotationCos / 2 + 0.5f, rotationSin / 2 + 0.5f) * Size;
var barSize = new Vector2(Size * (float)Math.Sqrt(2 * (1 - Math.Cos(MathHelper.DegreesToRadians(360f / bars_per_visualiser)))) / 2f, bar_length * AudioData[i]);
2017-06-19 12:52:42 +08:00
//The distance between the position and the sides of the bar.
var bottomOffset = new Vector2(-rotationSin * barSize.X / 2, rotationCos * barSize.X / 2);
//The distance between the bottom side of the bar and the top side.
var amplitudeOffset = new Vector2(rotationCos * barSize.Y, rotationSin * barSize.Y);
var rectangle = new Quad(
(barPosition - bottomOffset) * DrawInfo.Matrix,
(barPosition - bottomOffset + amplitudeOffset) * DrawInfo.Matrix,
(barPosition + bottomOffset) * DrawInfo.Matrix,
(barPosition + bottomOffset + amplitudeOffset) * DrawInfo.Matrix
);
Texture.DrawQuad(
rectangle,
colourInfo,
null,
Shared.VertexBatch.Add,
//barSize by itself will make it smooth more in the X axis than in the Y axis, this reverts that.
Vector2.Divide(inflation, barSize.Yx));
}
2017-06-19 08:33:50 +08:00
}
}
Shader.Unbind();
}
}
}
}