2021-01-07 13:35:15 +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.
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using osu.Framework.Input.Bindings;
|
|
|
|
using osu.Framework.Platform;
|
|
|
|
using osu.Game.Database;
|
|
|
|
using osu.Game.Input.Bindings;
|
|
|
|
using osu.Game.Rulesets;
|
|
|
|
|
2021-01-12 13:55:45 +08:00
|
|
|
#nullable enable
|
|
|
|
|
2021-01-07 13:35:15 +08:00
|
|
|
namespace osu.Game.Input
|
|
|
|
{
|
2021-01-11 18:47:43 +08:00
|
|
|
public class RealmKeyBindingStore : RealmBackedStore
|
2021-01-07 13:35:15 +08:00
|
|
|
{
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Fired whenever any key binding change occurs, across all rulesets and types.
|
|
|
|
/// </summary>
|
2021-01-12 13:55:45 +08:00
|
|
|
public event Action? KeyBindingChanged;
|
2021-01-07 13:35:15 +08:00
|
|
|
|
2021-01-12 13:59:48 +08:00
|
|
|
public RealmKeyBindingStore(RealmContextFactory contextFactory, Storage? storage = null)
|
2021-01-07 13:35:15 +08:00
|
|
|
: base(contextFactory, storage)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-01-07 14:51:16 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Retrieve all user-defined key combinations (in a format that can be displayed) for a specific action.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="globalAction">The action to lookup.</param>
|
|
|
|
/// <returns>A set of display strings for all the user's key configuration for the action.</returns>
|
|
|
|
public IEnumerable<string> GetReadableKeyCombinationsFor(GlobalAction globalAction)
|
|
|
|
{
|
2021-01-08 14:49:01 +08:00
|
|
|
foreach (var action in query().Where(b => (GlobalAction)b.Action == globalAction))
|
2021-01-07 14:51:16 +08:00
|
|
|
{
|
2021-01-08 14:49:01 +08:00
|
|
|
string str = ((IKeyBinding)action).KeyCombination.ReadableString();
|
2021-01-07 14:51:16 +08:00
|
|
|
|
|
|
|
// even if found, the readable string may be empty for an unbound action.
|
|
|
|
if (str.Length > 0)
|
|
|
|
yield return str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Register a new type of <see cref="KeyBindingContainer{T}"/>, adding default bindings from <see cref="KeyBindingContainer.DefaultKeyBindings"/>.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="container">The container to populate defaults from.</param>
|
|
|
|
public void Register(KeyBindingContainer container) => insertDefaults(container.DefaultKeyBindings);
|
2021-01-07 13:35:15 +08:00
|
|
|
|
2021-01-12 13:59:48 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Register a ruleset, adding default bindings for each of its variants.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="ruleset">The ruleset to populate defaults from.</param>
|
|
|
|
public void Register(RulesetInfo ruleset)
|
|
|
|
{
|
|
|
|
var instance = ruleset.CreateInstance();
|
|
|
|
|
|
|
|
using (ContextFactory.GetForWrite())
|
|
|
|
{
|
|
|
|
foreach (var variant in instance.AvailableVariants)
|
|
|
|
insertDefaults(instance.GetDefaultKeyBindings(variant), ruleset.ID, variant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Retrieve all key bindings for the provided specification.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="rulesetId">An optional ruleset ID. If null, global bindings are returned.</param>
|
|
|
|
/// <param name="variant">An optional ruleset variant. If null, the no-variant bindings are returned.</param>
|
|
|
|
/// <returns>A list of all key bindings found for the query, detached from the database.</returns>
|
2021-01-11 18:47:43 +08:00
|
|
|
public List<RealmKeyBinding> Query(int? rulesetId = null, int? variant = null) => query(rulesetId, variant).ToList();
|
2021-01-11 14:56:41 +08:00
|
|
|
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Retrieve all key bindings for the provided action type.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="action">The action to lookup.</param>
|
|
|
|
/// <typeparam name="T">The enum type of the action.</typeparam>
|
|
|
|
/// <returns>A list of all key bindings found for the query, detached from the database.</returns>
|
2021-01-11 18:47:43 +08:00
|
|
|
public List<RealmKeyBinding> Query<T>(T action)
|
2021-01-11 14:56:41 +08:00
|
|
|
where T : Enum
|
|
|
|
{
|
|
|
|
int lookup = (int)(object)action;
|
|
|
|
|
2021-01-11 18:47:43 +08:00
|
|
|
return query(null, null).Where(rkb => rkb.Action == lookup).ToList();
|
2021-01-11 14:56:41 +08:00
|
|
|
}
|
|
|
|
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Update the database mapping for the provided key binding.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="keyBinding">The key binding to update. Can be detached from the database.</param>
|
|
|
|
/// <param name="modification">The modification to apply to the key binding.</param>
|
2021-01-11 14:56:41 +08:00
|
|
|
public void Update(IHasGuidPrimaryKey keyBinding, Action<IKeyBinding> modification)
|
|
|
|
{
|
2021-01-11 15:59:56 +08:00
|
|
|
// the incoming instance could already be a live access object.
|
2021-01-12 13:55:45 +08:00
|
|
|
Live<RealmKeyBinding>? realmBinding = keyBinding as Live<RealmKeyBinding>;
|
2021-01-11 15:59:56 +08:00
|
|
|
|
2021-01-11 14:56:41 +08:00
|
|
|
using (var realm = ContextFactory.GetForWrite())
|
|
|
|
{
|
2021-01-11 15:59:56 +08:00
|
|
|
if (realmBinding == null)
|
|
|
|
{
|
|
|
|
// the incoming instance could be a raw realm object.
|
|
|
|
if (!(keyBinding is RealmKeyBinding rkb))
|
|
|
|
// if neither of the above cases succeeded, retrieve a realm object for further processing.
|
|
|
|
rkb = realm.Context.Find<RealmKeyBinding>(keyBinding.ID);
|
2021-01-11 15:18:25 +08:00
|
|
|
|
2021-01-11 15:59:56 +08:00
|
|
|
realmBinding = new Live<RealmKeyBinding>(rkb, ContextFactory);
|
|
|
|
}
|
2021-01-11 15:18:25 +08:00
|
|
|
|
2021-01-11 15:59:56 +08:00
|
|
|
realmBinding.PerformUpdate(modification);
|
2021-01-11 14:56:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
KeyBindingChanged?.Invoke();
|
|
|
|
}
|
|
|
|
|
2021-01-08 14:49:01 +08:00
|
|
|
private void insertDefaults(IEnumerable<IKeyBinding> defaults, int? rulesetId = null, int? variant = null)
|
2021-01-07 13:35:15 +08:00
|
|
|
{
|
|
|
|
using (var usage = ContextFactory.GetForWrite())
|
|
|
|
{
|
|
|
|
// compare counts in database vs defaults
|
|
|
|
foreach (var group in defaults.GroupBy(k => k.Action))
|
|
|
|
{
|
2021-01-08 14:49:01 +08:00
|
|
|
int count = query(rulesetId, variant).Count(k => k.Action == (int)group.Key);
|
2021-01-07 13:35:15 +08:00
|
|
|
int aimCount = group.Count();
|
|
|
|
|
|
|
|
if (aimCount <= count)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
foreach (var insertable in group.Skip(count).Take(aimCount - count))
|
|
|
|
{
|
|
|
|
// insert any defaults which are missing.
|
|
|
|
usage.Context.Add(new RealmKeyBinding
|
|
|
|
{
|
2021-01-07 14:42:21 +08:00
|
|
|
ID = Guid.NewGuid().ToString(),
|
2021-01-08 16:44:12 +08:00
|
|
|
KeyCombination = insertable.KeyCombination.ToString(),
|
|
|
|
Action = (int)insertable.Action,
|
2021-01-07 13:35:15 +08:00
|
|
|
RulesetID = rulesetId,
|
|
|
|
Variant = variant
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
2021-01-11 15:18:25 +08:00
|
|
|
/// Retrieve live queryable <see cref="RealmKeyBinding"/>s for a specified ruleset/variant content.
|
2021-01-07 13:35:15 +08:00
|
|
|
/// </summary>
|
2021-01-11 15:18:25 +08:00
|
|
|
/// <param name="rulesetId">An optional ruleset ID. If null, global bindings are returned.</param>
|
|
|
|
/// <param name="variant">An optional ruleset variant. If null, the no-variant bindings are returned.</param>
|
2021-01-08 14:49:01 +08:00
|
|
|
private IQueryable<RealmKeyBinding> query(int? rulesetId = null, int? variant = null) =>
|
|
|
|
ContextFactory.Get().All<RealmKeyBinding>().Where(b => b.RulesetID == rulesetId && b.Variant == variant);
|
2021-01-07 13:35:15 +08:00
|
|
|
}
|
|
|
|
}
|