// 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 Microsoft.EntityFrameworkCore.Internal; using osu.Framework.Allocation; using osu.Framework.Audio; using osu.Framework.Audio.Sample; using osu.Framework.Configuration; using osu.Framework.Graphics; using osu.Framework.Input.Bindings; using osu.Framework.Screens; using osu.Game.Beatmaps; using osu.Game.Graphics.Containers; using osu.Game.Input.Bindings; using osu.Game.Rulesets; using osu.Game.Screens.Menu; using osuTK; using osu.Game.Overlays; using osu.Framework.Graphics.Containers; namespace osu.Game.Screens { public abstract class OsuScreen : Screen, IKeyBindingHandler, IHasDescription { /// /// The amount of negative padding that should be applied to game background content which touches both the left and right sides of the screen. /// This allows for the game content to be pushed byt he options/notification overlays without causing black areas to appear. /// public const float HORIZONTAL_OVERFLOW_PADDING = 50; public BackgroundScreen Background { get; private set; } /// /// A user-facing title for this screen. /// public virtual string Title => GetType().ShortDisplayName(); public string Description => Title; protected virtual bool AllowBackButton => true; public virtual bool AllowExternalScreenChange => false; /// /// Override to create a BackgroundMode for the current screen. /// Note that the instance created may not be the used instance if it matches the BackgroundMode equality clause. /// protected virtual BackgroundScreen CreateBackground() => null; private Action updateOverlayStates; /// /// Whether all overlays should be hidden when this screen is entered or resumed. /// protected virtual bool HideOverlaysOnEnter => false; protected readonly Bindable OverlayActivationMode = new Bindable(); /// /// Whether overlays should be able to be opened once this screen is entered or resumed. /// protected virtual OverlayActivation InitialOverlayActivationMode => OverlayActivation.All; /// /// Whether this allows the cursor to be displayed. /// public virtual bool CursorVisible => true; protected new OsuGameBase Game => base.Game as OsuGameBase; private OsuLogo logo; /// /// Whether the beatmap or ruleset should be allowed to be changed by the user or game. /// Used to mark exclusive areas where this is strongly prohibited, like gameplay. /// public virtual bool AllowBeatmapRulesetChange => true; protected readonly Bindable Beatmap = new Bindable(); protected virtual float BackgroundParallaxAmount => 1; private ParallaxContainer backgroundParallaxContainer; protected readonly Bindable Ruleset = new Bindable(); private SampleChannel sampleExit; [BackgroundDependencyLoader(true)] private void load(BindableBeatmap beatmap, OsuGame osu, AudioManager audio, Bindable ruleset) { Beatmap.BindTo(beatmap); Ruleset.BindTo(ruleset); if (osu != null) { OverlayActivationMode.BindTo(osu.OverlayActivationMode); updateOverlayStates = () => { if (HideOverlaysOnEnter) osu.CloseAllOverlays(); else osu.Toolbar.State = Visibility.Visible; }; } sampleExit = audio.Sample.Get(@"UI/screen-back"); } public virtual bool OnPressed(GlobalAction action) { if (!IsCurrentScreen) return false; if (action == GlobalAction.Back && AllowBackButton) { Exit(); return true; } return false; } public bool OnReleased(GlobalAction action) => action == GlobalAction.Back && AllowBackButton; protected override void OnResuming(Screen last) { sampleExit?.Play(); applyArrivingDefaults(true); base.OnResuming(last); } protected override void OnSuspending(Screen next) { base.OnSuspending(next); onSuspendingLogo(); } protected override void OnEntering(Screen last) { OsuScreen lastOsu = last as OsuScreen; BackgroundScreen bg = CreateBackground(); if (lastOsu?.Background != null) { backgroundParallaxContainer = lastOsu.backgroundParallaxContainer; if (bg == null || lastOsu.Background.Equals(bg)) //we can keep the previous mode's background. Background = lastOsu.Background; else { lastOsu.Background.Push(Background = bg); } } else if (bg != null) { // this makes up for the fact our padding changes when the global toolbar is visible. bg.Scale = new Vector2(1.06f); AddInternal(backgroundParallaxContainer = new ParallaxContainer { Depth = float.MaxValue, Children = new[] { Background = bg } }); } if ((logo = lastOsu?.logo) == null) LoadComponentAsync(logo = new OsuLogo { Alpha = 0 }, AddInternal); applyArrivingDefaults(false); base.OnEntering(last); } protected override bool OnExiting(Screen next) { if (ValidForResume && logo != null) onExitingLogo(); OsuScreen nextOsu = next as OsuScreen; if (Background != null && !Background.Equals(nextOsu?.Background)) { Background.Exit(); //We need to use MakeCurrent in case we are jumping up multiple game screens. nextOsu?.Background?.MakeCurrent(); } if (base.OnExiting(next)) return true; Beatmap.UnbindAll(); return false; } /// /// Fired when this screen was entered or resumed and the logo state is required to be adjusted. /// protected virtual void LogoArriving(OsuLogo logo, bool resuming) { logo.Action = null; logo.FadeOut(300, Easing.OutQuint); logo.Anchor = Anchor.TopLeft; logo.Origin = Anchor.Centre; logo.RelativePositionAxes = Axes.None; logo.BeatMatching = true; logo.Triangles = true; logo.Ripple = true; } private void applyArrivingDefaults(bool isResuming) { logo.AppendAnimatingAction(() => { if (IsCurrentScreen) LogoArriving(logo, isResuming); }, true); if (backgroundParallaxContainer != null) backgroundParallaxContainer.ParallaxAmount = ParallaxContainer.DEFAULT_PARALLAX_AMOUNT * BackgroundParallaxAmount; OverlayActivationMode.Value = InitialOverlayActivationMode; updateOverlayStates?.Invoke(); } private void onExitingLogo() { logo.AppendAnimatingAction(() => { LogoExiting(logo); }, false); } /// /// Fired when this screen was exited to add any outwards transition to the logo. /// protected virtual void LogoExiting(OsuLogo logo) { } private void onSuspendingLogo() { logo.AppendAnimatingAction(() => { LogoSuspending(logo); }, false); } /// /// Fired when this screen was suspended to add any outwards transition to the logo. /// protected virtual void LogoSuspending(OsuLogo logo) { } } }