1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-22 00:47:24 +08:00
osu-lazer/osu.Game/OsuGame.cs

575 lines
21 KiB
C#
Raw Normal View History

2018-01-05 19:21:19 +08:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2016-08-26 11:28:23 +08:00
using System;
using System.Collections.Generic;
using System.Drawing.Imaging;
using System.IO;
using osu.Framework.Configuration;
2017-02-17 17:59:30 +08:00
using osu.Framework.Screens;
2016-08-26 11:28:23 +08:00
using osu.Game.Configuration;
2016-09-01 18:06:09 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2016-10-01 17:01:52 +08:00
using osu.Game.Overlays;
using osu.Framework.Logging;
2016-11-09 07:13:20 +08:00
using osu.Framework.Allocation;
2016-12-01 13:22:29 +08:00
using osu.Game.Overlays.Toolbar;
2016-11-14 16:23:33 +08:00
using osu.Game.Screens;
using osu.Game.Screens.Menu;
using OpenTK;
2017-02-05 05:03:39 +08:00
using System.Linq;
2017-12-25 17:22:58 +08:00
using System.Threading;
2017-02-24 17:10:37 +08:00
using System.Threading.Tasks;
using osu.Framework.Audio;
using osu.Framework.Input;
2017-08-11 15:11:46 +08:00
using osu.Framework.Input.Bindings;
using osu.Framework.Platform;
using osu.Framework.Threading;
2017-03-04 18:02:36 +08:00
using osu.Game.Graphics;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Overlays.Notifications;
2017-07-26 12:22:46 +08:00
using osu.Game.Rulesets;
2017-03-04 18:02:36 +08:00
using osu.Game.Screens.Play;
2017-08-11 15:11:46 +08:00
using osu.Game.Input.Bindings;
using osu.Game.Rulesets.Mods;
2018-02-22 15:29:05 +08:00
using osu.Game.Skinning;
2017-12-30 19:41:36 +08:00
using OpenTK.Graphics;
2018-03-04 02:08:35 +08:00
using osu.Game.Overlays.Volume;
2016-08-26 11:28:23 +08:00
namespace osu.Game
{
2017-08-12 18:54:07 +08:00
public class OsuGame : OsuGameBase, IKeyBindingHandler<GlobalAction>
2016-08-26 11:28:23 +08:00
{
2016-10-01 17:01:52 +08:00
public Toolbar Toolbar;
private ChatOverlay chat;
private MusicController musicController;
private NotificationOverlay notifications;
2017-02-10 15:26:43 +08:00
private DialogOverlay dialogOverlay;
private DirectOverlay direct;
2017-05-26 11:58:18 +08:00
private SocialOverlay social;
2017-06-15 17:03:33 +08:00
private UserProfileOverlay userProfile;
private BeatmapSetOverlay beatmapSetOverlay;
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;
public readonly BindableBool ShowOverlays = new BindableBool();
2017-02-17 17:59:30 +08:00
private OsuScreen screenStack;
2018-03-04 02:08:35 +08:00
private VolumeOverlay volume;
private OnScreenDisplay onscreenDisplay;
2017-04-15 04:52:46 +08:00
private Bindable<int> configRuleset;
2017-04-17 16:43:48 +08:00
public Bindable<RulesetInfo> Ruleset = new Bindable<RulesetInfo>();
2016-10-13 22:21:15 +08:00
2018-02-22 15:29:05 +08:00
private Bindable<int> configSkin;
private Bindable<ScreenshotFormat> screenshotFormat;
private readonly string[] args;
private SettingsOverlay settings;
2016-11-08 18:26:12 +08: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>());
public OsuGame(string[] args = null)
{
this.args = args;
}
2016-10-01 17:01:52 +08: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 11:28:23 +08:00
{
this.frameworkConfig = frameworkConfig;
2018-02-15 13:19:16 +08:00
ScoreStore.ScoreImported += score => Schedule(() => LoadScore(score));
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(@"-"));
2018-02-15 13:19:16 +08:00
Task.Run(() => Import(paths.ToArray()));
}
2018-01-29 14:05:07 +08:00
dependencies.CacheAs(this);
2016-11-11 06:40:42 +08:00
2018-02-22 15:29:05 +08:00
// bind config int to database RulesetInfo
2017-05-15 09:56:27 +08:00
configRuleset = LocalConfig.GetBindable<int>(OsuSetting.Ruleset);
Ruleset.Value = RulesetStore.GetRuleset(configRuleset.Value) ?? RulesetStore.AvailableRulesets.First();
2017-04-17 18:44:03 +08:00
Ruleset.ValueChanged += r => configRuleset.Value = r.ID ?? 0;
2018-02-22 15:29:05 +08:00
// bind config int to database SkinInfo
configSkin = LocalConfig.GetBindable<int>(OsuSetting.Skin);
screenshotFormat = LocalConfig.GetBindable<ScreenshotFormat>(OsuSetting.ScreenshotFormat);
2018-02-22 15:29:05 +08:00
SkinManager.CurrentSkinInfo.ValueChanged += s => configSkin.Value = s.ID;
2018-02-23 12:22:33 +08:00
configSkin.ValueChanged += id => SkinManager.CurrentSkinInfo.Value = SkinManager.Query(s => s.ID == id) ?? SkinInfo.Default;
2018-02-22 15:29:05 +08:00
configSkin.TriggerChange();
LocalConfig.BindWith(OsuSetting.VolumeInactive, inactiveVolumeAdjust);
}
private ScheduledDelegate scoreLoad;
2018-01-17 19:32:26 +08:00
/// <summary>
/// Open chat to a channel matching the provided name, if present.
/// </summary>
/// <param name="channelName">The name of the channel.</param>
public void OpenChannel(string channelName) => chat.OpenChannel(chat.AvailableChannels.Find(c => c.Name == channelName));
2018-01-09 23:11:45 +08:00
2018-01-17 19:32:26 +08:00
/// <summary>
/// Show a beatmap set as an overlay.
/// </summary>
/// <param name="setId">The set to display.</param>
public void ShowBeatmapSet(int setId) => beatmapSetOverlay.ShowBeatmapSet(setId);
2018-01-09 23:11:45 +08:00
/// <summary>
/// Show a user's profile as an overlay.
/// </summary>
/// <param name="userId">The user to display.</param>
public void ShowUser(long userId) => userProfile.ShowUser(userId);
2017-03-04 18:02:36 +08:00
protected void LoadScore(Score s)
{
scoreLoad?.Cancel();
2017-03-04 18:02:36 +08:00
var menu = intro.ChildScreen;
if (menu == null)
{
scoreLoad = Schedule(() => LoadScore(s));
2017-03-04 18:02:36 +08:00
return;
}
if (!menu.IsCurrentScreen)
{
menu.MakeCurrent();
this.Delay(500).Schedule(() => LoadScore(s), out scoreLoad);
2017-03-04 18:02:36 +08:00
return;
}
if (s.Beatmap == null)
{
notifications.Post(new SimpleNotification
2017-03-04 18:02:36 +08: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 18:02:36 +08:00
menu.Push(new PlayerLoader(new ReplayPlayer(s.Replay)));
2017-03-04 18:02:36 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
2016-11-01 22:24:14 +08: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;
// hook up notifications to components.
SkinManager.PostNotification = n => notifications?.Post(n);
BeatmapManager.PostNotification = n => notifications?.Post(n);
BeatmapManager.GetStableStorage = GetStorageForStableInstall;
2017-10-23 12:08:58 +08:00
AddRange(new Drawable[]
{
new VolumeControlReceptor
{
RelativeSizeAxes = Axes.Both,
ActionRequested = action => volume.Adjust(action)
},
2017-10-23 12:08:58 +08:00
mainContent = new Container { RelativeSizeAxes = Axes.Both },
overlayContent = new Container { RelativeSizeAxes = Axes.Both, Depth = float.MinValue },
2016-09-30 17:45:55 +08:00
});
loadComponentSingleFile(screenStack = new Loader(), d =>
2016-11-01 22:24:14 +08:00
{
2017-02-17 17:59:30 +08:00
screenStack.ModePushed += screenAdded;
screenStack.Exited += screenRemoved;
mainContent.Add(screenStack);
2016-11-01 22:24:14 +08:00
});
loadComponentSingleFile(Toolbar = new Toolbar
{
Depth = -5,
OnHome = delegate
{
hideAllOverlays();
intro?.ChildScreen?.MakeCurrent();
},
}, overlayContent.Add);
2018-03-07 13:19:31 +08:00
loadComponentSingleFile(volume = new VolumeOverlay(), overlayContent.Add);
loadComponentSingleFile(onscreenDisplay = new OnScreenDisplay(), Add);
2017-10-23 12:08:58 +08:00
//overlay elements
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
{
GetToolbarHeight = () => ToolbarOffset,
Depth = -1
}, overlayContent.Add);
loadComponentSingleFile(userProfile = new UserProfileOverlay { Depth = -2 }, mainContent.Add);
loadComponentSingleFile(beatmapSetOverlay = new BeatmapSetOverlay { Depth = -3 }, mainContent.Add);
loadComponentSingleFile(musicController = new MusicController
{
Depth = -4,
Position = new Vector2(0, Toolbar.HEIGHT),
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
2017-04-02 14:56:12 +08:00
}, overlayContent.Add);
loadComponentSingleFile(notifications = new NotificationOverlay
2017-02-10 15:26:43 +08:00
{
GetToolbarHeight = () => ToolbarOffset,
Depth = -4,
2017-02-10 15:26:43 +08:00
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
2017-04-02 14:56:12 +08:00
}, overlayContent.Add);
2017-02-10 15:26:43 +08:00
loadComponentSingleFile(dialogOverlay = new DialogOverlay
{
Depth = -6,
2017-04-02 14:56:12 +08:00
}, overlayContent.Add);
forwardLoggedErrorsToNotifications();
dependencies.Cache(settings);
dependencies.Cache(onscreenDisplay);
dependencies.Cache(social);
2017-08-24 19:18:47 +08:00
dependencies.Cache(direct);
dependencies.Cache(chat);
dependencies.Cache(userProfile);
dependencies.Cache(musicController);
dependencies.Cache(beatmapSetOverlay);
dependencies.Cache(notifications);
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 08:10:04 +08:00
overlay.StateChanged += state =>
{
if (state == Visibility.Hidden) return;
foreach (var c in singleDisplayOverlays)
{
2017-09-04 08:10:04 +08:00
if (c == overlay) continue;
c.State = Visibility.Hidden;
}
};
}
// 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;
}
};
}
void updateScreenOffset()
{
float offset = 0;
if (settings.State == Visibility.Visible)
offset += ToolbarButton.WIDTH / 2;
if (notifications.State == Visibility.Visible)
offset -= ToolbarButton.WIDTH / 2;
screenStack.MoveToX(offset, SettingsOverlay.TRANSITION_LENGTH, Easing.OutQuint);
}
settings.StateChanged += _ => updateScreenOffset();
notifications.StateChanged += _ => updateScreenOffset();
notifications.Enabled.BindTo(ShowOverlays);
2017-12-30 19:41:36 +08:00
ShowOverlays.ValueChanged += show =>
{
//central game screen change logic.
2017-12-30 19:41:36 +08:00
if (!show)
{
hideAllOverlays();
musicController.State = Visibility.Hidden;
Toolbar.State = Visibility.Hidden;
}
else
Toolbar.State = Visibility.Visible;
};
}
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 17:22:58 +08:00
Interlocked.Increment(ref recentErrorCount);
2017-12-25 17:22:58 +08:00
Scheduler.AddDelayed(() => Interlocked.Decrement(ref recentErrorCount), debounce);
};
}
private Task asyncLoadStream;
2017-12-30 19:41:36 +08:00
private int visibleOverlayCount;
private void loadComponentSingleFile<T>(T d, Action<T> add)
where T : Drawable
{
2017-12-30 19:41:36 +08: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);
};
}
// 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.
Schedule(() => { asyncLoadStream = asyncLoadStream?.ContinueWith(t => LoadComponentAsync(d, add).Wait()) ?? LoadComponentAsync(d, add); });
}
2017-08-10 18:52:45 +08: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);
frameworkConfig.GetBindable<ConfineMouseMode>(FrameworkSetting.ConfineMouseMode).SetDefault();
return true;
case GlobalAction.ToggleToolbar:
Toolbar.ToggleVisibility();
return true;
case GlobalAction.ToggleSettings:
settings.ToggleVisibility();
return true;
case GlobalAction.ToggleDirect:
direct.ToggleVisibility();
return true;
case GlobalAction.TakeScreenshot:
if (Window.ScreenshotTakenAction == null)
Window.ScreenshotTakenAction = (screenshotBitmap) =>
{
var fileName = getScreenshotFileName(screenshotFormat);
switch (screenshotFormat.Value)
{
case ScreenshotFormat.Bmp:
screenshotBitmap.Save(fileName, ImageFormat.Bmp);
break;
case ScreenshotFormat.Png:
screenshotBitmap.Save(fileName, ImageFormat.Png);
break;
case ScreenshotFormat.Jpg:
screenshotBitmap.Save(fileName, ImageFormat.Jpeg);
break;
default:
throw new ArgumentOutOfRangeException(nameof(screenshotFormat));
}
};
RequestScreenshot();
return true;
2016-11-08 18:27:37 +08:00
}
2017-02-27 23:18:12 +08:00
return false;
}
private string getScreenshotFileName(ScreenshotFormat screenshotFormat)
{
// TODO Change screenshots location
var baseDirectory = new DirectoryInfo(AppDomain.CurrentDomain.BaseDirectory);
var screenshotsDirectory = baseDirectory.CreateSubdirectory("Screenshots");
var screenshotExtension = screenshotFormat.ToString().ToLower();
var screenshots = screenshotsDirectory.GetFiles($"*.{screenshotExtension}");
return Path.Combine(screenshotsDirectory.FullName, $"screenshot{screenshots.Length + 1}.{screenshotExtension}");
}
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 18:52:45 +08:00
public bool OnReleased(GlobalAction action) => false;
2016-11-01 22:24:14 +08:00
private Container mainContent;
private Container overlayContent;
2017-03-16 22:58:36 +08:00
private OsuScreen currentScreen;
private FrameworkConfigManager frameworkConfig;
2017-03-16 22:58:36 +08:00
private void hideAllOverlays()
{
settings.State = Visibility.Hidden;
chat.State = Visibility.Hidden;
direct.State = Visibility.Hidden;
social.State = Visibility.Hidden;
userProfile.State = Visibility.Hidden;
notifications.State = Visibility.Hidden;
}
protected override bool OnExiting()
{
2017-02-18 13:16:46 +08:00
if (screenStack.ChildScreen == null) return false;
if (intro == null) return true;
2017-02-17 17:59:30 +08:00
if (!intro.DidLoadMenu || intro.ChildScreen != null)
{
2017-02-17 14:33:08 +08:00
Scheduler.Add(intro.MakeCurrent);
return true;
}
2016-10-13 22:21:15 +08: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 18:32:55 +08: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 16:21:19 +08:00
bool applyRestrictions = !currentScreen?.AllowBeatmapRulesetChange ?? false;
Ruleset.Disabled = applyRestrictions;
Beatmap.Disabled = applyRestrictions;
mainContent.Padding = new MarginPadding { Top = ToolbarOffset };
CursorOverrideContainer.CanShowCursor = currentScreen?.CursorVisible ?? false;
2017-02-08 18:32:55 +08:00
}
2017-02-17 17:59:30 +08:00
private void screenAdded(Screen newScreen)
{
2017-12-27 23:38:50 +08:00
currentScreen = (OsuScreen)newScreen;
2017-02-17 17:59:30 +08:00
newScreen.ModePushed += screenAdded;
newScreen.Exited += screenRemoved;
}
2017-02-17 17:59:30 +08:00
private void screenRemoved(Screen newScreen)
{
2017-12-27 23:38:50 +08:00
currentScreen = (OsuScreen)newScreen;
2017-12-26 15:09:40 +08:00
if (newScreen == null)
Exit();
2016-08-26 11:28:23 +08:00
}
}
}