WordleKeyboard und viele andere Improvements

This commit is contained in:
Gottfried Wilhelm Leibniz 2022-04-20 15:56:56 +02:00
parent 626b31a3ce
commit 64d18102be
8 changed files with 309 additions and 113 deletions

View File

@ -0,0 +1,9 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
</Project>

View File

@ -0,0 +1,32 @@
using System.Numerics;
namespace GottfriedsNackenUtility;
/// <summary>
/// Bietet nütliche mathematische Funktionen.
/// </summary>
public static class MathHelper
{
/// <summary>
/// Enumeriert die Zahlen der Fibonacci-Folge.
/// </summary>
/// <remarks>
/// Der Datentyp ist <see cref="BigInteger"/>, da es ein paar Fibonacci-Zahlen gibt die nicht von <see cref="int"/> dargestellt werden können.
/// </remarks>
public static IEnumerable<BigInteger> Fibonacci()
{
BigInteger i = 0;
BigInteger j = 1;
while (true)
{
BigInteger fib = i + j;
i = j;
j = fib;
yield return fib;
}
}
}

View File

@ -0,0 +1,42 @@
using System.Numerics;
namespace GottfriedsNackenUtility;
/// <summary>
/// Hier sind suspiziöse Funktionen versteckt.
/// </summary>
public static class SusHelper
{
/// <summary>
/// Dies ist der ausgeklügelste fake-Fortschrittbalkensalgorithmus den die Welt je erblickt hat.
/// </summary>
/// <remarks>
/// Dieser Task läuft in einer Endlosschleife und MUSS mit Hilfe des Abbruchtokens abgebrochen werden!
/// </remarks>
/// <param name="progressChangedCallback">Das callback das aufgerufen wird, wenn der Fortschritt sich geändert hat.
/// Gibt einen Int zwischen 0 und 100 zurück oder -1, wenn der Fortschrittsbalken auf einen "indefinite" Zustand wechseln soll.</param>
/// <param name="cancellationToken">Ein Abbruchtoken, das beim Warten auf den Abschluss der Aufgabe überwacht werden soll.</param>
public static async Task UpdateDefinitelyNotFakeProgress(Action<int> progressChangedCallback, CancellationToken cancellationToken)
{
try
{
foreach (BigInteger fibonacci in MathHelper.Fibonacci())
{
await Task.Delay(250 + (int)(fibonacci % 420), cancellationToken);
if (cancellationToken.IsCancellationRequested)
break;
if (fibonacci.IsEven && (int)(fibonacci % 42) == 0)
{
progressChangedCallback(-1);
await Task.Delay(1337, cancellationToken);
continue;
}
progressChangedCallback((int)(fibonacci % 69));
}
}
catch (TaskCanceledException) { }
}
}

View File

@ -5,6 +5,8 @@ VisualStudioVersion = 17.1.32210.238
MinimumVisualStudioVersion = 10.0.40219.1 MinimumVisualStudioVersion = 10.0.40219.1
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "GottfriedsNackenWebseite", "GottfriedsNackenWebseite\GottfriedsNackenWebseite.csproj", "{464DD643-DAD9-4E16-BB24-36F789ED4C15}" Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "GottfriedsNackenWebseite", "GottfriedsNackenWebseite\GottfriedsNackenWebseite.csproj", "{464DD643-DAD9-4E16-BB24-36F789ED4C15}"
EndProject EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "GottfriedsNackenUtility", "GottfriedsNackenUtility\GottfriedsNackenUtility.csproj", "{0122762F-9961-4D28-8AA9-0D6AE7630208}"
EndProject
Global Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU Debug|Any CPU = Debug|Any CPU
@ -15,6 +17,10 @@ Global
{464DD643-DAD9-4E16-BB24-36F789ED4C15}.Debug|Any CPU.Build.0 = Debug|Any CPU {464DD643-DAD9-4E16-BB24-36F789ED4C15}.Debug|Any CPU.Build.0 = Debug|Any CPU
{464DD643-DAD9-4E16-BB24-36F789ED4C15}.Release|Any CPU.ActiveCfg = Release|Any CPU {464DD643-DAD9-4E16-BB24-36F789ED4C15}.Release|Any CPU.ActiveCfg = Release|Any CPU
{464DD643-DAD9-4E16-BB24-36F789ED4C15}.Release|Any CPU.Build.0 = Release|Any CPU {464DD643-DAD9-4E16-BB24-36F789ED4C15}.Release|Any CPU.Build.0 = Release|Any CPU
{0122762F-9961-4D28-8AA9-0D6AE7630208}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{0122762F-9961-4D28-8AA9-0D6AE7630208}.Debug|Any CPU.Build.0 = Debug|Any CPU
{0122762F-9961-4D28-8AA9-0D6AE7630208}.Release|Any CPU.ActiveCfg = Release|Any CPU
{0122762F-9961-4D28-8AA9-0D6AE7630208}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection EndGlobalSection
GlobalSection(SolutionProperties) = preSolution GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE HideSolutionNode = FALSE

