npx claudepluginhub davidortinau/maui-skills --plugin maui-skillsThis skill uses the workspace's default tool permissions.
For full service implementation, types, and UI integration patterns, see `references/speech-to-text-api.md`.
Guards .NET MAUI projects against deprecated, obsolete, or removed APIs in XAML/C#, Blazor Hybrid, and MauiReactor. Detects target frameworks/library versions and provides replacement patterns for code generation/review/editing.
Transcribes speech to text on iOS with Speech framework: live mic via AVAudioEngine, audio files, on-device/server, authorization, SpeechAnalyzer (iOS 26+).
Builds voice-enabled web interfaces using Web Speech API with React hooks for speech recognition, continuous dictation, text-to-speech, and accessibility features.
Share bugs, ideas, or general feedback.
For full service implementation, types, and UI integration patterns, see references/speech-to-text-api.md.
Always request permissions before starting speech recognition. Both microphone and speech permissions are required.
// ❌ Starting recognition without checking permissions
var result = await _speechService.StartListeningAsync();
// ✅ Always check permissions first
if (!await _speechService.RequestPermissionsAsync())
return; // Gracefully handle denial
var result = await _speechService.StartListeningAsync(cancellationToken);
Missing either NSSpeechRecognitionUsageDescription or NSMicrophoneUsageDescription in Info.plist will cause a runtime crash — not a graceful failure.
On Android, if the user denies the RECORD_AUDIO permission twice, the OS stops showing the prompt. You must guide users to Settings manually.
Always set a timeout to prevent indefinite listening sessions that drain battery:
// ❌ No timeout — listens forever if no speech detected
await _speechToText.StartListenAsync(options, CancellationToken.None);
// ✅ Use a combined timeout + user cancellation token
using var timeoutCts = new CancellationTokenSource(TimeSpan.FromSeconds(60));
using var combinedCts = CancellationTokenSource.CreateLinkedTokenSource(
userCancellationToken, timeoutCts.Token);
await _speechToText.StartListenAsync(options, combinedCts.Token);
Leaking event subscriptions causes duplicate processing and memory leaks:
// ❌ Subscribe without unsubscribe
_speechToText.RecognitionResultUpdated += OnRecognitionResultUpdated;
// ✅ Always unsubscribe in finally block
try
{
_speechToText.RecognitionResultUpdated += OnRecognitionResultUpdated;
// ... listen ...
}
finally
{
_speechToText.RecognitionResultUpdated -= OnRecognitionResultUpdated;
}
// ❌ Leaked CTS
_currentCts = new CancellationTokenSource();
// ✅ Dispose in finally
try { /* ... */ }
finally
{
_currentCts?.Dispose();
_currentCts = null;
}
| Platform | Pitfall |
|---|---|
| iOS | Missing either plist key → runtime crash |
| Android | User denies permission twice → OS stops prompting; must redirect to Settings |
| All | No timeout → battery drain from indefinite listening |
| All | Calling StartListeningAsync while already listening → returns error, not exception |
Wrap ISpeechToText in a service — Don't use SpeechToText.Default directly in ViewModels. Wrap in ISpeechRecognitionService for testability and state management.
Use partial results for UX — Subscribe to PartialResultReceived for live transcription feedback. Users expect to see words appear as they speak.
Continuous listening = loop with delay — Loop StartListeningAsync with small delays (Task.Delay(100)) for conversation mode.
Guard against double-start — Check state before starting:
if (State == SpeechRecognitionState.Listening)
return new SpeechRecognitionResultDto { Success = false, ErrorMessage = "Already listening" };
Natural language output — CommunityToolkit.Maui returns normalized, punctuated text — not raw phonemes. No post-processing needed for basic use cases.
UI-agnostic service — The ISpeechRecognitionService pattern works identically with XAML/MVVM, C# Markup, and MauiReactor. See references/speech-to-text-api.md for all three patterns.
CommunityToolkit.Maui NuGet installed (look up current version)UseMauiCommunityToolkit() called in MauiProgram.csISpeechToText registered as singleton via DINSSpeechRecognitionUsageDescription and NSMicrophoneUsageDescription in Info.plistRECORD_AUDIO permission in AndroidManifest.xmlStartListeningAsync callfinally blocksCancellationTokenSource disposed after use