mirror of
https://github.com/MindWorkAI/AI-Studio.git
synced 2026-03-14 08:51:37 +00:00
Some checks are pending
Build and Release / Read metadata (push) Waiting to run
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-apple-darwin, osx-arm64, macos-latest, aarch64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-pc-windows-msvc.exe, win-arm64, windows-latest, aarch64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-unknown-linux-gnu, linux-arm64, ubuntu-22.04-arm, aarch64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-apple-darwin, osx-x64, macos-latest, x86_64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-pc-windows-msvc.exe, win-x64, windows-latest, x86_64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-unknown-linux-gnu, linux-x64, ubuntu-22.04, x86_64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Prepare & create release (push) Blocked by required conditions
Build and Release / Publish release (push) Blocked by required conditions
188 lines
7.0 KiB
C#
188 lines
7.0 KiB
C#
using AIStudio.Settings;
|
|
using AIStudio.Settings.DataModel;
|
|
using AIStudio.Tools.Rust;
|
|
|
|
using Microsoft.AspNetCore.Components;
|
|
|
|
namespace AIStudio.Tools.Services;
|
|
|
|
public sealed class GlobalShortcutService : BackgroundService, IMessageBusReceiver
|
|
{
|
|
private static bool IS_STARTUP_COMPLETED;
|
|
|
|
private enum ShortcutSyncSource
|
|
{
|
|
CONFIGURATION_CHANGED,
|
|
STARTUP_COMPLETED,
|
|
PLUGINS_RELOADED,
|
|
}
|
|
|
|
private readonly SemaphoreSlim registrationSemaphore = new(1, 1);
|
|
private readonly ILogger<GlobalShortcutService> logger;
|
|
private readonly SettingsManager settingsManager;
|
|
private readonly MessageBus messageBus;
|
|
private readonly RustService rustService;
|
|
|
|
public GlobalShortcutService(
|
|
ILogger<GlobalShortcutService> logger,
|
|
SettingsManager settingsManager,
|
|
MessageBus messageBus,
|
|
RustService rustService)
|
|
{
|
|
this.logger = logger;
|
|
this.settingsManager = settingsManager;
|
|
this.messageBus = messageBus;
|
|
this.rustService = rustService;
|
|
|
|
this.messageBus.RegisterComponent(this);
|
|
this.ApplyFilters([], [Event.CONFIGURATION_CHANGED, Event.PLUGINS_RELOADED, Event.STARTUP_COMPLETED]);
|
|
}
|
|
|
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
|
{
|
|
this.logger.LogInformation("The global shortcut service was initialized.");
|
|
await Task.Delay(Timeout.InfiniteTimeSpan, stoppingToken);
|
|
}
|
|
|
|
public override async Task StopAsync(CancellationToken cancellationToken)
|
|
{
|
|
this.messageBus.Unregister(this);
|
|
this.registrationSemaphore.Dispose();
|
|
await base.StopAsync(cancellationToken);
|
|
}
|
|
|
|
#region IMessageBusReceiver
|
|
|
|
public async Task ProcessMessage<T>(ComponentBase? sendingComponent, Event triggeredEvent, T? data)
|
|
{
|
|
switch (triggeredEvent)
|
|
{
|
|
case Event.CONFIGURATION_CHANGED:
|
|
if (!IS_STARTUP_COMPLETED)
|
|
return;
|
|
|
|
await this.RegisterAllShortcuts(ShortcutSyncSource.CONFIGURATION_CHANGED);
|
|
break;
|
|
|
|
case Event.STARTUP_COMPLETED:
|
|
IS_STARTUP_COMPLETED = true;
|
|
await this.RegisterAllShortcuts(ShortcutSyncSource.STARTUP_COMPLETED);
|
|
break;
|
|
|
|
case Event.PLUGINS_RELOADED:
|
|
if (!IS_STARTUP_COMPLETED)
|
|
return;
|
|
|
|
await this.RegisterAllShortcuts(ShortcutSyncSource.PLUGINS_RELOADED);
|
|
break;
|
|
}
|
|
}
|
|
|
|
public Task<TResult?> ProcessMessageWithResult<TPayload, TResult>(ComponentBase? sendingComponent, Event triggeredEvent, TPayload? data) => Task.FromResult<TResult?>(default);
|
|
|
|
#endregion
|
|
|
|
private async Task RegisterAllShortcuts(ShortcutSyncSource source)
|
|
{
|
|
await this.registrationSemaphore.WaitAsync();
|
|
try
|
|
{
|
|
this.logger.LogInformation("Registering global shortcuts (source='{Source}').", source);
|
|
foreach (var shortcutId in Enum.GetValues<Shortcut>())
|
|
{
|
|
if(shortcutId is Shortcut.NONE)
|
|
continue;
|
|
|
|
var shortcutState = await this.GetShortcutState(shortcutId, source);
|
|
var shortcut = shortcutState.Shortcut;
|
|
var isEnabled = shortcutState.IsEnabled;
|
|
this.logger.LogInformation(
|
|
"Sync shortcut '{ShortcutId}' (source='{Source}', enabled={IsEnabled}, configured='{Shortcut}').",
|
|
shortcutId,
|
|
source,
|
|
isEnabled,
|
|
shortcut);
|
|
|
|
if (shortcutState.UsesPersistedFallback)
|
|
{
|
|
this.logger.LogWarning(
|
|
"Using persisted shortcut fallback for '{ShortcutId}' during startup completion (source='{Source}', configured='{Shortcut}').",
|
|
shortcutId,
|
|
source,
|
|
shortcut);
|
|
}
|
|
|
|
if (isEnabled && !string.IsNullOrWhiteSpace(shortcut))
|
|
{
|
|
var success = await this.rustService.UpdateGlobalShortcut(shortcutId, shortcut);
|
|
if (success)
|
|
this.logger.LogInformation("Global shortcut '{ShortcutId}' ({Shortcut}) registered.", shortcutId, shortcut);
|
|
else
|
|
this.logger.LogWarning("Failed to register global shortcut '{ShortcutId}' ({Shortcut}).", shortcutId, shortcut);
|
|
}
|
|
else
|
|
{
|
|
this.logger.LogInformation(
|
|
"Disabling global shortcut '{ShortcutId}' (source='{Source}', enabled={IsEnabled}, configured='{Shortcut}').",
|
|
shortcutId,
|
|
source,
|
|
isEnabled,
|
|
shortcut);
|
|
|
|
// Disable the shortcut when empty or feature is disabled:
|
|
await this.rustService.UpdateGlobalShortcut(shortcutId, string.Empty);
|
|
}
|
|
}
|
|
|
|
this.logger.LogInformation("Global shortcuts registration completed (source='{Source}').", source);
|
|
}
|
|
finally
|
|
{
|
|
this.registrationSemaphore.Release();
|
|
}
|
|
}
|
|
|
|
private string GetShortcutValue(Shortcut name) => name switch
|
|
{
|
|
Shortcut.VOICE_RECORDING_TOGGLE => this.settingsManager.ConfigurationData.App.ShortcutVoiceRecording,
|
|
|
|
_ => string.Empty,
|
|
};
|
|
|
|
private bool IsShortcutAllowed(Shortcut name) => name switch
|
|
{
|
|
// Voice recording is a preview feature:
|
|
Shortcut.VOICE_RECORDING_TOGGLE => PreviewFeatures.PRE_SPEECH_TO_TEXT_2026.IsEnabled(this.settingsManager),
|
|
|
|
// Other shortcuts are always allowed:
|
|
_ => true,
|
|
};
|
|
|
|
private async Task<ShortcutState> GetShortcutState(Shortcut shortcutId, ShortcutSyncSource source)
|
|
{
|
|
var shortcut = this.GetShortcutValue(shortcutId);
|
|
var isEnabled = this.IsShortcutAllowed(shortcutId);
|
|
if (isEnabled && !string.IsNullOrWhiteSpace(shortcut))
|
|
return new(shortcut, true, false);
|
|
|
|
if (source is not ShortcutSyncSource.STARTUP_COMPLETED || shortcutId is not Shortcut.VOICE_RECORDING_TOGGLE)
|
|
return new(shortcut, isEnabled, false);
|
|
|
|
var settingsSnapshot = await this.settingsManager.TryReadSettingsSnapshot();
|
|
if (settingsSnapshot is null)
|
|
return new(shortcut, isEnabled, false);
|
|
|
|
var fallbackShortcut = settingsSnapshot.App.ShortcutVoiceRecording;
|
|
var fallbackEnabled =
|
|
settingsSnapshot.App.EnabledPreviewFeatures.Contains(PreviewFeatures.PRE_SPEECH_TO_TEXT_2026) &&
|
|
!string.IsNullOrWhiteSpace(settingsSnapshot.App.UseTranscriptionProvider);
|
|
|
|
if (!fallbackEnabled || string.IsNullOrWhiteSpace(fallbackShortcut))
|
|
return new(shortcut, isEnabled, false);
|
|
|
|
return new(fallbackShortcut, true, true);
|
|
}
|
|
|
|
private readonly record struct ShortcutState(string Shortcut, bool IsEnabled, bool UsesPersistedFallback);
|
|
}
|