View File

@ -1,15 +1,25 @@
@using System.Diagnostics @using System.Diagnostics
<!-- Hidden Alerts --> <!-- Hidden Alerts -->
<div hidden="@hideAlert"> <div hidden="@(_alert == AlertState.None)">
<MudContainer MaxWidth="MaxWidth.Small"> <MudContainer MaxWidth="MaxWidth.Small">
<MudGrid> <MudGrid>
<MudItem xs="1"></MudItem> <MudItem xs="1"></MudItem>
<MudItem xs="8"> <MudItem xs="8">
<MudAlert Severity="Severity.Success">Korrekt!</MudAlert> @if (_alert == AlertState.Correct)
{
<MudAlert Severity="Severity.Success">Korrekt!</MudAlert>
}
else if (_alert == AlertState.Invalid)
{
<MudAlert Severity="Severity.Error">Eingabe ungültig!</MudAlert>
}
</MudItem> </MudItem>
<MudItem xs="2"> <MudItem xs="2">
<MudIconButton Icon="@Icons.Filled.Replay" Color="Color.Primary" aria-label="nochmal spielen" OnClick="@PlayAgain"></MudIconButton> @if (_alert == AlertState.Correct)
{
<MudIconButton Icon="@Icons.Filled.Replay" Color="Color.Primary" aria-label="nochmal spielen" OnClick="@PlayAgain" />
}
</MudItem> </MudItem>
<MudItem xs="1"></MudItem> <MudItem xs="1"></MudItem>
</MudGrid> </MudGrid>
@ -19,7 +29,7 @@
<!-- Matrix mit Buchstabenfeldern --> <!-- Matrix mit Buchstabenfeldern -->
<div class="d-flex justify-center mt-2 mb-2"> <div class="d-flex justify-center mt-2 mb-2">
<div> <div>
@foreach (CharData[] guess in PreviousGuesses) @foreach (CharData[] guess in _previousGuesses)
{ {
<div class="d-flex mb-2"> <div class="d-flex mb-2">
@foreach (CharData cd in guess) @foreach (CharData cd in guess)
@ -48,25 +58,35 @@
</div> </div>
</div> </div>
<p>@Input</p> @*
<MudTextField T="string" FullWidth="true" Immediate="true" @bind-Value="Input" Label="Bla" Variant="Variant.Outlined" MaxLength="ColumnCount" Counter="ColumnCount" />
<MudTextField T="string" FullWidth="true" Immediate="true" @bind-Value="Input" Label="Blub" Variant="Variant.Outlined" MaxLength="ColumnCount" Counter="ColumnCount" />
*@
<!-- Input dialogue feld --> <!-- Input dialogue feld -->
<MudContainer MaxWidth="MaxWidth.Small" @onkeyup="@Check4Enter"> <MudContainer MaxWidth="MaxWidth.Small">
<MudGrid Spacing="2"> <MudGrid Spacing="2">
<MudItem xs="1"></MudItem> <MudItem xs="1" />
<MudItem xs="8"> <MudItem xs="8">
<MudTextField Class="" @ref="textField" T="string" FullWidth="true" Immediate="true" @bind-Value="@_input" Label="Dein Tipp:" Variant="Variant.Outlined" MaxLength="5" Mask="@InputMask" Counter="ColumnCount" /> <MudTextField @ref="_textField" T="string" FullWidth="true" Immediate="true" @bind-Value="Input" Label="Dein Tipp:" Variant="Variant.Outlined" MaxLength="ColumnCount" Mask="@InputMask" Counter="ColumnCount" OnKeyDown="OnKeyDown" />
@*<input Type="text" value="@Input"/>*@
</MudItem> </MudItem>
<MudItem xs="2"> <MudItem xs="2">
<MudButton style="height: 56px; margin-top: 6px;" Class="justify-center" Disabled="_input.Length != ColumnCount" <MudButton style="height: 56px; margin-top: 6px;" Class="justify-center" Disabled="Input.Length != ColumnCount"
Size="Size.Large" Variant="Variant.Outlined" EndIcon="@Icons.Material.Filled.Send" Color="Color.Primary" OnClick=@ButtonOnClick>OK</MudButton> Size="Size.Large" Variant="Variant.Outlined" EndIcon="@Icons.Material.Filled.Send" Color="Color.Primary" OnClick=@OnEnterGuessClick>OK</MudButton>
</MudItem> </MudItem>
<MudItem xs="1"></MudItem> <MudItem xs="1" />
</MudGrid> </MudGrid>
</MudContainer> </MudContainer>
@code { @code {
private enum AlertState
{
None,
Correct,
Invalid
}
public enum CharState public enum CharState
{ {
None, None,
@ -86,112 +106,88 @@
State = state; State = state;
} }
} }
private string _input = string.Empty;
[Parameter]
public string Input
{
get => _input;
set => _input = value;
}
public async Task SetText(string text) private bool _found = false;
{
textField.SetText(text);
}
public bool hideAlert = true; private AlertState _alert = AlertState.None;
public MudTextField<string> textField; private List<CharData[]> _previousGuesses = new();
public List<CharData[]> PreviousGuesses = new(); private MudTextField<string> _textField = default!;
private bool found = false;
private int _columnCount;
private string? _secret = null;
private IMask _inputMask = new RegexMask(@"^[A-Za-z]{0,5}$");
private Func<string, bool>? _checkInput;
private Func<string>? _generateSecret;
[Parameter] [Parameter]
public int ColumnCount public string Input { get; set; } = String.Empty;
{
get => _columnCount;
set => _columnCount = value;
}
[Parameter] [Parameter]
public string? Secret public int ColumnCount { get; set; }
{
get => _secret; public string? Secret { get; set; } = null;
set => _secret = value;
}
[Parameter] [Parameter]
public IMask InputMask public IMask InputMask { get; set; } = new RegexMask("^.*$");
{
get => _inputMask;
set => _inputMask = value;
}
/// <summary>
/// Prüft ob die getätigte Eingabe eine mögliche Lösung ist.
/// </summary>
[Parameter] [Parameter]
public Func<string, bool>? CheckInput public Func<string, bool> CheckInput { get; set; } = _ => true;
{
get => _checkInput;
set => _checkInput = value;
}
[Parameter]
public Func<string>? GenerateSecret
{
get => _generateSecret;
set => _generateSecret = value;
}
/// <summary> /// <summary>
/// Wird ausgelöst wenn eine Eingabe getätigt und validiert wurde. /// Wird ausgelöst wenn eine Eingabe getätigt und validiert wurde.
/// </summary> /// </summary>
public event EventHandler<CharData[]>? InputValidated; [Parameter]
public EventCallback<CharData[]> InputValidated { get; set; }
public void Check4Enter(KeyboardEventArgs e) /// <summary>
/// Wird ausgelöst wenn ein neues Secret generiert werden muss.
/// </summary>
[Parameter]
public EventCallback GenerateSecret { get; set; }
public async Task OnKeyDown(KeyboardEventArgs e)
{ {
_alert = AlertState.None;
if (e.Code == "Enter" || e.Code == "NumpadEnter") if (e.Code == "Enter" || e.Code == "NumpadEnter")
{ {
if(found) if(_found)
{ {
PlayAgain(); await PlayAgain();
} }
else else
{ {
ButtonOnClick(); await OnEnterGuessClick();
} }
} }
} }
public void ButtonOnClick() public async Task OnEnterGuessClick()
{ {
if (Input.Length != ColumnCount || found) return; if (Input.Length != ColumnCount || _found) return;
if (CheckInput != null && !CheckInput(Input)) if (!CheckInput(Input))
return; // TODO: Eingabe ungültig!
if (Secret == null && GenerateSecret != null)
Secret = GenerateSecret();
if(TestInput(Input))
{ {
found = true; _alert = AlertState.Invalid;
hideAlert = false; StateHasChanged();
return;
} }
textField.Clear(); if (Secret == null)
await GenerateSecret.InvokeAsync();
if(await TestInput(Input))
{
_found = true;
_alert = AlertState.Correct;
}
await _textField.Clear();
} }
/// <summary> /// <summary>
/// Prüft ob die Eingabe korrekt ist. /// Prüft ob die Eingabe korrekt ist.
/// </summary> /// </summary>
private bool TestInput(string input) private async Task<bool> TestInput(string input)
{ {
CharData[] validatedInput = new CharData[ColumnCount]; CharData[] validatedInput = new CharData[ColumnCount];
@ -221,20 +217,19 @@
validatedInput[i] = new CharData(guessedChar, state); validatedInput[i] = new CharData(guessedChar, state);
} }
PreviousGuesses.Add(validatedInput); _previousGuesses.Add(validatedInput);
InputValidated?.Invoke(this, validatedInput); await InputValidated.InvokeAsync(validatedInput);
return correct; return correct;
} }
public void PlayAgain() public async Task PlayAgain()
{ {
PreviousGuesses.Clear(); _previousGuesses.Clear();
hideAlert = true; _alert = AlertState.None;
found = false; _found = false;
if (GenerateSecret != null) await GenerateSecret.InvokeAsync();
Secret = GenerateSecret();
} }
} }

