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
using System ;
using System.Collections.Generic ;
using osu.Framework.Allocation ;
using osu.Framework.Input.Bindings ;
using osu.Game.Rulesets ;
using System.Linq ;
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>
public class DatabasedKeyBindingContainer < T > : KeyBindingContainer < T >
where T : struct
{
private readonly RulesetInfo ruleset ;
private readonly int? variant ;
private KeyBindingStore store ;
2021-01-15 12:29:37 +08:00
public override IEnumerable < IKeyBinding > DefaultKeyBindings = > ruleset . CreateInstance ( ) . GetDefaultKeyBindings ( variant ? ? 0 ) ;
2018-04-13 17:19:50 +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 )
2018-04-13 17:19:50 +08:00
{
this . ruleset = ruleset ;
this . variant = variant ;
if ( ruleset ! = null & & variant = = null )
throw new InvalidOperationException ( $"{nameof(variant)} can not be null when a non-null {nameof(ruleset)} is provided." ) ;
}
[BackgroundDependencyLoader]
private void load ( KeyBindingStore keyBindings )
{
store = keyBindings ;
}
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
store . KeyBindingChanged + = ReloadMappings ;
}
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
if ( store ! = null )
store . KeyBindingChanged - = ReloadMappings ;
}
2020-04-20 08:35:00 +08:00
protected override void ReloadMappings ( )
{
2021-04-07 16:41:05 +08:00
var defaults = DefaultKeyBindings . ToList ( ) ;
2020-04-20 08:35:00 +08:00
if ( ruleset ! = null & & ! ruleset . ID . HasValue )
// if the provided ruleset is not stored to the database, we have no way to retrieve custom bindings.
// fallback to defaults instead.
2021-04-07 16:41:05 +08:00
KeyBindings = defaults ;
2020-04-20 08:35:00 +08:00
else
2021-04-07 16:41:05 +08:00
{
KeyBindings = store . Query ( ruleset ? . ID , variant )
2021-05-24 16:22:55 +08:00
. OrderBy ( b = > defaults . FindIndex ( d = > ( int ) d . Action = = b . IntAction ) )
2021-04-07 16:41:05 +08:00
// 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.
2021-05-24 16:22:55 +08:00
. ToList ( ) ;
2021-04-07 16:41:05 +08:00
}
2020-04-20 08:35:00 +08:00
}
2018-04-13 17:19:50 +08:00
}
}