osu/osu.Game/OsuGame.cs

407 lines
13 KiB
C#
Raw Normal View History

// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2016-08-26 03:28:23 +00:00
using System;
using osu.Framework.Configuration;
2017-02-17 09:59:30 +00:00
using osu.Framework.Screens;
2016-08-26 03:28:23 +00:00
using osu.Game.Configuration;
2016-09-01 10:06:09 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2016-10-01 09:01:52 +00:00
using osu.Game.Overlays;
using osu.Framework.Logging;
using osu.Game.Graphics.UserInterface.Volume;
2016-11-08 23:13:20 +00:00
using osu.Framework.Allocation;
2016-12-01 05:22:29 +00:00
using osu.Game.Overlays.Toolbar;
2016-11-14 08:23:33 +00:00
using osu.Game.Screens;
using osu.Game.Screens.Menu;
using OpenTK;
2017-02-04 21:03:39 +00:00
using System.Linq;
2017-02-24 09:10:37 +00:00
using System.Threading.Tasks;
2017-08-11 07:11:46 +00:00
using osu.Framework.Input.Bindings;
using osu.Framework.Platform;
using osu.Framework.Threading;
2017-03-04 10:02:36 +00:00
using osu.Game.Graphics;
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Overlays.Notifications;
2017-07-26 04:22:46 +00:00
using osu.Game.Rulesets;
2017-03-04 10:02:36 +00:00
using osu.Game.Screens.Play;
2017-08-11 07:11:46 +00:00
using osu.Game.Input.Bindings;
2016-08-26 03:28:23 +00:00
namespace osu.Game
{
2017-08-12 10:54:07 +00:00
public class OsuGame : OsuGameBase, IKeyBindingHandler<GlobalAction>
2016-08-26 03:28:23 +00:00
{
2016-10-01 09:01:52 +00:00
public Toolbar Toolbar;
private ChatOverlay chat;
private MusicController musicController;
private NotificationOverlay notificationOverlay;
2017-02-10 07:26:43 +00:00
private DialogOverlay dialogOverlay;
private DirectOverlay direct;
2017-05-26 03:58:18 +00:00
private SocialOverlay social;
2017-06-15 09:03:33 +00:00
private UserProfileOverlay userProfile;
public virtual Storage GetStorageForStableInstall() => null;
private Intro intro
{
get
{
Screen s = screenStack;
while (s != null && !(s is Intro))
s = s.ChildScreen;
return s as Intro;
}
}
public float ToolbarOffset => Toolbar.Position.Y + Toolbar.DrawHeight;
2017-02-17 09:59:30 +00:00
private OsuScreen screenStack;
private VolumeControl volume;
2017-04-14 20:52:46 +00:00
private Bindable<int> configRuleset;
2017-04-17 08:43:48 +00:00
public Bindable<RulesetInfo> Ruleset = new Bindable<RulesetInfo>();
2016-10-13 14:21:15 +00:00
private readonly string[] args;
private SettingsOverlay settings;
2016-11-08 10:26:12 +00:00
public OsuGame(string[] args = null)
{
this.args = args;
}
2016-10-01 09:01:52 +00:00
public void ToggleSettings() => settings.ToggleVisibility();
public void ToggleDirect() => direct.ToggleVisibility();
private DependencyContainer dependencies;
protected override IReadOnlyDependencyContainer CreateLocalDependencies(IReadOnlyDependencyContainer parent) =>
dependencies = new DependencyContainer(base.CreateLocalDependencies(parent));
[BackgroundDependencyLoader]
private void load(FrameworkConfigManager frameworkConfig)
2016-08-26 03:28:23 +00:00
{
this.frameworkConfig = frameworkConfig;
if (!Host.IsPrimaryInstance)
{
Logger.Log(@"osu! does not support multiple running instances.", LoggingTarget.Runtime, LogLevel.Error);
Environment.Exit(0);
}
if (args?.Length > 0)
{
var paths = args.Where(a => !a.StartsWith(@"-"));
Task.Run(() => BeatmapManager.Import(paths.ToArray()));
}
dependencies.Cache(this);
2016-11-10 22:40:42 +00:00
2017-05-15 01:56:27 +00:00
configRuleset = LocalConfig.GetBindable<int>(OsuSetting.Ruleset);
Ruleset.Value = RulesetStore.GetRuleset(configRuleset.Value);
2017-04-17 10:44:03 +00:00
Ruleset.ValueChanged += r => configRuleset.Value = r.ID ?? 0;
}
private ScheduledDelegate scoreLoad;
2017-03-04 10:02:36 +00:00
protected void LoadScore(Score s)
{
scoreLoad?.Cancel();
2017-03-04 10:02:36 +00:00
var menu = intro.ChildScreen;
if (menu == null)
{
scoreLoad = Schedule(() => LoadScore(s));
2017-03-04 10:02:36 +00:00
return;
}
if (!menu.IsCurrentScreen)
{
menu.MakeCurrent();
this.Delay(500).Schedule(() => LoadScore(s), out scoreLoad);
2017-03-04 10:02:36 +00:00
return;
}
if (s.Beatmap == null)
{
notificationOverlay.Post(new SimpleNotification
2017-03-04 10:02:36 +00:00
{
Text = @"Tried to load a score for a beatmap we don't have!",
Icon = FontAwesome.fa_life_saver,
});
return;
}
Beatmap.Value = BeatmapManager.GetWorkingBeatmap(s.Beatmap);
2017-03-04 10:02:36 +00:00
menu.Push(new PlayerLoader(new ReplayPlayer(s.Replay)));
2017-03-04 10:02:36 +00:00
}
protected override void LoadComplete()
{
base.LoadComplete();
2016-11-01 14:24:14 +00:00
// hook up notifications to components.
BeatmapManager.PostNotification = n => notificationOverlay?.Post(n);
BeatmapManager.GetStableStorage = GetStorageForStableInstall;
2017-07-11 13:58:06 +00:00
AddRange(new Drawable[] {
new VolumeControlReceptor
{
RelativeSizeAxes = Axes.Both,
ActionRequested = action => volume.Adjust(action)
},
2016-11-01 14:24:14 +00:00
mainContent = new Container
{
2016-11-01 14:24:14 +00:00
RelativeSizeAxes = Axes.Both,
},
2016-11-23 11:26:46 +00:00
volume = new VolumeControl(),
2017-06-07 11:53:37 +00:00
overlayContent = new Container { RelativeSizeAxes = Axes.Both },
2017-05-16 03:53:50 +00:00
new OnScreenDisplay(),
2016-09-30 09:45:55 +00:00
});
2017-04-02 06:56:12 +00:00
LoadComponentAsync(screenStack = new Loader(), d =>
2016-11-01 14:24:14 +00:00
{
2017-02-17 09:59:30 +00:00
screenStack.ModePushed += screenAdded;
screenStack.Exited += screenRemoved;
mainContent.Add(screenStack);
2016-11-01 14:24:14 +00:00
});
//overlay elements
LoadComponentAsync(direct = new DirectOverlay { Depth = -1 }, mainContent.Add);
2017-05-26 03:58:18 +00:00
LoadComponentAsync(social = new SocialOverlay { Depth = -1 }, mainContent.Add);
LoadComponentAsync(chat = new ChatOverlay { Depth = -1 }, mainContent.Add);
LoadComponentAsync(settings = new MainSettings
{
GetToolbarHeight = () => ToolbarOffset,
Depth = -1
}, overlayContent.Add);
LoadComponentAsync(userProfile = new UserProfileOverlay { Depth = -2 }, mainContent.Add);
2017-04-02 06:56:12 +00:00
LoadComponentAsync(musicController = new MusicController
{
Depth = -3,
Position = new Vector2(0, Toolbar.HEIGHT),
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
2017-04-02 06:56:12 +00:00
}, overlayContent.Add);
LoadComponentAsync(notificationOverlay = new NotificationOverlay
2017-02-10 07:26:43 +00:00
{
Depth = -3,
2017-02-10 07:26:43 +00:00
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
2017-04-02 06:56:12 +00:00
}, overlayContent.Add);
2017-02-10 07:26:43 +00:00
2017-04-02 06:56:12 +00:00
LoadComponentAsync(dialogOverlay = new DialogOverlay
{
Depth = -5,
2017-04-02 06:56:12 +00:00
}, overlayContent.Add);
Logger.NewEntry += entry =>
{
if (entry.Level < LogLevel.Important) return;
notificationOverlay.Post(new SimpleNotification
{
Text = $@"{entry.Level}: {entry.Message}"
});
};
dependencies.Cache(settings);
dependencies.Cache(social);
2017-08-24 11:18:47 +00:00
dependencies.Cache(direct);
dependencies.Cache(chat);
dependencies.Cache(userProfile);
dependencies.Cache(musicController);
dependencies.Cache(notificationOverlay);
dependencies.Cache(dialogOverlay);
// ensure only one of these overlays are open at once.
var singleDisplayOverlays = new OverlayContainer[] { chat, social, direct };
foreach (var overlay in singleDisplayOverlays)
{
2017-09-04 00:10:04 +00:00
overlay.StateChanged += state =>
{
if (state == Visibility.Hidden) return;
foreach (var c in singleDisplayOverlays)
{
2017-09-04 00:10:04 +00:00
if (c == overlay) continue;
c.State = Visibility.Hidden;
}
};
}
2017-04-02 06:56:12 +00:00
LoadComponentAsync(Toolbar = new Toolbar
2016-11-01 14:24:14 +00:00
{
Depth = -4,
OnHome = delegate
{
hideAllOverlays();
intro?.ChildScreen?.MakeCurrent();
},
}, overlayContent.Add);
settings.StateChanged += delegate
{
switch (settings.State)
{
case Visibility.Hidden:
2017-07-22 18:50:25 +00:00
intro.MoveToX(0, SettingsOverlay.TRANSITION_LENGTH, Easing.OutQuint);
break;
case Visibility.Visible:
2017-07-22 18:50:25 +00:00
intro.MoveToX(SettingsOverlay.SIDEBAR_WIDTH / 2, SettingsOverlay.TRANSITION_LENGTH, Easing.OutQuint);
break;
}
};
2017-03-16 14:58:36 +00:00
Cursor.State = Visibility.Hidden;
}
2017-08-10 10:52:45 +00:00
public bool OnPressed(GlobalAction action)
{
if (intro == null) return false;
switch (action)
{
case GlobalAction.ToggleChat:
chat.ToggleVisibility();
return true;
case GlobalAction.ToggleSocial:
social.ToggleVisibility();
return true;
case GlobalAction.ResetInputSettings:
var sensitivity = frameworkConfig.GetBindable<double>(FrameworkSetting.CursorSensitivity);
sensitivity.Disabled = false;
sensitivity.Value = 1;
sensitivity.Disabled = true;
frameworkConfig.Set(FrameworkSetting.ActiveInputHandlers, string.Empty);
return true;
case GlobalAction.ToggleToolbar:
Toolbar.ToggleVisibility();
return true;
case GlobalAction.ToggleSettings:
settings.ToggleVisibility();
return true;
case GlobalAction.ToggleDirect:
direct.ToggleVisibility();
return true;
2016-11-08 10:27:37 +00:00
}
2017-02-27 15:18:12 +00:00
return false;
}
2017-08-10 10:52:45 +00:00
public bool OnReleased(GlobalAction action) => false;
2017-03-13 23:22:46 +00:00
public event Action<Screen> ScreenChanged;
2016-11-01 14:24:14 +00:00
private Container mainContent;
private Container overlayContent;
2017-03-16 14:58:36 +00:00
private OsuScreen currentScreen;
private FrameworkConfigManager frameworkConfig;
2017-03-16 14:58:36 +00:00
private void hideAllOverlays()
{
settings.State = Visibility.Hidden;
chat.State = Visibility.Hidden;
direct.State = Visibility.Hidden;
social.State = Visibility.Hidden;
userProfile.State = Visibility.Hidden;
notificationOverlay.State = Visibility.Hidden;
}
2017-03-13 23:22:46 +00:00
private void screenChanged(Screen newScreen)
{
2017-03-16 14:58:36 +00:00
currentScreen = newScreen as OsuScreen;
if (currentScreen == null)
{
Exit();
return;
}
2017-04-18 07:05:58 +00:00
//central game screen change logic.
2017-03-17 17:03:44 +00:00
if (!currentScreen.ShowOverlays)
2016-10-06 14:32:56 +00:00
{
hideAllOverlays();
musicController.State = Visibility.Hidden;
Toolbar.State = Visibility.Hidden;
2016-10-06 14:32:56 +00:00
}
else
2016-10-13 14:21:15 +00:00
Toolbar.State = Visibility.Visible;
2017-03-13 23:22:46 +00:00
ScreenChanged?.Invoke(newScreen);
}
protected override bool OnExiting()
{
2017-02-18 05:16:46 +00:00
if (screenStack.ChildScreen == null) return false;
if (intro == null) return true;
2017-02-17 09:59:30 +00:00
if (!intro.DidLoadMenu || intro.ChildScreen != null)
{
2017-02-17 06:33:08 +00:00
Scheduler.Add(intro.MakeCurrent);
return true;
}
2016-10-13 14:21:15 +00:00
return base.OnExiting();
}
/// <summary>
/// Use to programatically exit the game as if the user was triggering via alt-f4.
/// Will keep persisting until an exit occurs (exit may be blocked multiple times).
/// </summary>
public void GracefullyExit()
{
if (!OnExiting())
Exit();
else
Scheduler.AddDelayed(GracefullyExit, 2000);
}
2017-02-08 10:32:55 +00:00
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
// we only want to apply these restrictions when we are inside a screen stack.
// the use case for not applying is in visual/unit tests.
2017-08-22 08:21:19 +00:00
bool applyRestrictions = !currentScreen?.AllowBeatmapRulesetChange ?? false;
Ruleset.Disabled = applyRestrictions;
Beatmap.Disabled = applyRestrictions;
mainContent.Padding = new MarginPadding { Top = ToolbarOffset };
2017-03-16 14:58:36 +00:00
Cursor.State = currentScreen?.HasLocalCursorDisplayed == false ? Visibility.Visible : Visibility.Hidden;
2017-02-08 10:32:55 +00:00
}
2017-02-17 09:59:30 +00:00
private void screenAdded(Screen newScreen)
{
2017-02-17 09:59:30 +00:00
newScreen.ModePushed += screenAdded;
newScreen.Exited += screenRemoved;
2017-03-13 23:22:46 +00:00
screenChanged(newScreen);
}
2017-02-17 09:59:30 +00:00
private void screenRemoved(Screen newScreen)
{
2017-03-13 23:22:46 +00:00
screenChanged(newScreen);
2016-08-26 03:28:23 +00:00
}
}
}