View File

@ -1,5 +1,4 @@
 <!-- Matrix mit Buchstabenfeldern -->
<!-- Matrix mit Buchstabenfeldern -->
<div class="d-flex justify-center mt-2 mb-2"> <div class="d-flex justify-center mt-2 mb-2">
<div> <div>
@{ @{
@ -13,28 +12,30 @@
char key = Keys[index]; char key = Keys[index];
char upperKey = char.ToUpper(key); char upperKey = char.ToUpper(key);
_knownKeyStates.TryGetValue(upperKey, out var keyState);
if (_correctKeys.Contains(upperKey)) if (keyState == WordleComponent.CharState.Correct)
{ {
<MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Success" @onclick="() => ButtonPressed(key)"> <MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Success">
<MudText Typo="Typo.h5">@upperKey</MudText> <MudText Typo="Typo.h5">@upperKey</MudText>
</MudButton> </MudButton>
} }
else if (_misplacedKeys.Contains(upperKey)) else if (keyState == WordleComponent.CharState.Misplaced)
{ {
<MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Warning" @onclick="() => ButtonPressed(key)"> <MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Warning">
<MudText Typo="Typo.h5">@upperKey</MudText> <MudText Typo="Typo.h5">@upperKey</MudText>
</MudButton> </MudButton>
} }
else if (_wrongKeys.Contains(upperKey)) else if (keyState == WordleComponent.CharState.Wrong)
{ {
<MudButton Disabled="true" Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Error" @onclick="() => ButtonPressed(key)"> <MudButton Disabled="true" Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" Color="Color.Error">
<MudText Typo="Typo.h5">@upperKey</MudText> <MudText Typo="Typo.h5">@upperKey</MudText>
</MudButton> </MudButton>
} }
else else
{ {
<MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em" @onclick="() => ButtonPressed(key)"> <MudButton Variant="Variant.Outlined" Class="mr-2" Style="width: 5em; height: 5em">
<MudText Typo="Typo.h5">@upperKey</MudText> <MudText Typo="Typo.h5">@upperKey</MudText>
</MudButton> </MudButton>
} }
@ -49,13 +50,9 @@
[Parameter] [Parameter]
public char[] Keys { get; set; } = Array.Empty<char>(); public char[] Keys { get; set; } = Array.Empty<char>();
private List<char> _correctKeys = new();
private List<char> _misplacedKeys = new();
private List<char> _wrongKeys = new();
private Dictionary<char, WordleComponent.CharState> _knownKeyStates = new();
[Parameter] [Parameter]
public int KeysPerRow { get; set; } = 10; public int KeysPerRow { get; set; } = 10;
@ -69,4 +66,36 @@
{ {
await KeyPressed.InvokeAsync(c); await KeyPressed.InvokeAsync(c);
} }
public void UpdateKeyInfo(WordleComponent.CharData cd)
{
_knownKeyStates.TryGetValue(cd.Char, out var oldState);
WordleComponent.CharState newState = WordleComponent.CharState.None;
switch (cd.State)
{
case WordleComponent.CharState.Correct:
newState = WordleComponent.CharState.Correct;
break;
case WordleComponent.CharState.Misplaced:
// Den besseren Zustand nicht durch schlechteren ersetzen
if (oldState != WordleComponent.CharState.Correct)
{
newState = WordleComponent.CharState.Misplaced;
}
break;
case WordleComponent.CharState.Wrong:
newState = WordleComponent.CharState.Wrong;
break;
}
_knownKeyStates[cd.Char] = newState;
}
public void ResetKeyInfo()
{
_knownKeyStates.Clear();
StateHasChanged();
}
} }

