2019-01-24 08:43:03 +00:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
using System ;
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
using System.Reflection ;
2021-10-12 07:04:09 +00:00
using System.Threading ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Allocation ;
using osu.Framework.Audio ;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables ;
2019-06-20 03:48:45 +00:00
using osu.Framework.Development ;
2020-07-29 03:39:18 +00:00
using osu.Framework.Extensions ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
using osu.Framework.Graphics.Performance ;
using osu.Framework.Graphics.Textures ;
2018-07-09 06:26:22 +00:00
using osu.Framework.Input ;
2021-09-30 15:27:54 +00:00
using osu.Framework.IO.Stores ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Logging ;
2021-09-30 15:27:54 +00:00
using osu.Framework.Platform ;
2018-05-09 11:52:46 +00:00
using osu.Game.Audio ;
2021-09-30 15:27:54 +00:00
using osu.Game.Beatmaps ;
2022-02-16 07:57:28 +00:00
using osu.Game.Beatmaps.Formats ;
2021-09-30 15:27:54 +00:00
using osu.Game.Configuration ;
2018-04-13 09:19:50 +00:00
using osu.Game.Database ;
2021-09-30 15:27:54 +00:00
using osu.Game.Graphics ;
using osu.Game.Graphics.Cursor ;
2018-04-13 09:19:50 +00:00
using osu.Game.Input ;
using osu.Game.Input.Bindings ;
using osu.Game.IO ;
2020-12-24 08:58:38 +00:00
using osu.Game.Online ;
2021-09-30 15:27:54 +00:00
using osu.Game.Online.API ;
2021-02-12 05:54:19 +00:00
using osu.Game.Online.Chat ;
2020-12-25 04:38:11 +00:00
using osu.Game.Online.Multiplayer ;
2020-10-22 04:41:54 +00:00
using osu.Game.Online.Spectator ;
2020-08-06 10:01:23 +00:00
using osu.Game.Overlays ;
2019-12-28 13:13:18 +00:00
using osu.Game.Resources ;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets ;
2019-05-15 04:00:11 +00:00
using osu.Game.Rulesets.Mods ;
2018-11-28 07:12:57 +00:00
using osu.Game.Scoring ;
2018-04-13 09:19:50 +00:00
using osu.Game.Skinning ;
2021-04-08 23:34:35 +00:00
using osu.Game.Utils ;
2020-08-03 09:48:10 +00:00
using RuntimeInfo = osu . Framework . RuntimeInfo ;
2018-04-13 09:19:50 +00:00
namespace osu.Game
{
/// <summary>
/// The most basic <see cref="Game"/> that can be used to host osu! components and systems.
/// Unlike <see cref="OsuGame"/>, this class will not load any kind of UI, allowing it to be used
/// for provide dependencies to test cases without interfering with them.
/// </summary>
2021-05-27 17:44:44 +00:00
public partial class OsuGameBase : Framework . Game , ICanAcceptFiles
2018-04-13 09:19:50 +00:00
{
2022-02-18 06:57:37 +00:00
public const string OSU_PROTOCOL = "osu://" ;
2021-05-27 17:27:06 +00:00
public const string CLIENT_STREAM_NAME = @"lazer" ;
2019-06-03 04:16:05 +00:00
2020-03-21 17:16:28 +00:00
public const int SAMPLE_CONCURRENCY = 6 ;
2021-09-05 04:25:10 +00:00
/// <summary>
/// Length of debounce (in milliseconds) for commonly occuring sample playbacks that could stack.
/// </summary>
public const int SAMPLE_DEBOUNCE_TIME = 20 ;
2021-05-27 17:37:14 +00:00
/// <summary>
/// The maximum volume at which audio tracks should playback. This can be set lower than 1 to create some head-room for sound effects.
/// </summary>
2021-11-09 08:27:07 +00:00
private const double global_track_volume_adjust = 0.8 ;
2021-05-27 17:37:14 +00:00
2022-03-14 04:54:54 +00:00
public virtual bool UseDevelopmentServer = > DebugUtils . IsDebugBuild ;
2020-12-24 08:58:38 +00:00
2021-05-27 17:37:14 +00:00
public virtual Version AssemblyVersion = > Assembly . GetEntryAssembly ( ) ? . GetName ( ) . Version ? ? new Version ( ) ;
/// <summary>
/// MD5 representation of the game executable.
/// </summary>
public string VersionHash { get ; private set ; }
public bool IsDeployedBuild = > AssemblyVersion . Major > 0 ;
public virtual string Version
{
get
{
if ( ! IsDeployedBuild )
return @"local " + ( DebugUtils . IsDebugBuild ? @"debug" : @"release" ) ;
var version = AssemblyVersion ;
2021-07-04 03:39:50 +00:00
return $@"{version.Major}.{version.Minor}.{version.Build}-lazer" ;
2021-05-27 17:37:14 +00:00
}
}
2022-02-04 09:58:29 +00:00
/// <summary>
/// The <see cref="Edges"/> that the game should be drawn over at a top level.
/// Defaults to <see cref="Edges.None"/>.
/// </summary>
2022-02-04 13:10:49 +00:00
protected virtual Edges SafeAreaOverrideEdges = > Edges . None ;
2022-02-04 09:58:29 +00:00
2021-05-27 17:27:06 +00:00
protected OsuConfigManager LocalConfig { get ; private set ; }
2018-04-13 09:19:50 +00:00
2021-04-19 02:30:55 +00:00
protected SessionStatics SessionStatics { get ; private set ; }
2021-05-27 17:27:06 +00:00
protected BeatmapManager BeatmapManager { get ; private set ; }
2018-04-13 09:19:50 +00:00
2021-11-25 08:23:46 +00:00
protected BeatmapModelDownloader BeatmapDownloader { get ; private set ; }
2021-05-27 17:27:06 +00:00
protected ScoreManager ScoreManager { get ; private set ; }
2018-11-28 07:47:10 +00:00
2021-11-25 08:21:05 +00:00
protected ScoreModelDownloader ScoreDownloader { get ; private set ; }
2021-05-27 17:27:06 +00:00
protected SkinManager SkinManager { get ; private set ; }
2018-04-13 09:19:50 +00:00
2022-02-16 07:57:28 +00:00
protected RealmRulesetStore RulesetStore { get ; private set ; }
2018-04-13 09:19:50 +00:00
2021-06-10 04:58:08 +00:00
protected RealmKeyBindingStore KeyBindingStore { get ; private set ; }
2018-04-13 09:19:50 +00:00
2021-05-27 17:27:06 +00:00
protected MenuCursorContainer MenuCursorContainer { get ; private set ; }
2018-12-06 03:17:08 +00:00
2021-05-27 17:27:06 +00:00
protected MusicController MusicController { get ; private set ; }
protected IAPIProvider API { get ; set ; }
2020-10-22 04:41:54 +00:00
2021-05-27 17:27:06 +00:00
protected Storage Storage { get ; set ; }
protected Bindable < WorkingBeatmap > Beatmap { get ; private set ; } // cached via load() method
2018-04-13 09:19:50 +00:00
2019-05-15 04:00:11 +00:00
[Cached]
[Cached(typeof(IBindable<RulesetInfo>))]
protected readonly Bindable < RulesetInfo > Ruleset = new Bindable < RulesetInfo > ( ) ;
2021-02-10 05:38:15 +00:00
/// <summary>
/// The current mod selection for the local user.
/// </summary>
/// <remarks>
/// If a mod select overlay is present, mod instances set to this value are not guaranteed to remain as the provided instance and will be overwritten by a copy.
/// In such a case, changes to settings of a mod will *not* propagate after a mod is added to this collection.
/// As such, all settings should be finalised before adding a mod to this collection.
/// </remarks>
2019-05-15 04:00:11 +00:00
[Cached]
2019-12-13 11:13:53 +00:00
[Cached(typeof(IBindable<IReadOnlyList<Mod>>))]
2019-12-13 12:45:38 +00:00
protected readonly Bindable < IReadOnlyList < Mod > > SelectedMods = new Bindable < IReadOnlyList < Mod > > ( Array . Empty < Mod > ( ) ) ;
2019-02-01 06:42:15 +00:00
2019-12-12 09:53:25 +00:00
/// <summary>
/// Mods available for the current <see cref="Ruleset"/>.
/// </summary>
public readonly Bindable < Dictionary < ModType , IReadOnlyList < Mod > > > AvailableMods = new Bindable < Dictionary < ModType , IReadOnlyList < Mod > > > ( ) ;
2021-05-27 17:30:31 +00:00
private BeatmapDifficultyCache difficultyCache ;
private UserLookupCache userCache ;
2021-11-30 10:27:43 +00:00
private BeatmapLookupCache beatmapCache ;
2021-05-27 17:30:31 +00:00
private RulesetConfigCache rulesetConfigCache ;
private SpectatorClient spectatorClient ;
private MultiplayerClient multiplayerClient ;
2022-01-24 10:59:58 +00:00
private RealmAccess realm ;
2021-01-07 05:07:36 +00:00
2021-05-27 17:30:31 +00:00
protected override Container < Drawable > Content = > content ;
private Container content ;
2021-05-27 17:37:14 +00:00
private DependencyContainer dependencies ;
2018-04-13 09:19:50 +00:00
2021-05-27 17:37:14 +00:00
private Bindable < bool > fpsDisplayVisible ;
2018-04-13 09:19:50 +00:00
2021-11-09 08:27:07 +00:00
private readonly BindableNumber < double > globalTrackVolumeAdjust = new BindableNumber < double > ( global_track_volume_adjust ) ;
2018-04-13 09:19:50 +00:00
2022-01-21 05:56:28 +00:00
/// <summary>
/// A legacy EF context factory if migration has not been performed to realm yet.
/// </summary>
protected DatabaseContextFactory EFContextFactory { get ; private set ; }
2018-04-13 09:19:50 +00:00
public OsuGameBase ( )
{
2021-07-04 03:39:50 +00:00
Name = @"osu!" ;
2018-04-13 09:19:50 +00:00
}
[BackgroundDependencyLoader]
2021-11-08 09:24:37 +00:00
private void load ( ReadableKeyCombinationProvider keyCombinationProvider )
2018-04-13 09:19:50 +00:00
{
2020-08-03 09:48:10 +00:00
try
{
using ( var str = File . OpenRead ( typeof ( OsuGameBase ) . Assembly . Location ) )
VersionHash = str . ComputeMD5Hash ( ) ;
}
catch
{
// special case for android builds, which can't read DLLs from a packed apk.
// should eventually be handled in a better way.
VersionHash = $"{Version}-{RuntimeInfo.OS}" . ComputeMD5Hash ( ) ;
}
2020-07-29 03:39:18 +00:00
2019-12-28 13:13:18 +00:00
Resources . AddStore ( new DllResourceStore ( OsuResources . ResourceAssembly ) ) ;
2018-06-04 11:01:55 +00:00
2022-01-21 05:56:28 +00:00
if ( Storage . Exists ( DatabaseContextFactory . DATABASE_NAME ) )
dependencies . Cache ( EFContextFactory = new DatabaseContextFactory ( Storage ) ) ;
2021-11-23 02:48:56 +00:00
2022-01-24 10:59:58 +00:00
dependencies . Cache ( realm = new RealmAccess ( Storage , "client" , EFContextFactory ) ) ;
2021-11-23 02:48:56 +00:00
2022-02-16 07:57:28 +00:00
dependencies . CacheAs < RulesetStore > ( RulesetStore = new RealmRulesetStore ( realm , Storage ) ) ;
2021-12-03 09:14:44 +00:00
dependencies . CacheAs < IRulesetStore > ( RulesetStore ) ;
2018-04-13 09:19:50 +00:00
2022-02-16 07:57:28 +00:00
Decoder . RegisterDependencies ( RulesetStore ) ;
2022-01-22 02:50:47 +00:00
// Backup is taken here rather than in EFToRealmMigrator to avoid recycling realm contexts
// after initial usages below. It can be moved once a direction is established for handling re-subscription.
// See https://github.com/ppy/osu/pull/16547 for more discussion.
if ( EFContextFactory ! = null )
{
2022-01-27 05:39:07 +00:00
const string backup_folder = "backups" ;
2022-01-22 02:50:47 +00:00
string migration = $"before_final_migration_{DateTimeOffset.UtcNow.ToUnixTimeSeconds()}" ;
2022-01-27 05:39:07 +00:00
EFContextFactory . CreateBackup ( Path . Combine ( backup_folder , $"client.{migration}.db" ) ) ;
realm . CreateBackup ( Path . Combine ( backup_folder , $"client.{migration}.realm" ) ) ;
2022-01-22 02:50:47 +00:00
using ( var source = Storage . GetStream ( "collection.db" ) )
2022-01-25 06:45:23 +00:00
{
if ( source ! = null )
{
2022-01-27 05:39:07 +00:00
using ( var destination = Storage . GetStream ( Path . Combine ( backup_folder , $"collection.{migration}.db" ) , FileAccess . Write , FileMode . CreateNew ) )
2022-01-25 06:45:23 +00:00
source . CopyTo ( destination ) ;
}
}
2022-01-22 02:50:47 +00:00
}
2020-05-04 08:01:05 +00:00
dependencies . CacheAs ( Storage ) ;
2019-01-25 02:41:44 +00:00
var largeStore = new LargeTextureStore ( Host . CreateTextureLoaderStore ( new NamespacedResourceStore < byte [ ] > ( Resources , @"Textures" ) ) ) ;
largeStore . AddStore ( Host . CreateTextureLoaderStore ( new OnlineStore ( ) ) ) ;
2018-09-08 17:41:47 +00:00
dependencies . Cache ( largeStore ) ;
2018-04-13 09:19:50 +00:00
dependencies . CacheAs ( this ) ;
2020-12-24 08:58:38 +00:00
dependencies . CacheAs ( LocalConfig ) ;
2018-04-13 09:19:50 +00:00
2021-09-06 15:45:53 +00:00
InitialiseFonts ( ) ;
2018-04-13 09:19:50 +00:00
2020-03-21 17:16:28 +00:00
Audio . Samples . PlaybackConcurrency = SAMPLE_CONCURRENCY ;
2022-01-24 10:59:58 +00:00
dependencies . Cache ( SkinManager = new SkinManager ( Storage , realm , Host , Resources , Audio , Scheduler ) ) ;
2018-09-01 08:39:54 +00:00
dependencies . CacheAs < ISkinSource > ( SkinManager ) ;
2020-12-24 08:58:38 +00:00
EndpointConfiguration endpoints = UseDevelopmentServer ? ( EndpointConfiguration ) new DevelopmentEndpointConfiguration ( ) : new ProductionEndpointConfiguration ( ) ;
2021-02-12 05:54:19 +00:00
MessageFormatter . WebsiteRootUrl = endpoints . WebsiteRootUrl ;
2021-02-14 14:31:57 +00:00
dependencies . CacheAs ( API ? ? = new APIAccess ( LocalConfig , endpoints , VersionHash ) ) ;
2018-09-01 08:39:54 +00:00
2021-05-20 07:30:56 +00:00
dependencies . CacheAs ( spectatorClient = new OnlineSpectatorClient ( endpoints ) ) ;
2021-05-20 06:39:45 +00:00
dependencies . CacheAs ( multiplayerClient = new OnlineMultiplayerClient ( endpoints ) ) ;
2018-09-01 08:39:54 +00:00
2019-05-31 05:51:12 +00:00
var defaultBeatmap = new DummyWorkingBeatmap ( Audio , Textures ) ;
2018-12-25 09:34:45 +00:00
2019-05-09 06:15:28 +00:00
// ordering is important here to ensure foreign keys rules are not broken in ModelStore.Cleanup()
2022-02-01 20:20:59 +00:00
dependencies . Cache ( ScoreManager = new ScoreManager ( RulesetStore , ( ) = > BeatmapManager , Storage , realm , Scheduler , ( ) = > difficultyCache , LocalConfig ) ) ;
2022-01-24 10:59:58 +00:00
dependencies . Cache ( BeatmapManager = new BeatmapManager ( Storage , realm , RulesetStore , API , Audio , Resources , Host , defaultBeatmap , performOnlineLookups : true ) ) ;
2019-05-09 06:15:28 +00:00
2021-11-25 08:42:41 +00:00
dependencies . Cache ( BeatmapDownloader = new BeatmapModelDownloader ( BeatmapManager , API ) ) ;
dependencies . Cache ( ScoreDownloader = new ScoreModelDownloader ( ScoreManager , API ) ) ;
2021-11-25 08:21:05 +00:00
2021-05-27 17:30:31 +00:00
dependencies . Cache ( difficultyCache = new BeatmapDifficultyCache ( ) ) ;
AddInternal ( difficultyCache ) ;
2020-07-21 14:13:04 +00:00
2021-05-27 17:30:31 +00:00
dependencies . Cache ( userCache = new UserLookupCache ( ) ) ;
AddInternal ( userCache ) ;
2020-11-06 07:38:57 +00:00
2021-11-30 10:27:43 +00:00
dependencies . Cache ( beatmapCache = new BeatmapLookupCache ( ) ) ;
AddInternal ( beatmapCache ) ;
2020-11-06 04:15:33 +00:00
var scorePerformanceManager = new ScorePerformanceCache ( ) ;
2020-11-02 05:49:25 +00:00
dependencies . Cache ( scorePerformanceManager ) ;
AddInternal ( scorePerformanceManager ) ;
2020-09-27 10:44:29 +00:00
2022-01-24 10:59:58 +00:00
dependencies . CacheAs < IRulesetConfigCache > ( rulesetConfigCache = new RulesetConfigCache ( realm , RulesetStore ) ) ;
2021-04-11 06:00:37 +00:00
2021-04-12 14:52:12 +00:00
var powerStatus = CreateBatteryInfo ( ) ;
2021-04-11 06:00:37 +00:00
if ( powerStatus ! = null )
dependencies . CacheAs ( powerStatus ) ;
2021-04-19 02:30:55 +00:00
dependencies . Cache ( SessionStatics = new SessionStatics ( ) ) ;
2018-09-01 08:39:54 +00:00
dependencies . Cache ( new OsuColour ( ) ) ;
2020-10-02 07:08:11 +00:00
RegisterImportHandler ( BeatmapManager ) ;
RegisterImportHandler ( ScoreManager ) ;
RegisterImportHandler ( SkinManager ) ;
2018-09-01 08:39:54 +00:00
2021-02-11 06:02:34 +00:00
// drop track volume game-wide to leave some head-room for UI effects / samples.
// this means that for the time being, gameplay sample playback is louder relative to the audio track, compared to stable.
// we may want to revisit this if users notice or complain about the difference (consider this a bit of a trial).
Audio . Tracks . AddAdjustment ( AdjustableProperty . Volume , globalTrackVolumeAdjust ) ;
2018-04-13 09:19:50 +00:00
2019-11-12 09:45:42 +00:00
Beatmap = new NonNullableBindable < WorkingBeatmap > ( defaultBeatmap ) ;
2020-03-04 10:07:34 +00:00
2019-11-12 09:45:42 +00:00
dependencies . CacheAs < IBindable < WorkingBeatmap > > ( Beatmap ) ;
dependencies . CacheAs ( Beatmap ) ;
2018-04-13 09:19:50 +00:00
2020-10-22 05:54:27 +00:00
// add api components to hierarchy.
2020-05-28 10:05:35 +00:00
if ( API is APIAccess apiAccess )
AddInternal ( apiAccess ) ;
2021-05-20 06:55:07 +00:00
AddInternal ( spectatorClient ) ;
2020-12-20 15:21:41 +00:00
AddInternal ( multiplayerClient ) ;
2020-10-22 05:54:27 +00:00
2021-05-27 17:30:31 +00:00
AddInternal ( rulesetConfigCache ) ;
2018-04-21 19:10:06 +00:00
2021-04-08 06:17:53 +00:00
GlobalActionContainer globalBindings ;
2022-02-04 07:07:05 +00:00
base . Content . Add ( new SafeAreaContainer
2018-04-21 19:10:06 +00:00
{
2022-02-04 09:58:29 +00:00
SafeAreaOverrideEdges = SafeAreaOverrideEdges ,
2022-02-04 07:07:05 +00:00
RelativeSizeAxes = Axes . Both ,
Child = CreateScalingContainer ( ) . WithChildren ( new Drawable [ ]
{
( MenuCursorContainer = new MenuCursorContainer
{
RelativeSizeAxes = Axes . Both
} ) . WithChild ( content = new OsuTooltipContainer ( MenuCursorContainer . Cursor )
{
RelativeSizeAxes = Axes . Both
} ) ,
// to avoid positional input being blocked by children, ensure the GlobalActionContainer is above everything.
globalBindings = new GlobalActionContainer ( this )
} )
} ) ;
2018-04-21 19:10:06 +00:00
2022-01-24 10:59:58 +00:00
KeyBindingStore = new RealmKeyBindingStore ( realm , keyCombinationProvider ) ;
2021-09-07 06:19:23 +00:00
KeyBindingStore . Register ( globalBindings , RulesetStore . AvailableRulesets ) ;
2021-01-12 05:59:48 +00:00
2021-04-08 06:17:53 +00:00
dependencies . Cache ( globalBindings ) ;
2018-05-09 11:52:46 +00:00
PreviewTrackManager previewTrackManager ;
2021-11-09 08:27:07 +00:00
dependencies . Cache ( previewTrackManager = new PreviewTrackManager ( BeatmapManager . BeatmapTrackStore ) ) ;
2018-05-09 11:52:46 +00:00
Add ( previewTrackManager ) ;
2019-12-12 09:53:25 +00:00
2020-08-06 10:01:23 +00:00
AddInternal ( MusicController = new MusicController ( ) ) ;
dependencies . CacheAs ( MusicController ) ;
2019-12-12 09:53:25 +00:00
Ruleset . BindValueChanged ( onRulesetChanged ) ;
2022-01-18 03:20:52 +00:00
Beatmap . BindValueChanged ( onBeatmapChanged ) ;
2019-12-12 09:53:25 +00:00
}
2021-09-06 15:45:53 +00:00
protected virtual void InitialiseFonts ( )
{
AddFont ( Resources , @"Fonts/osuFont" ) ;
AddFont ( Resources , @"Fonts/Torus/Torus-Regular" ) ;
AddFont ( Resources , @"Fonts/Torus/Torus-Light" ) ;
AddFont ( Resources , @"Fonts/Torus/Torus-SemiBold" ) ;
AddFont ( Resources , @"Fonts/Torus/Torus-Bold" ) ;
2021-10-03 21:36:39 +00:00
AddFont ( Resources , @"Fonts/Torus-Alternate/Torus-Alternate-Regular" ) ;
AddFont ( Resources , @"Fonts/Torus-Alternate/Torus-Alternate-Light" ) ;
AddFont ( Resources , @"Fonts/Torus-Alternate/Torus-Alternate-SemiBold" ) ;
AddFont ( Resources , @"Fonts/Torus-Alternate/Torus-Alternate-Bold" ) ;
2021-09-06 15:45:53 +00:00
AddFont ( Resources , @"Fonts/Inter/Inter-Regular" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-RegularItalic" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-Light" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-LightItalic" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-SemiBold" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-SemiBoldItalic" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-Bold" ) ;
AddFont ( Resources , @"Fonts/Inter/Inter-BoldItalic" ) ;
AddFont ( Resources , @"Fonts/Noto/Noto-Basic" ) ;
AddFont ( Resources , @"Fonts/Noto/Noto-Hangul" ) ;
AddFont ( Resources , @"Fonts/Noto/Noto-CJK-Basic" ) ;
AddFont ( Resources , @"Fonts/Noto/Noto-CJK-Compatibility" ) ;
AddFont ( Resources , @"Fonts/Noto/Noto-Thai" ) ;
AddFont ( Resources , @"Fonts/Venera/Venera-Light" ) ;
AddFont ( Resources , @"Fonts/Venera/Venera-Bold" ) ;
AddFont ( Resources , @"Fonts/Venera/Venera-Black" ) ;
}
2021-05-27 17:37:14 +00:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
// TODO: This is temporary until we reimplement the local FPS display.
// It's just to allow end-users to access the framework FPS display without knowing the shortcut key.
fpsDisplayVisible = LocalConfig . GetBindable < bool > ( OsuSetting . ShowFpsDisplay ) ;
fpsDisplayVisible . ValueChanged + = visible = > { FrameStatistics . Value = visible . NewValue ? FrameStatisticsMode . Minimal : FrameStatisticsMode . None ; } ;
fpsDisplayVisible . TriggerChange ( ) ;
FrameStatistics . ValueChanged + = e = > fpsDisplayVisible . Value = e . NewValue ! = FrameStatisticsMode . None ;
}
2021-05-31 04:39:18 +00:00
protected override IReadOnlyDependencyContainer CreateChildDependencies ( IReadOnlyDependencyContainer parent ) = >
dependencies = new DependencyContainer ( base . CreateChildDependencies ( parent ) ) ;
public override void SetHost ( GameHost host )
{
base . SetHost ( host ) ;
// may be non-null for certain tests
Storage ? ? = host . Storage ;
LocalConfig ? ? = UseDevelopmentServer
? new DevelopmentOsuConfigManager ( Storage )
: new OsuConfigManager ( Storage ) ;
}
/// <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 ) ;
}
2022-02-10 09:48:37 +00:00
public bool Migrate ( string path )
2021-05-31 04:39:18 +00:00
{
2021-06-29 11:16:57 +00:00
Logger . Log ( $@"Migrating osu! data from ""{Storage.GetFullPath(string.Empty)}"" to ""{path}""..." ) ;
2021-10-12 07:04:09 +00:00
IDisposable realmBlocker = null ;
try
2021-06-10 04:58:08 +00:00
{
2021-10-12 07:04:09 +00:00
ManualResetEventSlim readyToRun = new ManualResetEventSlim ( ) ;
Scheduler . Add ( ( ) = >
{
2022-01-24 10:59:58 +00:00
realmBlocker = realm . BlockAllOperations ( ) ;
2021-10-12 07:04:09 +00:00
readyToRun . Set ( ) ;
} , false ) ;
readyToRun . Wait ( ) ;
2022-02-10 09:48:37 +00:00
bool? cleanupSucceded = ( Storage as OsuStorage ) ? . Migrate ( Host . GetStorage ( path ) ) ;
Logger . Log ( @"Migration complete!" ) ;
return cleanupSucceded ! = false ;
2021-06-10 04:58:08 +00:00
}
2021-10-12 07:04:09 +00:00
finally
{
realmBlocker ? . Dispose ( ) ;
}
2021-05-31 04:39:18 +00:00
}
protected override UserInputManager CreateUserInputManager ( ) = > new OsuUserInputManager ( ) ;
protected virtual BatteryInfo CreateBatteryInfo ( ) = > null ;
protected virtual Container CreateScalingContainer ( ) = > new DrawSizePreservingFillContainer ( ) ;
protected override Storage CreateStorage ( GameHost host , Storage defaultStorage ) = > new OsuStorage ( host , defaultStorage ) ;
2022-01-18 03:20:52 +00:00
private void onBeatmapChanged ( ValueChangedEvent < WorkingBeatmap > valueChangedEvent )
{
2022-01-18 07:11:03 +00:00
if ( IsLoaded & & ! ThreadSafety . IsUpdateThread )
2022-01-18 03:20:52 +00:00
throw new InvalidOperationException ( "Global beatmap bindable must be changed from update thread." ) ;
}
2019-12-12 09:53:25 +00:00
private void onRulesetChanged ( ValueChangedEvent < RulesetInfo > r )
{
2022-01-18 07:11:03 +00:00
if ( IsLoaded & & ! ThreadSafety . IsUpdateThread )
2022-01-18 03:20:52 +00:00
throw new InvalidOperationException ( "Global ruleset bindable must be changed from update thread." ) ;
2022-01-10 04:59:33 +00:00
Ruleset instance = null ;
2021-11-24 03:16:08 +00:00
2022-01-10 04:59:33 +00:00
try
{
if ( r . NewValue ? . Available = = true )
{
instance = r . NewValue . CreateInstance ( ) ;
}
}
catch ( Exception e )
{
Logger . Error ( e , "Ruleset load failed and has been rolled back" ) ;
}
if ( instance = = null )
2019-12-14 17:36:49 +00:00
{
2021-07-26 07:34:38 +00:00
// reject the change if the ruleset is not available.
2021-07-29 16:54:30 +00:00
Ruleset . Value = r . OldValue ? . Available = = true ? r . OldValue : RulesetStore . AvailableRulesets . First ( ) ;
2021-07-26 07:34:38 +00:00
return ;
2019-12-14 17:36:49 +00:00
}
2019-12-12 09:53:25 +00:00
2021-07-26 07:34:38 +00:00
var dict = new Dictionary < ModType , IReadOnlyList < Mod > > ( ) ;
foreach ( ModType type in Enum . GetValues ( typeof ( ModType ) ) )
2021-11-24 03:16:08 +00:00
{
dict [ type ] = instance . GetModsFor ( type ) . ToList ( ) ;
}
2021-07-26 07:34:38 +00:00
2019-12-13 15:42:31 +00:00
if ( ! SelectedMods . Disabled )
SelectedMods . Value = Array . Empty < Mod > ( ) ;
2021-01-18 08:50:32 +00:00
2019-12-12 09:53:25 +00:00
AvailableMods . Value = dict ;
2018-04-13 09:19:50 +00:00
}
2019-10-15 07:14:06 +00:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
2020-10-19 10:03:22 +00:00
2019-10-15 07:14:06 +00:00
RulesetStore ? . Dispose ( ) ;
2020-05-22 14:26:37 +00:00
BeatmapManager ? . Dispose ( ) ;
2020-10-19 10:03:22 +00:00
LocalConfig ? . Dispose ( ) ;
2020-05-11 12:37:07 +00:00
2022-01-24 10:59:58 +00:00
realm ? . Dispose ( ) ;
2019-10-15 07:14:06 +00:00
}
2018-04-13 09:19:50 +00:00
}
}