KeyEventArgs.Key для char

Есть ли способ конвертировать WPF KeyEventArgs.Key в Char?

Я попытался использовать KeyInterop:

var x = (Char)KeyInterop.VirtualKeyFromKey(e.Key);

Для чисел и букв он отлично работает, но для других символов это не так. Например. для OemComma он возвращает '1/4' вместо ','.

Мне нужно, чтобы пользователь не ввел в TextBox любые символы, кроме чисел и разделителей (т.е. запятые или периоды).

Ответ 1

Нашел это где-то, что я забыл. Все еще используя его, не знаете, какие у него есть отверстия, но он работал, насколько я знаю.

public static class KeyEventUtility
{
    // ReSharper disable InconsistentNaming
    public enum MapType : uint
    {
        MAPVK_VK_TO_VSC = 0x0,
        MAPVK_VSC_TO_VK = 0x1,
        MAPVK_VK_TO_CHAR = 0x2,
        MAPVK_VSC_TO_VK_EX = 0x3,
    }
    // ReSharper restore InconsistentNaming

    [DllImport( "user32.dll" )]
    public static extern int ToUnicode(
        uint wVirtKey,
        uint wScanCode,
        byte[] lpKeyState,
        [Out, MarshalAs( UnmanagedType.LPWStr, SizeParamIndex = 4 )] 
        StringBuilder pwszBuff,
        int cchBuff,
        uint wFlags );

    [DllImport( "user32.dll" )]
    public static extern bool GetKeyboardState( byte[] lpKeyState );

    [DllImport( "user32.dll" )]
    public static extern uint MapVirtualKey( uint uCode, MapType uMapType );

    public static char GetCharFromKey( Key key )
    {
        char ch = ' ';

        int virtualKey = KeyInterop.VirtualKeyFromKey( key );
        var keyboardState = new byte[256];
        GetKeyboardState( keyboardState );

        uint scanCode = MapVirtualKey( (uint)virtualKey, MapType.MAPVK_VK_TO_VSC );
        var stringBuilder = new StringBuilder( 2 );

        int result = ToUnicode( (uint)virtualKey, scanCode, keyboardState, stringBuilder, stringBuilder.Capacity, 0 );
        switch ( result )
        {
        case -1:
            break;
        case 0:
            break;
        case 1:
            {
                ch = stringBuilder[0];
                break;
            }
        default:
            {
                ch = stringBuilder[0];
                break;
            }
        }
        return ch;
    }
}

Ответ 2

Почему такой подход? Существуют разные способы запретить пользователям выполнять такой ввод. Например, определите обработчик событий для TextBox.TextChanged -event и когда вход не является десятичным или разделителем, отмените ввод.

private void textChangedEventHandler(object sender, TextChangedEventArgs args){
//args contains a property `Changes` where you can see what happened
//check for invalid characters and revoke them
}

Смотрите TextChangedEventArgs и событие.

Ответ 3

Мое первое предложение будет использовать PreviewTextInput и использовать TextCompositionEventArgs. Однако это "ест" пробельные символы. Для этого "космического" вопроса см.:

другое решение будет использовать KeyConverter в событии KeyUp:

  KeyConverter kc = new KeyConverter();
  var str = kc.ConvertToString(e.Key);

Для WPF: https://msdn.microsoft.com/en-us/library/system.windows.input.keyconverter(v=vs.110).aspx Для Winforms: https://msdn.microsoft.com/en-us/library/system.windows.forms.keysconverter.aspx

Обратите внимание, что KeyConverter выдаст вам строку "Пробел" вместо ". (то же самое для" Esc" при нажатии клавиши эвакуации.

Другое решение WPF использует это в событии KeyUp:

((char)KeyInterop.VirtualKeyFromKey(e.Key))