View File

@ -17,4 +17,8 @@
<PackageReference Include="MudBlazor" Version="6.0.7" /> <PackageReference Include="MudBlazor" Version="6.0.7" />
</ItemGroup> </ItemGroup>
<ItemGroup>
<ProjectReference Include="..\GottfriedsNackenUtility\GottfriedsNackenUtility.csproj" />
</ItemGroup>
</Project> </Project>

View File

@ -1,29 +1,50 @@
@page "/numberle" @page "/numberle"
@using GottfriedsNackenWebseite.Components @using GottfriedsNackenWebseite.Components
@using System.Diagnostics
@using GottfriedsNackenWebseite.Numberle @using GottfriedsNackenWebseite.Numberle
@using GottfriedsNackenUtility
@using System.Diagnostics
@using System.Numerics
<PageTitle>Nacken Numberle</PageTitle> <PageTitle>Nacken Numberle</PageTitle>
<MudText Align="Align.Center" Class="mt-2 mb-2"> Gebe eine (korrekte) mathematische Gleichung ein:</MudText> <MudText Align="Align.Center" Class="mt-2 mb-2"> Gebe eine (korrekte) mathematische Gleichung ein:</MudText>
<WordleComponent ColumnCount="8" GenerateSecret="GenerateSecret" InputMask="@_inputMask" CheckInput="CheckInput" @ref="_wordle" /> <WordleComponent ColumnCount="8" GenerateSecret="OnGenerateNewSecret" InputMask="@_inputMask" CheckInput="CheckInput" InputValidated="OnInputValidated" @ref="_wordle" />
<WordleKeyboardComponent KeyPressed="KeyboardKeyPressed" Keys="Chars" /> <WordleKeyboardComponent @ref="_keyboard" KeyPressed="KeyboardKeyPressed" Keys="Chars" />
<MudOverlay @bind-Visible="_showLoadingOverlay" DarkBackground="true">
<MudCard>
<MudCardHeader>
<MudText Typo="Typo.h5">Bitte warten...</MudText>
</MudCardHeader>
<MudCardContent>
<MudText>Erzeuge eine unknackbare Formel mit roher Gewalt!</MudText>
<MudProgressLinear Color="Color.Secondary" Indeterminate="_progress < 0" Value="_progress" Class="my-7"/>
</MudCardContent>
</MudCard>
</MudOverlay>
@code @code
{ {
private IMask _inputMask = new RegexMask(@"^[0-9+\-*/=]{0,8}$"); private static readonly IMask _inputMask = new RegexMask(@"^[0-9+\-*/=]{0,8}$");
private Parser _parser = new(); private readonly Parser _parser = new();
private WordleComponent _wordle; private WordleComponent _wordle = default!;
private WordleKeyboardComponent _keyboard = default!;
private bool _showLoadingOverlay;
private int _progress;
protected override async Task OnInitializedAsync() protected override async Task OnInitializedAsync()
{ {
await base.OnInitializedAsync(); await base.OnInitializedAsync();
} }
/// <summary>
/// Prüft ob input eine gültige Gleichung ist.
/// </summary>
private bool CheckInput(string input) private bool CheckInput(string input)
{ {
try try
@ -36,7 +57,7 @@
(decimal value, int equalSigns) = CrunchyValidaty(ast); (decimal value, int equalSigns) = CrunchyValidaty(ast);
// Wir akzeptieren nur, wenn es exakt ein Gleichheitszeichen gibt und diese erfüllt ist. // Wir akzeptieren nur, wenn es exakt ein Gleichheitszeichen gibt und die Gleichung erfüllt ist.
return value == 1 && equalSigns == 1; return value == 1 && equalSigns == 1;
} }
catch catch
@ -84,12 +105,55 @@
return (0, 0); return (0, 0);
} }
/// <summary>
/// Die Zeichen aus denen eine Antwort besteht.
/// </summary>
private static readonly char[] Chars = { private static readonly char[] Chars = {
'=', '+', '-', '*', '/', '=', '+', '-', '*', '/',
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9' '0', '1', '2', '3', '4', '5', '6', '7', '8', '9'
}; };
private string GenerateSecret() /// <summary>
/// Generiert ein neues Geheimnis und zeigt während dessen den Ladebalken an.
/// </summary>
/// <returns></returns>
private async Task OnGenerateNewSecret()
{
_keyboard.ResetKeyInfo();
CancellationTokenSource cts = new CancellationTokenSource();
Task? amazingTask = null;
try
{
// Ladebalkenaktualisierer starten
amazingTask = SusHelper.UpdateDefinitelyNotFakeProgress(progress =>
{
_progress = progress;
StateHasChanged();
}, cts.Token);
_showLoadingOverlay = true;
_wordle.Secret = await Task.Run(GenerateFormula);
_showLoadingOverlay = false;
}
finally
{
// Ladebalkenaktualisierer stoppen
cts.Cancel();
if (amazingTask != null)
await amazingTask;
}
}
/// <summary>
/// Erzeugt eine neue (gültige) Formel.
/// </summary>
/// <returns>Die Formel</returns>
private string GenerateFormula()
{ {
int charCount = 8; int charCount = 8;
@ -99,6 +163,8 @@
int equalsIndex = Random.Shared.Next(1, charCount - 1); int equalsIndex = Random.Shared.Next(1, charCount - 1);
formula[equalsIndex] = '='; formula[equalsIndex] = '=';
// Ersetzt das Zeichen mit Index "index" durch einen Zufälligen wert.
// TODO: ein etwas gewiefterer Algorithmus würde nicht schaden.
void PlaceRandomChar(int index) void PlaceRandomChar(int index)
{ {
char left = (index > 0) ? formula[index - 1] : '\0'; char left = (index > 0) ? formula[index - 1] : '\0';
@ -127,13 +193,26 @@
Debug.WriteLine(str); Debug.WriteLine(str);
if (CheckInput(str)) if (CheckInput(str))
{
return str; return str;
}
} }
} }
} }
private async Task KeyboardKeyPressed(char key) private async Task KeyboardKeyPressed(char key)
{ {
await _wordle.SetText(_wordle.Input + key); //Input += key; // TODO: Virtuelles Keyboard
//await _wordle.SetText(_wordle.Input + key); //Input += key;
await Task.CompletedTask;
}
private void OnInputValidated(WordleComponent.CharData[] data)
{
// Auswertung in das Keyboard packen
foreach (WordleComponent.CharData cd in data)
{
_keyboard.UpdateKeyInfo(cd);
}
} }
} }