2018-01-05 20:21:19 +09:00
|
|
|
|
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
|
2017-02-07 13:59:30 +09:00
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
2016-08-26 12:28:23 +09:00
|
|
|
|
|
2016-10-06 21:10:01 +09:00
|
|
|
|
using System;
|
2018-01-26 19:30:29 +09:00
|
|
|
|
using System.Collections.Generic;
|
2016-10-04 17:15:03 +09:00
|
|
|
|
using osu.Framework.Configuration;
|
2017-02-17 18:59:30 +09:00
|
|
|
|
using osu.Framework.Screens;
|
2016-08-26 12:28:23 +09:00
|
|
|
|
using osu.Game.Configuration;
|
2016-09-01 19:06:09 +09:00
|
|
|
|
using osu.Framework.Graphics;
|
2016-10-08 00:43:06 +09:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
2016-10-01 18:01:52 +09:00
|
|
|
|
using osu.Game.Overlays;
|
2016-10-14 14:09:35 +03:00
|
|
|
|
using osu.Framework.Logging;
|
2016-10-26 18:45:48 +09:00
|
|
|
|
using osu.Game.Graphics.UserInterface.Volume;
|
2016-11-08 18:13:20 -05:00
|
|
|
|
using osu.Framework.Allocation;
|
2016-12-01 14:22:29 +09:00
|
|
|
|
using osu.Game.Overlays.Toolbar;
|
2016-11-14 17:23:33 +09:00
|
|
|
|
using osu.Game.Screens;
|
|
|
|
|
using osu.Game.Screens.Menu;
|
2017-01-30 22:00:23 +09:00
|
|
|
|
using OpenTK;
|
2017-02-04 22:03:39 +01:00
|
|
|
|
using System.Linq;
|
2017-12-25 18:22:58 +09:00
|
|
|
|
using System.Threading;
|
2017-02-24 18:10:37 +09:00
|
|
|
|
using System.Threading.Tasks;
|
2018-01-31 18:11:38 +09:00
|
|
|
|
using osu.Framework.Audio;
|
2017-08-11 16:11:46 +09:00
|
|
|
|
using osu.Framework.Input.Bindings;
|
2017-08-01 15:12:12 +09:00
|
|
|
|
using osu.Framework.Platform;
|
2017-03-04 21:35:12 +09:00
|
|
|
|
using osu.Framework.Threading;
|
2017-03-04 19:02:36 +09:00
|
|
|
|
using osu.Game.Graphics;
|
2017-04-18 16:05:58 +09:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2017-02-15 12:37:57 +09:00
|
|
|
|
using osu.Game.Overlays.Notifications;
|
2017-07-26 13:22:46 +09:00
|
|
|
|
using osu.Game.Rulesets;
|
2017-03-04 19:02:36 +09:00
|
|
|
|
using osu.Game.Screens.Play;
|
2017-08-11 16:11:46 +09:00
|
|
|
|
using osu.Game.Input.Bindings;
|
2018-01-26 19:30:29 +09:00
|
|
|
|
using osu.Game.Rulesets.Mods;
|
2017-12-30 20:41:36 +09:00
|
|
|
|
using OpenTK.Graphics;
|
2016-08-26 12:28:23 +09:00
|
|
|
|
|
|
|
|
|
namespace osu.Game
|
|
|
|
|
{
|
2017-08-12 19:54:07 +09:00
|
|
|
|
public class OsuGame : OsuGameBase, IKeyBindingHandler<GlobalAction>
|
2016-08-26 12:28:23 +09:00
|
|
|
|
{
|
2016-10-01 18:01:52 +09:00
|
|
|
|
public Toolbar Toolbar;
|
2016-11-09 15:23:10 +09:00
|
|
|
|
|
2016-11-30 15:15:43 +09:00
|
|
|
|
private ChatOverlay chat;
|
2016-11-09 15:23:10 +09:00
|
|
|
|
|
2016-11-11 13:30:57 +09:00
|
|
|
|
private MusicController musicController;
|
|
|
|
|
|
2017-12-24 10:28:37 +01:00
|
|
|
|
private NotificationOverlay notifications;
|
2017-02-10 16:26:43 +09:00
|
|
|
|
|
2017-02-28 02:09:36 -04:00
|
|
|
|
private DialogOverlay dialogOverlay;
|
|
|
|
|
|
2017-05-17 05:58:34 -03:00
|
|
|
|
private DirectOverlay direct;
|
|
|
|
|
|
2017-05-26 00:58:18 -03:00
|
|
|
|
private SocialOverlay social;
|
|
|
|
|
|
2017-06-15 17:03:33 +08:00
|
|
|
|
private UserProfileOverlay userProfile;
|
|
|
|
|
|
2017-09-25 17:58:03 +08:00
|
|
|
|
private BeatmapSetOverlay beatmapSetOverlay;
|
|
|
|
|
|
2017-08-01 15:12:12 +09:00
|
|
|
|
public virtual Storage GetStorageForStableInstall() => null;
|
|
|
|
|
|
2017-02-17 20:07:11 +09:00
|
|
|
|
private Intro intro
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
Screen s = screenStack;
|
|
|
|
|
while (s != null && !(s is Intro))
|
|
|
|
|
s = s.ChildScreen;
|
|
|
|
|
return s as Intro;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-11-09 15:23:10 +09:00
|
|
|
|
|
2017-07-13 16:39:27 +09:00
|
|
|
|
public float ToolbarOffset => Toolbar.Position.Y + Toolbar.DrawHeight;
|
|
|
|
|
|
2017-12-25 18:52:09 +09:00
|
|
|
|
public readonly BindableBool ShowOverlays = new BindableBool();
|
|
|
|
|
|
2017-02-17 18:59:30 +09:00
|
|
|
|
private OsuScreen screenStack;
|
2016-10-04 17:15:03 +09:00
|
|
|
|
|
2016-10-26 18:45:48 +09:00
|
|
|
|
private VolumeControl volume;
|
2018-01-18 17:00:23 +09:00
|
|
|
|
private OnScreenDisplay onscreenDisplay;
|
2016-10-26 18:45:48 +09:00
|
|
|
|
|
2017-04-15 05:52:46 +09:00
|
|
|
|
private Bindable<int> configRuleset;
|
2017-04-17 17:43:48 +09:00
|
|
|
|
public Bindable<RulesetInfo> Ruleset = new Bindable<RulesetInfo>();
|
2016-10-13 22:21:15 +08:00
|
|
|
|
|
2017-03-23 13:41:50 +09:00
|
|
|
|
private readonly string[] args;
|
2016-10-21 18:25:22 +09:00
|
|
|
|
|
2017-05-15 10:55:29 +09:00
|
|
|
|
private SettingsOverlay settings;
|
2016-11-08 19:26:12 +09:00
|
|
|
|
|
2018-01-26 19:30:29 +09:00
|
|
|
|
// todo: move this to SongSelect once Screen has the ability to unsuspend.
|
|
|
|
|
public readonly Bindable<IEnumerable<Mod>> SelectedMods = new Bindable<IEnumerable<Mod>>(new List<Mod>());
|
|
|
|
|
|
2016-10-21 18:25:22 +09:00
|
|
|
|
public OsuGame(string[] args = null)
|
2016-10-10 16:56:01 -04:00
|
|
|
|
{
|
|
|
|
|
this.args = args;
|
|
|
|
|
}
|
2016-10-01 18:01:52 +09:00
|
|
|
|
|
2017-05-15 10:55:29 +09:00
|
|
|
|
public void ToggleSettings() => settings.ToggleVisibility();
|
2016-11-12 19:44:16 +09:00
|
|
|
|
|
2017-05-17 05:58:34 -03:00
|
|
|
|
public void ToggleDirect() => direct.ToggleVisibility();
|
|
|
|
|
|
2017-07-21 19:03:43 +02:00
|
|
|
|
private DependencyContainer dependencies;
|
|
|
|
|
|
|
|
|
|
protected override IReadOnlyDependencyContainer CreateLocalDependencies(IReadOnlyDependencyContainer parent) =>
|
|
|
|
|
dependencies = new DependencyContainer(base.CreateLocalDependencies(parent));
|
|
|
|
|
|
2016-11-12 19:44:16 +09:00
|
|
|
|
[BackgroundDependencyLoader]
|
2017-06-08 17:58:28 +09:00
|
|
|
|
private void load(FrameworkConfigManager frameworkConfig)
|
2016-08-26 12:28:23 +09:00
|
|
|
|
{
|
2017-06-08 17:58:28 +09:00
|
|
|
|
this.frameworkConfig = frameworkConfig;
|
|
|
|
|
|
2016-10-10 16:56:01 -04:00
|
|
|
|
if (!Host.IsPrimaryInstance)
|
|
|
|
|
{
|
2016-10-14 14:10:01 -04:00
|
|
|
|
Logger.Log(@"osu! does not support multiple running instances.", LoggingTarget.Runtime, LogLevel.Error);
|
2016-10-10 16:56:01 -04:00
|
|
|
|
Environment.Exit(0);
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-21 18:25:22 +09:00
|
|
|
|
if (args?.Length > 0)
|
2017-02-12 14:53:33 +09:00
|
|
|
|
{
|
|
|
|
|
var paths = args.Where(a => !a.StartsWith(@"-"));
|
2017-07-27 16:56:41 +09:00
|
|
|
|
Task.Run(() => BeatmapManager.Import(paths.ToArray()));
|
2017-02-12 14:53:33 +09:00
|
|
|
|
}
|
2016-10-21 18:25:22 +09:00
|
|
|
|
|
2018-01-29 15:05:07 +09:00
|
|
|
|
dependencies.CacheAs(this);
|
2016-11-10 17:40:42 -05:00
|
|
|
|
|
2017-05-15 10:56:27 +09:00
|
|
|
|
configRuleset = LocalConfig.GetBindable<int>(OsuSetting.Ruleset);
|
2017-11-21 15:13:00 +01:00
|
|
|
|
Ruleset.Value = RulesetStore.GetRuleset(configRuleset.Value) ?? RulesetStore.AvailableRulesets.First();
|
2017-04-17 19:44:03 +09:00
|
|
|
|
Ruleset.ValueChanged += r => configRuleset.Value = r.ID ?? 0;
|
2018-01-31 18:11:38 +09:00
|
|
|
|
|
|
|
|
|
LocalConfig.BindWith(OsuSetting.VolumeInactive, inactiveVolumeAdjust);
|
2016-11-12 18:34:36 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-04 21:35:12 +09:00
|
|
|
|
private ScheduledDelegate scoreLoad;
|
|
|
|
|
|
2018-01-17 20:32:26 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Open chat to a channel matching the provided name, if present.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="channelName">The name of the channel.</param>
|
2018-01-17 19:45:10 +09:00
|
|
|
|
public void OpenChannel(string channelName) => chat.OpenChannel(chat.AvailableChannels.Find(c => c.Name == channelName));
|
2018-01-09 16:11:45 +01:00
|
|
|
|
|
2018-01-17 20:32:26 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Show a beatmap set as an overlay.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="setId">The set to display.</param>
|
2018-01-17 19:45:10 +09:00
|
|
|
|
public void ShowBeatmapSet(int setId) => beatmapSetOverlay.ShowBeatmapSet(setId);
|
2018-01-09 16:11:45 +01:00
|
|
|
|
|
2017-03-04 19:02:36 +09:00
|
|
|
|
protected void LoadScore(Score s)
|
|
|
|
|
{
|
2017-03-04 21:35:12 +09:00
|
|
|
|
scoreLoad?.Cancel();
|
|
|
|
|
|
2017-03-04 19:02:36 +09:00
|
|
|
|
var menu = intro.ChildScreen;
|
|
|
|
|
|
|
|
|
|
if (menu == null)
|
|
|
|
|
{
|
2017-03-04 21:35:12 +09:00
|
|
|
|
scoreLoad = Schedule(() => LoadScore(s));
|
2017-03-04 19:02:36 +09:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!menu.IsCurrentScreen)
|
|
|
|
|
{
|
|
|
|
|
menu.MakeCurrent();
|
2017-07-17 16:51:21 +03:00
|
|
|
|
this.Delay(500).Schedule(() => LoadScore(s), out scoreLoad);
|
2017-03-04 19:02:36 +09:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (s.Beatmap == null)
|
|
|
|
|
{
|
2017-12-24 10:28:37 +01:00
|
|
|
|
notifications.Post(new SimpleNotification
|
2017-03-04 19:02:36 +09:00
|
|
|
|
{
|
|
|
|
|
Text = @"Tried to load a score for a beatmap we don't have!",
|
|
|
|
|
Icon = FontAwesome.fa_life_saver,
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 16:56:41 +09:00
|
|
|
|
Beatmap.Value = BeatmapManager.GetWorkingBeatmap(s.Beatmap);
|
2017-03-04 19:02:36 +09:00
|
|
|
|
|
2017-03-31 15:59:53 +09:00
|
|
|
|
menu.Push(new PlayerLoader(new ReplayPlayer(s.Replay)));
|
2017-03-04 19:02:36 +09:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-12 18:34:36 +01:00
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
2016-11-01 23:24:14 +09:00
|
|
|
|
|
2018-01-16 13:40:02 +09:00
|
|
|
|
// The next time this is updated is in UpdateAfterChildren, which occurs too late and results
|
|
|
|
|
// in the cursor being shown for a few frames during the intro.
|
|
|
|
|
// This prevents the cursor from showing until we have a screen with CursorVisible = true
|
|
|
|
|
CursorOverrideContainer.CanShowCursor = currentScreen?.CursorVisible ?? false;
|
|
|
|
|
|
2017-07-31 18:03:55 +09:00
|
|
|
|
// hook up notifications to components.
|
2017-12-24 10:28:37 +01:00
|
|
|
|
BeatmapManager.PostNotification = n => notifications?.Post(n);
|
2017-08-01 15:12:12 +09:00
|
|
|
|
BeatmapManager.GetStableStorage = GetStorageForStableInstall;
|
2017-07-31 18:03:55 +09:00
|
|
|
|
|
2017-10-23 13:08:58 +09:00
|
|
|
|
AddRange(new Drawable[]
|
|
|
|
|
{
|
2016-10-26 18:45:48 +09:00
|
|
|
|
new VolumeControlReceptor
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
2017-08-22 14:44:13 +09:00
|
|
|
|
ActionRequested = action => volume.Adjust(action)
|
2016-10-26 18:45:48 +09:00
|
|
|
|
},
|
2017-10-23 13:08:58 +09:00
|
|
|
|
mainContent = new Container { RelativeSizeAxes = Axes.Both },
|
|
|
|
|
overlayContent = new Container { RelativeSizeAxes = Axes.Both, Depth = float.MinValue },
|
2016-09-30 18:45:55 +09:00
|
|
|
|
});
|
2016-10-04 17:15:03 +09:00
|
|
|
|
|
2017-10-23 01:20:12 +09:00
|
|
|
|
loadComponentSingleFile(screenStack = new Loader(), d =>
|
2016-11-01 23:24:14 +09:00
|
|
|
|
{
|
2017-02-17 18:59:30 +09:00
|
|
|
|
screenStack.ModePushed += screenAdded;
|
|
|
|
|
screenStack.Exited += screenRemoved;
|
|
|
|
|
mainContent.Add(screenStack);
|
2016-11-01 23:24:14 +09:00
|
|
|
|
});
|
|
|
|
|
|
2017-10-23 01:20:12 +09:00
|
|
|
|
loadComponentSingleFile(Toolbar = new Toolbar
|
|
|
|
|
{
|
|
|
|
|
Depth = -5,
|
|
|
|
|
OnHome = delegate
|
|
|
|
|
{
|
|
|
|
|
hideAllOverlays();
|
|
|
|
|
intro?.ChildScreen?.MakeCurrent();
|
|
|
|
|
},
|
|
|
|
|
}, overlayContent.Add);
|
|
|
|
|
|
2017-10-24 13:10:17 +09:00
|
|
|
|
loadComponentSingleFile(volume = new VolumeControl(), Add);
|
2018-01-18 17:00:23 +09:00
|
|
|
|
loadComponentSingleFile(onscreenDisplay = new OnScreenDisplay(), Add);
|
2017-10-23 13:08:58 +09:00
|
|
|
|
|
2016-11-09 15:23:10 +09:00
|
|
|
|
//overlay elements
|
2017-10-23 01:20:12 +09:00
|
|
|
|
loadComponentSingleFile(direct = new DirectOverlay { Depth = -1 }, mainContent.Add);
|
|
|
|
|
loadComponentSingleFile(social = new SocialOverlay { Depth = -1 }, mainContent.Add);
|
|
|
|
|
loadComponentSingleFile(chat = new ChatOverlay { Depth = -1 }, mainContent.Add);
|
|
|
|
|
loadComponentSingleFile(settings = new MainSettings
|
2017-08-17 17:31:14 +09:00
|
|
|
|
{
|
|
|
|
|
GetToolbarHeight = () => ToolbarOffset,
|
|
|
|
|
Depth = -1
|
|
|
|
|
}, overlayContent.Add);
|
2017-10-23 01:20:12 +09:00
|
|
|
|
loadComponentSingleFile(userProfile = new UserProfileOverlay { Depth = -2 }, mainContent.Add);
|
|
|
|
|
loadComponentSingleFile(beatmapSetOverlay = new BeatmapSetOverlay { Depth = -3 }, mainContent.Add);
|
|
|
|
|
loadComponentSingleFile(musicController = new MusicController
|
2017-01-30 22:00:23 +09:00
|
|
|
|
{
|
2017-09-30 05:41:32 +03:00
|
|
|
|
Depth = -4,
|
2017-01-30 22:00:23 +09:00
|
|
|
|
Position = new Vector2(0, Toolbar.HEIGHT),
|
2017-01-30 22:53:50 +09:00
|
|
|
|
Anchor = Anchor.TopRight,
|
|
|
|
|
Origin = Anchor.TopRight,
|
2017-04-02 15:56:12 +09:00
|
|
|
|
}, overlayContent.Add);
|
2016-12-02 18:43:01 +09:00
|
|
|
|
|
2017-12-24 10:28:37 +01:00
|
|
|
|
loadComponentSingleFile(notifications = new NotificationOverlay
|
2017-02-10 16:26:43 +09:00
|
|
|
|
{
|
2017-12-23 14:33:43 +01:00
|
|
|
|
GetToolbarHeight = () => ToolbarOffset,
|
2017-09-30 05:41:32 +03:00
|
|
|
|
Depth = -4,
|
2017-02-10 16:26:43 +09:00
|
|
|
|
Anchor = Anchor.TopRight,
|
|
|
|
|
Origin = Anchor.TopRight,
|
2017-04-02 15:56:12 +09:00
|
|
|
|
}, overlayContent.Add);
|
2017-02-10 16:26:43 +09:00
|
|
|
|
|
2017-10-23 01:20:12 +09:00
|
|
|
|
loadComponentSingleFile(dialogOverlay = new DialogOverlay
|
2017-02-28 02:09:36 -04:00
|
|
|
|
{
|
2017-10-23 01:20:12 +09:00
|
|
|
|
Depth = -6,
|
2017-04-02 15:56:12 +09:00
|
|
|
|
}, overlayContent.Add);
|
2017-02-28 02:09:36 -04:00
|
|
|
|
|
2017-12-25 17:12:21 +09:00
|
|
|
|
forwardLoggedErrorsToNotifications();
|
2017-02-15 12:37:57 +09:00
|
|
|
|
|
2017-07-21 19:03:43 +02:00
|
|
|
|
dependencies.Cache(settings);
|
2018-01-18 17:00:23 +09:00
|
|
|
|
dependencies.Cache(onscreenDisplay);
|
2017-07-21 19:03:43 +02:00
|
|
|
|
dependencies.Cache(social);
|
2017-08-24 20:18:47 +09:00
|
|
|
|
dependencies.Cache(direct);
|
2017-07-21 19:03:43 +02:00
|
|
|
|
dependencies.Cache(chat);
|
|
|
|
|
dependencies.Cache(userProfile);
|
|
|
|
|
dependencies.Cache(musicController);
|
2017-09-25 17:58:03 +08:00
|
|
|
|
dependencies.Cache(beatmapSetOverlay);
|
2017-12-24 10:28:37 +01:00
|
|
|
|
dependencies.Cache(notifications);
|
2017-07-21 19:03:43 +02:00
|
|
|
|
dependencies.Cache(dialogOverlay);
|
2016-12-02 18:43:01 +09:00
|
|
|
|
|
2017-08-25 10:51:28 +09:00
|
|
|
|
// 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 09:10:04 +09:00
|
|
|
|
overlay.StateChanged += state =>
|
2017-08-25 10:51:28 +09:00
|
|
|
|
{
|
|
|
|
|
if (state == Visibility.Hidden) return;
|
|
|
|
|
|
|
|
|
|
foreach (var c in singleDisplayOverlays)
|
|
|
|
|
{
|
2017-09-04 09:10:04 +09:00
|
|
|
|
if (c == overlay) continue;
|
2017-08-25 10:51:28 +09:00
|
|
|
|
c.State = Visibility.Hidden;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-15 18:48:37 +09:00
|
|
|
|
// eventually informational overlays should be displayed in a stack, but for now let's only allow one to stay open at a time.
|
|
|
|
|
var informationalOverlays = new OverlayContainer[] { beatmapSetOverlay, userProfile };
|
|
|
|
|
foreach (var overlay in informationalOverlays)
|
|
|
|
|
{
|
|
|
|
|
overlay.StateChanged += state =>
|
|
|
|
|
{
|
|
|
|
|
if (state == Visibility.Hidden) return;
|
|
|
|
|
|
|
|
|
|
foreach (var c in informationalOverlays)
|
|
|
|
|
{
|
|
|
|
|
if (c == overlay) continue;
|
|
|
|
|
c.State = Visibility.Hidden;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-25 11:24:13 +09:00
|
|
|
|
void updateScreenOffset()
|
2016-11-30 18:28:08 +09:00
|
|
|
|
{
|
2017-12-24 14:32:56 +01:00
|
|
|
|
float offset = 0;
|
2017-12-23 14:56:23 +01:00
|
|
|
|
|
2017-12-24 14:32:56 +01:00
|
|
|
|
if (settings.State == Visibility.Visible)
|
2017-12-24 10:28:37 +01:00
|
|
|
|
offset += ToolbarButton.WIDTH / 2;
|
2017-12-24 14:32:56 +01:00
|
|
|
|
if (notifications.State == Visibility.Visible)
|
|
|
|
|
offset -= ToolbarButton.WIDTH / 2;
|
2017-12-24 10:28:37 +01:00
|
|
|
|
|
2017-12-25 11:24:13 +09:00
|
|
|
|
screenStack.MoveToX(offset, SettingsOverlay.TRANSITION_LENGTH, Easing.OutQuint);
|
|
|
|
|
}
|
2016-11-30 18:28:08 +09:00
|
|
|
|
|
2017-12-25 11:24:13 +09:00
|
|
|
|
settings.StateChanged += _ => updateScreenOffset();
|
|
|
|
|
notifications.StateChanged += _ => updateScreenOffset();
|
2017-12-24 10:28:37 +01:00
|
|
|
|
|
2017-12-25 18:52:09 +09:00
|
|
|
|
notifications.Enabled.BindTo(ShowOverlays);
|
|
|
|
|
|
2017-12-30 20:41:36 +09:00
|
|
|
|
ShowOverlays.ValueChanged += show =>
|
2017-12-25 18:52:09 +09:00
|
|
|
|
{
|
|
|
|
|
//central game screen change logic.
|
2017-12-30 20:41:36 +09:00
|
|
|
|
if (!show)
|
2017-12-25 18:52:09 +09:00
|
|
|
|
{
|
|
|
|
|
hideAllOverlays();
|
|
|
|
|
musicController.State = Visibility.Hidden;
|
|
|
|
|
Toolbar.State = Visibility.Hidden;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Toolbar.State = Visibility.Visible;
|
|
|
|
|
};
|
2016-10-06 21:10:01 +09:00
|
|
|
|
}
|
|
|
|
|
|
2017-12-25 17:12:21 +09:00
|
|
|
|
private void forwardLoggedErrorsToNotifications()
|
|
|
|
|
{
|
|
|
|
|
int recentErrorCount = 0;
|
|
|
|
|
|
|
|
|
|
const double debounce = 5000;
|
|
|
|
|
|
|
|
|
|
Logger.NewEntry += entry =>
|
|
|
|
|
{
|
|
|
|
|
if (entry.Level < LogLevel.Error || entry.Target == null) return;
|
|
|
|
|
|
|
|
|
|
if (recentErrorCount < 2)
|
|
|
|
|
{
|
|
|
|
|
notifications.Post(new SimpleNotification
|
|
|
|
|
{
|
|
|
|
|
Icon = FontAwesome.fa_bomb,
|
|
|
|
|
Text = (recentErrorCount == 0 ? entry.Message : "Subsequent errors occurred and have been logged.") + "\nClick to view log files.",
|
|
|
|
|
Activated = () =>
|
|
|
|
|
{
|
|
|
|
|
Host.Storage.GetStorageForDirectory("logs").OpenInNativeExplorer();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-25 18:22:58 +09:00
|
|
|
|
Interlocked.Increment(ref recentErrorCount);
|
2017-12-25 17:12:21 +09:00
|
|
|
|
|
2017-12-25 18:22:58 +09:00
|
|
|
|
Scheduler.AddDelayed(() => Interlocked.Decrement(ref recentErrorCount), debounce);
|
2017-12-25 17:12:21 +09:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-23 01:20:12 +09:00
|
|
|
|
private Task asyncLoadStream;
|
2017-12-30 20:41:36 +09:00
|
|
|
|
private int visibleOverlayCount;
|
2017-10-23 01:20:12 +09:00
|
|
|
|
|
|
|
|
|
private void loadComponentSingleFile<T>(T d, Action<T> add)
|
|
|
|
|
where T : Drawable
|
|
|
|
|
{
|
2017-12-30 20:41:36 +09:00
|
|
|
|
var focused = d as FocusedOverlayContainer;
|
|
|
|
|
if (focused != null)
|
|
|
|
|
{
|
|
|
|
|
focused.StateChanged += s =>
|
|
|
|
|
{
|
|
|
|
|
visibleOverlayCount += s == Visibility.Visible ? 1 : -1;
|
|
|
|
|
screenStack.FadeColour(visibleOverlayCount > 0 ? OsuColour.Gray(0.5f) : Color4.White, 500, Easing.OutQuint);
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-24 11:50:18 +09:00
|
|
|
|
// schedule is here to ensure that all component loads are done after LoadComplete is run (and thus all dependencies are cached).
|
|
|
|
|
// with some better organisation of LoadComplete to do construction and dependency caching in one step, followed by calls to loadComponentSingleFile,
|
|
|
|
|
// we could avoid the need for scheduling altogether.
|
2017-10-24 11:40:38 +09:00
|
|
|
|
Schedule(() => { asyncLoadStream = asyncLoadStream?.ContinueWith(t => LoadComponentAsync(d, add).Wait()) ?? LoadComponentAsync(d, add); });
|
2017-10-23 01:20:12 +09:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-10 19:52:45 +09:00
|
|
|
|
public bool OnPressed(GlobalAction action)
|
2016-10-08 00:43:06 +09:00
|
|
|
|
{
|
2017-08-10 19:28:24 +09:00
|
|
|
|
if (intro == null) return false;
|
2016-12-05 19:33:38 +09:00
|
|
|
|
|
2017-08-10 19:28:24 +09:00
|
|
|
|
switch (action)
|
2016-10-08 00:43:06 +09:00
|
|
|
|
{
|
2017-08-10 19:28:24 +09:00
|
|
|
|
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 19:27:37 +09:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-27 18:18:12 +03:00
|
|
|
|
return false;
|
2016-10-08 00:43:06 +09:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-31 18:11:38 +09:00
|
|
|
|
private readonly BindableDouble inactiveVolumeAdjust = new BindableDouble();
|
|
|
|
|
|
|
|
|
|
protected override void OnDeactivated()
|
|
|
|
|
{
|
|
|
|
|
base.OnDeactivated();
|
|
|
|
|
Audio.AddAdjustment(AdjustableProperty.Volume, inactiveVolumeAdjust);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnActivated()
|
|
|
|
|
{
|
|
|
|
|
base.OnActivated();
|
|
|
|
|
Audio.RemoveAdjustment(AdjustableProperty.Volume, inactiveVolumeAdjust);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-10 19:52:45 +09:00
|
|
|
|
public bool OnReleased(GlobalAction action) => false;
|
|
|
|
|
|
2016-11-01 23:24:14 +09:00
|
|
|
|
private Container mainContent;
|
|
|
|
|
|
2016-11-09 15:23:10 +09:00
|
|
|
|
private Container overlayContent;
|
|
|
|
|
|
2017-03-16 23:58:36 +09:00
|
|
|
|
private OsuScreen currentScreen;
|
2017-06-08 17:58:28 +09:00
|
|
|
|
private FrameworkConfigManager frameworkConfig;
|
2017-03-16 23:58:36 +09:00
|
|
|
|
|
2017-09-08 16:15:41 +09:00
|
|
|
|
private void hideAllOverlays()
|
|
|
|
|
{
|
|
|
|
|
settings.State = Visibility.Hidden;
|
|
|
|
|
chat.State = Visibility.Hidden;
|
|
|
|
|
direct.State = Visibility.Hidden;
|
|
|
|
|
social.State = Visibility.Hidden;
|
|
|
|
|
userProfile.State = Visibility.Hidden;
|
2017-12-24 10:28:37 +01:00
|
|
|
|
notifications.State = Visibility.Hidden;
|
2017-09-08 16:15:41 +09:00
|
|
|
|
}
|
|
|
|
|
|
2016-10-07 19:12:36 +09:00
|
|
|
|
protected override bool OnExiting()
|
|
|
|
|
{
|
2017-02-18 14:16:46 +09:00
|
|
|
|
if (screenStack.ChildScreen == null) return false;
|
|
|
|
|
|
|
|
|
|
if (intro == null) return true;
|
|
|
|
|
|
2017-02-17 18:59:30 +09:00
|
|
|
|
if (!intro.DidLoadMenu || intro.ChildScreen != null)
|
2016-10-07 19:12:36 +09:00
|
|
|
|
{
|
2017-02-17 15:33:08 +09:00
|
|
|
|
Scheduler.Add(intro.MakeCurrent);
|
2016-10-07 19:12:36 +09:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2016-10-13 22:21:15 +08:00
|
|
|
|
|
2016-10-07 19:12:36 +09:00
|
|
|
|
return base.OnExiting();
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-11 18:40:39 +09:00
|
|
|
|
/// <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 19:32:55 +09:00
|
|
|
|
protected override void UpdateAfterChildren()
|
|
|
|
|
{
|
|
|
|
|
base.UpdateAfterChildren();
|
|
|
|
|
|
2017-08-22 15:58:47 +09:00
|
|
|
|
// 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 17:21:19 +09:00
|
|
|
|
bool applyRestrictions = !currentScreen?.AllowBeatmapRulesetChange ?? false;
|
2017-08-22 15:58:47 +09:00
|
|
|
|
|
|
|
|
|
Ruleset.Disabled = applyRestrictions;
|
|
|
|
|
Beatmap.Disabled = applyRestrictions;
|
|
|
|
|
|
2017-07-13 16:39:27 +09:00
|
|
|
|
mainContent.Padding = new MarginPadding { Top = ToolbarOffset };
|
2018-01-12 19:34:55 +09:00
|
|
|
|
|
2018-01-15 14:00:13 +09:00
|
|
|
|
CursorOverrideContainer.CanShowCursor = currentScreen?.CursorVisible ?? false;
|
2017-02-08 19:32:55 +09:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-17 18:59:30 +09:00
|
|
|
|
private void screenAdded(Screen newScreen)
|
2016-10-06 21:10:01 +09:00
|
|
|
|
{
|
2017-12-28 00:38:50 +09:00
|
|
|
|
currentScreen = (OsuScreen)newScreen;
|
|
|
|
|
|
2017-02-17 18:59:30 +09:00
|
|
|
|
newScreen.ModePushed += screenAdded;
|
|
|
|
|
newScreen.Exited += screenRemoved;
|
2016-10-06 21:10:01 +09:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-17 18:59:30 +09:00
|
|
|
|
private void screenRemoved(Screen newScreen)
|
2016-10-06 21:10:01 +09:00
|
|
|
|
{
|
2017-12-28 00:38:50 +09:00
|
|
|
|
currentScreen = (OsuScreen)newScreen;
|
|
|
|
|
|
2017-12-26 16:09:40 +09:00
|
|
|
|
if (newScreen == null)
|
|
|
|
|
Exit();
|
2016-08-26 12:28:23 +09:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|