2019-01-24 16:43:03 +08: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 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
2017-08-17 16:47:44 +08:00
using System ;
2017-08-14 19:19:25 +08:00
using System.Collections.Generic ;
2021-01-12 16:01:40 +08:00
using System.Linq ;
2017-08-11 15:11:46 +08:00
using osu.Framework.Allocation ;
using osu.Framework.Input.Bindings ;
2021-01-11 18:47:51 +08:00
using osu.Game.Database ;
2017-08-11 15:11:46 +08:00
using osu.Game.Rulesets ;
2022-01-25 15:44:44 +08:00
using Realms ;
2018-04-13 17:19:50 +08:00
2017-08-11 15:11:46 +08:00
namespace osu.Game.Input.Bindings
{
/// <summary>
/// A KeyBindingInputManager with a database backing for custom overrides.
/// </summary>
/// <typeparam name="T">The type of the custom action.</typeparam>
2018-01-30 13:49:12 +08:00
public class DatabasedKeyBindingContainer < T > : KeyBindingContainer < T >
2017-08-11 15:11:46 +08:00
where T : struct
{
private readonly RulesetInfo ruleset ;
2018-04-13 17:19:50 +08:00
2017-08-17 16:47:44 +08:00
private readonly int? variant ;
2018-04-13 17:19:50 +08:00
2021-01-13 15:53:04 +08:00
private IDisposable realmSubscription ;
2021-01-12 16:01:40 +08:00
[Resolved]
2022-01-24 18:59:58 +08:00
private RealmAccess realm { get ; set ; }
2021-01-12 16:01:40 +08:00
2021-01-08 14:49:01 +08:00
public override IEnumerable < IKeyBinding > DefaultKeyBindings = > ruleset . CreateInstance ( ) . GetDefaultKeyBindings ( variant ? ? 0 ) ;
2018-04-13 17:19:50 +08:00
2017-08-11 15:11:46 +08:00
/// <summary>
/// Create a new instance.
/// </summary>
/// <param name="ruleset">A reference to identify the current <see cref="Ruleset"/>. Used to lookup mappings. Null for global mappings.</param>
/// <param name="variant">An optional variant for the specified <see cref="Ruleset"/>. Used when a ruleset has more than one possible keyboard layouts.</param>
2019-11-17 20:48:23 +08:00
/// <param name="simultaneousMode">Specify how to deal with multiple matches of <see cref="KeyCombination"/>s and <typeparamref name="T"/>s.</param>
2020-03-02 17:54:00 +08:00
/// <param name="matchingMode">Specify how to deal with exact <see cref="KeyCombination"/> matches.</param>
public DatabasedKeyBindingContainer ( RulesetInfo ruleset = null , int? variant = null , SimultaneousBindingMode simultaneousMode = SimultaneousBindingMode . None , KeyCombinationMatchingMode matchingMode = KeyCombinationMatchingMode . Any )
: base ( simultaneousMode , matchingMode )
2017-08-11 15:11:46 +08:00
{
this . ruleset = ruleset ;
this . variant = variant ;
2018-04-13 17:19:50 +08:00
2017-08-17 16:47:44 +08:00
if ( ruleset ! = null & & variant = = null )
throw new InvalidOperationException ( $"{nameof(variant)} can not be null when a non-null {nameof(ruleset)} is provided." ) ;
2017-08-11 15:11:46 +08:00
}
2018-04-13 17:19:50 +08:00
2022-01-21 18:39:49 +08:00
protected override void LoadComplete ( )
{
2022-01-25 15:44:44 +08:00
realmSubscription = realm . RegisterForNotifications ( queryRealmKeyBindings , ( sender , changes , error ) = >
2022-01-23 18:50:29 +08:00
{
// The first fire of this is a bit redundant as this is being called in base.LoadComplete,
// but this is safest in case the subscription is restored after a context recycle.
2022-01-25 15:44:44 +08:00
reloadMappings ( sender . AsQueryable ( ) ) ;
2022-01-23 18:50:29 +08:00
} ) ;
2021-01-13 15:53:04 +08:00
base . LoadComplete ( ) ;
2017-08-11 15:11:46 +08:00
}
2018-04-13 17:19:50 +08:00
2022-01-25 15:44:44 +08:00
protected override void ReloadMappings ( ) = > reloadMappings ( queryRealmKeyBindings ( realm . Realm ) ) ;
2021-01-12 16:01:40 +08:00
2022-01-25 15:44:44 +08:00
private IQueryable < RealmKeyBinding > queryRealmKeyBindings ( Realm realm )
{
string rulesetName = ruleset ? . ShortName ;
return realm . All < RealmKeyBinding > ( )
. Where ( b = > b . RulesetName = = rulesetName & & b . Variant = = variant ) ;
2020-04-20 08:35:00 +08:00
}
2018-04-13 17:19:50 +08:00
2022-01-25 15:44:44 +08:00
private void reloadMappings ( IQueryable < RealmKeyBinding > realmKeyBindings )
2020-04-20 08:35:00 +08:00
{
2021-04-07 16:41:05 +08:00
var defaults = DefaultKeyBindings . ToList ( ) ;
2022-01-25 15:44:44 +08:00
List < RealmKeyBinding > newBindings = realmKeyBindings . Detach ( )
// this ordering is important to ensure that we read entries from the database in the order
// enforced by DefaultKeyBindings. allow for song select to handle actions that may otherwise
// have been eaten by the music controller due to query order.
. OrderBy ( b = > defaults . FindIndex ( d = > ( int ) d . Action = = b . ActionInt ) ) . ToList ( ) ;
2022-01-15 00:15:09 +08:00
// In the case no bindings were found in the database, presume this usage is for a non-databased ruleset.
// This actually should never be required and can be removed if it is ever deemed to cause a problem.
// See https://github.com/ppy/osu/issues/8805 for original reasoning, which is no longer valid as we use ShortName
// for lookups these days.
if ( newBindings . Count = = 0 )
2021-04-07 16:41:05 +08:00
KeyBindings = defaults ;
2020-04-20 08:35:00 +08:00
else
2022-01-15 00:15:09 +08:00
KeyBindings = newBindings ;
2020-04-20 08:35:00 +08:00
}
2022-01-25 15:44:44 +08:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
realmSubscription ? . Dispose ( ) ;
}
2017-08-11 15:11:46 +08:00
}
2017-10-06 05:23:26 +08:00
}