// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Collections.Generic; using System.IO; using osu.Framework.Allocation; using osu.Framework.Audio; using osu.Framework.Audio.Sample; using osu.Framework.Audio.Track; using osu.Framework.Bindables; using osu.Framework.Screens; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Shapes; using osu.Framework.Graphics.Textures; using osu.Framework.Graphics.Video; using osu.Framework.MathUtils; using osu.Framework.Timing; using osu.Game.Beatmaps; using osu.Game.Configuration; using osu.Game.Graphics; using osu.Game.Graphics.Containers; using osu.Game.Graphics.Sprites; using osu.Game.IO.Archives; using osu.Game.Rulesets; using osu.Game.Screens.Backgrounds; using osuTK; using osuTK.Graphics; namespace osu.Game.Screens.Menu { public class IntroTriangles : IntroScreen { private const string menu_music_beatmap_hash = "a1556d0801b3a6b175dda32ef546f0ec812b400499f575c44fccbe9c67f9b1e5"; private SampleChannel welcome; protected override BackgroundScreen CreateBackground() => background = new BackgroundScreenDefault(false) { Alpha = 0, }; [Resolved] private AudioManager audio { get; set; } private Bindable menuMusic; private Track track; private WorkingBeatmap introBeatmap; private BackgroundScreenDefault background; [BackgroundDependencyLoader] private void load(OsuConfigManager config, BeatmapManager beatmaps, Framework.Game game) { menuMusic = config.GetBindable(OsuSetting.MenuMusic); BeatmapSetInfo setInfo = null; if (!menuMusic.Value) { var sets = beatmaps.GetAllUsableBeatmapSets(); if (sets.Count > 0) setInfo = beatmaps.QueryBeatmapSet(s => s.ID == sets[RNG.Next(0, sets.Count - 1)].ID); } if (setInfo == null) { setInfo = beatmaps.QueryBeatmapSet(b => b.Hash == menu_music_beatmap_hash); if (setInfo == null) { // we need to import the default menu background beatmap setInfo = beatmaps.Import(new ZipArchiveReader(game.Resources.GetStream(@"Tracks/triangles.osz"), "triangles.osz")).Result; setInfo.Protected = true; beatmaps.Update(setInfo); } } introBeatmap = beatmaps.GetWorkingBeatmap(setInfo.Beatmaps[0]); track = introBeatmap.Track; track.Reset(); if (config.Get(OsuSetting.MenuVoice) && !menuMusic.Value) // triangles has welcome sound included in the track. only play this if the user doesn't want menu music. welcome = audio.Samples.Get(@"welcome"); } protected override void LogoArriving(OsuLogo logo, bool resuming) { base.LogoArriving(logo, resuming); logo.Triangles = true; if (!resuming) { Beatmap.Value = introBeatmap; introBeatmap = null; PrepareMenuLoad(); LoadComponentAsync(new TrianglesIntroSequence(logo, background) { RelativeSizeAxes = Axes.Both, Clock = new FramedClock(menuMusic.Value ? track : null), LoadMenu = LoadMenu }, t => { AddInternal(t); welcome?.Play(); // Only start the current track if it is the menu music. A beatmap's track is started when entering the Main Menu. if (menuMusic.Value) track.Start(); }); } } public override void OnResuming(IScreen last) { base.OnResuming(last); background.FadeOut(100); } public override void OnSuspending(IScreen next) { track = null; base.OnSuspending(next); } private class TrianglesIntroSequence : CompositeDrawable { private readonly OsuLogo logo; private readonly BackgroundScreenDefault background; private OsuSpriteText welcomeText; private RulesetFlow rulesets; private Container rulesetsScale; private Drawable logoContainerSecondary; private Drawable logoContainer; private GlitchingTriangles triangles; public Action LoadMenu; public TrianglesIntroSequence(OsuLogo logo, BackgroundScreenDefault background) { this.logo = logo; this.background = background; } private OsuGameBase game; [BackgroundDependencyLoader] private void load(TextureStore textures, OsuGameBase game) { this.game = game; InternalChildren = new[] { triangles = new GlitchingTriangles { Alpha = 0, Anchor = Anchor.Centre, Origin = Anchor.Centre, Size = new Vector2(0.4f, 0.16f) }, welcomeText = new OsuSpriteText { Anchor = Anchor.Centre, Origin = Anchor.Centre, Padding = new MarginPadding { Bottom = 10 }, Font = OsuFont.GetFont(weight: FontWeight.Light, size: 42), Alpha = 1, Spacing = new Vector2(5), }, rulesetsScale = new Container { RelativeSizeAxes = Axes.Both, Anchor = Anchor.Centre, Origin = Anchor.Centre, Children = new Drawable[] { rulesets = new RulesetFlow() } }, logoContainerSecondary = new Container { RelativeSizeAxes = Axes.Both, Anchor = Anchor.Centre, Origin = Anchor.Centre, Child = logoContainer = new LazerLogo(textures.GetStream("Menu/logo-triangles.mp4")) { Anchor = Anchor.Centre, Origin = Anchor.Centre, } }, }; } private const double text_1 = 200; private const double text_2 = 400; private const double text_3 = 700; private const double text_4 = 900; private const double text_glitch = 1060; private const double rulesets_1 = 1450; private const double rulesets_2 = 1650; private const double rulesets_3 = 1850; private const double logo_scale_duration = 920; private const double logo_1 = 2080; private const double logo_2 = logo_1 + logo_scale_duration; protected override void LoadComplete() { base.LoadComplete(); const float scale_start = 1.2f; const float scale_adjust = 0.8f; rulesets.Hide(); logoContainer.Hide(); background.Hide(); using (BeginAbsoluteSequence(0, true)) { using (BeginDelayedSequence(text_1, true)) welcomeText.FadeIn().OnComplete(t => t.Text = "wel"); using (BeginDelayedSequence(text_2, true)) welcomeText.FadeIn().OnComplete(t => t.Text = "welcome"); using (BeginDelayedSequence(text_3, true)) welcomeText.FadeIn().OnComplete(t => t.Text = "welcome to"); using (BeginDelayedSequence(text_4, true)) { welcomeText.FadeIn().OnComplete(t => t.Text = "welcome to osu!"); welcomeText.TransformTo(nameof(welcomeText.Spacing), new Vector2(50, 0), 5000); } using (BeginDelayedSequence(text_glitch, true)) triangles.FadeIn(); using (BeginDelayedSequence(rulesets_1, true)) { rulesetsScale.ScaleTo(0.8f, 1000); rulesets.FadeIn().ScaleTo(1).TransformSpacingTo(new Vector2(200, 0)); welcomeText.FadeOut(); triangles.FadeOut(); } using (BeginDelayedSequence(rulesets_2, true)) { rulesets.ScaleTo(2).TransformSpacingTo(new Vector2(30, 0)); } using (BeginDelayedSequence(rulesets_3, true)) { rulesets.ScaleTo(4).TransformSpacingTo(new Vector2(10, 0)); rulesetsScale.ScaleTo(1.3f, 1000); } using (BeginDelayedSequence(logo_1, true)) { rulesets.FadeOut(); // matching flyte curve y = 0.25x^2 + (max(0, x - 0.7) / 0.3) ^ 5 logoContainer.FadeIn().ScaleTo(scale_start).Then().Delay(logo_scale_duration * 0.7f).ScaleTo(scale_start - scale_adjust, logo_scale_duration * 0.3f, Easing.InQuint); logoContainerSecondary.ScaleTo(scale_start).Then().ScaleTo(scale_start - scale_adjust * 0.25f, logo_scale_duration, Easing.InQuad); } using (BeginDelayedSequence(logo_2, true)) { logoContainer.FadeOut().OnComplete(_ => { logo.FadeIn(); background.FadeIn(); game.Add(new GameWideFlash()); LoadMenu(); }); } } } private class GameWideFlash : Box { private const double flash_length = 1000; public GameWideFlash() { Colour = Color4.White; RelativeSizeAxes = Axes.Both; Blending = BlendingParameters.Additive; } protected override void LoadComplete() { base.LoadComplete(); this.FadeOutFromOne(flash_length, Easing.Out); } } private class LazerLogo : CompositeDrawable { public LazerLogo(Stream videoStream) { Size = new Vector2(960); InternalChild = new VideoSprite(videoStream) { RelativeSizeAxes = Axes.Both, Clock = new FramedOffsetClock(Clock) { Offset = -logo_1 } }; } } private class RulesetFlow : FillFlowContainer { [BackgroundDependencyLoader] private void load(RulesetStore rulesets) { var modes = new List(); foreach (var ruleset in rulesets.AvailableRulesets) { var icon = new ConstrainedIconContainer { Icon = ruleset.CreateInstance().CreateIcon(), Size = new Vector2(30), }; modes.Add(icon); } AutoSizeAxes = Axes.Both; Children = modes; Anchor = Anchor.Centre; Origin = Anchor.Centre; } } private class GlitchingTriangles : CompositeDrawable { public GlitchingTriangles() { RelativeSizeAxes = Axes.Both; } private double? lastGenTime; private const double time_between_triangles = 22; protected override void Update() { base.Update(); if (lastGenTime == null || Time.Current - lastGenTime > time_between_triangles) { lastGenTime = (lastGenTime ?? Time.Current) + time_between_triangles; Drawable triangle = new OutlineTriangle(RNG.NextBool(), (RNG.NextSingle() + 0.2f) * 80) { RelativePositionAxes = Axes.Both, Position = new Vector2(RNG.NextSingle(), RNG.NextSingle()), }; AddInternal(triangle); triangle.FadeOutFromOne(120); } } /// /// Represents a sprite that is drawn in a triangle shape, instead of a rectangle shape. /// public class OutlineTriangle : BufferedContainer { public OutlineTriangle(bool outlineOnly, float size) { Size = new Vector2(size); InternalChildren = new Drawable[] { new Triangle { RelativeSizeAxes = Axes.Both }, }; if (outlineOnly) { AddInternal(new Triangle { Anchor = Anchor.Centre, Origin = Anchor.Centre, Colour = Color4.Black, Size = new Vector2(size - 5), Blending = BlendingParameters.None, }); } Blending = BlendingParameters.Additive; CacheDrawnFrameBuffer = true; } } } } } }