Microsoft MVP성태의 닷넷 이야기
기타: 25. 가상 키보드 관련 정리 [링크 복사], [링크+제목 복사],
조회: 39286
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

가상 키보드 관련 정리


가상 키보드라... 사실 이런 프로그램을 만들 일은 거의 없죠. 그래서 저도 그와 관련한 몇 가지 특징들은 아예 모르고 살고 있었는데요. 하다 보니 제법 재미있는 사항들이 몇 가지 눈에 띄어 정리해 봅니다.

1. 포커스를 어떻게 유지하는가?


개인적으로... 이 기능이야말로 가상 키보드의 구현 핵심(?)이 아닐까 싶은데요. 예를 들어, 입력 포커스가 텍스트 박스에 있는데, 가상 키보드가 구현된 윈도우의 영역을 클릭하게 되면 포커스는 이미 새롭게 활성화된 윈도우로 바뀌어 버리고 그 상태에서 키입력을 발생해 봤자 가상 키보드를 소유한 윈도우 자체가 받게 됩니다.

'혹시... 이전에 포커스를 가진 윈도우를 기억해 두었다가, 다시 활성화를 시키고 키 입력을 발생시키는 것이 아닐까?'라고 생각했는데, 이미 이러한 입력 포커스 관련한 윈도우 메시지와 스타일이 잘 정의되어 있더군요.

윈도우 메시지를 통한 방식은, 하나의 EXE 프로세스 내에서 생성된 윈도우들 내에서만 유효한 방식인데 WM_MOUSEACTIVATE 메시지가 전송되어지면 이에 대해 MA_NOACTIVATE 값을 반환하는 방식입니다.

NO focus on Gui 
; http://www.autohotkey.com/forum/topic14522.html

다른 하나는, EXE들 간에서도 포커스를 갖지 않도록 하는 것인데, 이건 메시지를 처리하는 방식으로는 안 되고 아예 해당 윈도우 자체를 생성할 때 라는 확장 윈도우 스타일을 지정해 주어야 합니다.

Preventing "owner" Form from losing focus when showing another Form?
; http://social.msdn.microsoft.com/forums/en-US/winforms/thread/3f1c4191-38f4-4d15-a633-9f57cff289e4/

참고로, WPF의 경우에 제공되는 "UIElement.Focusable"은 또 다른 범위의 옵션을 제공하는데요. 이걸 사용하면 "단일 윈도우" 내에서의 포커스를 갖지 않게 됩니다.

2. WPF에서의 가상키 처리


이전의 System.Windows.Forms.Keys에서 제공되는 상수값들이 가상 키(Virtual Key) 코드값이었던 반면, WPF에서 제공되는 키 상수 열거 타입인 "System.Windows.Input.Key"는 애석하게도 가상 키 코드값이 아닙니다. 대신에 이 둘 사이의 변환을 해줄 수 있는 KeyInterop.VirtualKeyFromKey 메서드가 제공됩니다.

3. 문자에 대한 가상 키 코드값 구하기


예를 들어, 'a' 키에 대해 그에 대한 가상 키 값을 구하고 싶은 경우인데. 제가 아는 범위 내에서는 명시적으로 닷넷에서 제공되는 함수는 없고 Win32 API 중에 VkKeyScan 메서드를 P/Invoke로 호출해 줘야 합니다.

4. Virtual Key와 Scan 코드 값끼리의 변환


이것 역시 닷넷에서 제공되는 함수는 없고, Win32 API 중에 MapVirtualKey 메서드를 이용하여 상호 변환을 할 수 있습니다.

5. 특수 문자를 입력하려면?


특수 문자인 경우, 처리가 특정 code page에 종속되게 하고 싶지 않다면... 아마도 유일한 대안인 "unicode"를 사용하는 수밖에는 없지요. 우선, 자신이 사용하고 싶은 특수 문자가 유니코드로 어떤 값에서 표현되고 있는지 찾아야 합니다. 예를 들어, 다음과 같은 사이트가 도움이 되겠지요.

Unicode Character Map
; http://www.bazon.net/mishoo/charmap/

가령, "θ"와 같은 문자는 0x3b8이란 코드 값을 가지고 있습니다. 이 키 코드값을 Virtual Key로 변환할 필요 없이 바로 SendInput의 메서드에 전달해주어 처리할 수 있습니다.

KEYBOARDINPUT [] keyInput = new KEYBOARDINPUT[2]; // 누르는 동작과 떼는 동작을 표현하기 위해 배열은 2개

// 누르는 동작
keyInput[0].type = 1; // 1 == INPUT_KEYBOARD
keyInput[0].dwFlags = 0x04; // 0x04 : KEYEVENTF_UNICODE
keyInput[0].wScan = 0x3b8;

// 떼는 동작
keyInput[1].type = 1;
keyInput[1].dwFlags = 0x04 | 0x02; // 0x02 : KEYEVENTF_KEYUP
keyInput[1].wScan = 0x3b8;

SendInput(2, keyInput, 0x1c); // 0x1C : KEYBOARDINPUT 2개 요소의 크기



[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 9/20/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2011-03-06 09시00분
Windows Input Simulator (C# SendInput Wrapper - Simulate Keyboard and Mouse)
; http://inputsimulator.codeplex.com/
정성태
2014-02-13 06시01분
When something gets added to a queue, it takes time for it to come out the front of the queue
; http://blogs.msdn.com/b/oldnewthing/archive/2014/02/13/10499047.aspx
정성태
2018-03-22 12시00분
C# INPUT Interop

        [StructLayout(LayoutKind.Explicit)]
        public struct INPUT
        {
            [FieldOffset(0)]
            public Int32 type;
#if _X64_
            [FieldOffset(8)] //*
            public MOUSEINPUT mi;
            [FieldOffset(8)] //*
            public KEYBDINPUT ki;
            [FieldOffset(8)] //*
            public HARDWAREINPUT hi;
#else
            [FieldOffset(4)] //*
            public MOUSEINPUT mi;
            [FieldOffset(4)] //*
            public KEYBDINPUT ki;
            [FieldOffset(4)] //*
            public HARDWAREINPUT hi;
#endif
        }
정성태
2019-05-15 09시11분
주제와는 다르지만, WPF의 경우 EventManager.RegisterClassHandler를 이용해 키보드 입력을 처리할 수 있습니다.

Capture all Keyboard input, regardless of what control has focus
; https://social.msdn.microsoft.com/Forums/vstudio/ko-KR/cf884a91-c135-447d-b16b-214d2d9e9972/capture-all-keyboard-input-regardless-of-what-control-has-focus?forum=wpf

---------------------------------------------------------------
EventManager.RegisterClassHandler(typeof(Window), Keyboard.KeyUpEvent,new KeyEventHandler(keyUp), true);

private void keyUp(object sender, KeyEventArgs e)
{
      if(e.Key == Key.OemComma) MessageBox.Show("YAY!!!");
}
정성태

... 106  107  108  109  110  111  112  113  114  115  116  117  118  119  [120]  ...
NoWriterDateCnt.TitleFile(s)
10924정성태3/22/201621124오류 유형: 324. Visual Studio에서 Azure 클라우드 서비스 생성 시 Failed to initialize the PowerShell host 에러 발생
10923정성태3/21/201622143.NET Framework: 564. C# - DGML로 바이너리 트리 출력하는 방법 [1]파일 다운로드1
10922정성태3/21/201622568.NET Framework: 563. 디버깅 용도로 이진 트리의 내용을 출력하는 방법파일 다운로드1
10921정성태3/17/201625547.NET Framework: 562. BBI 인터프리터 C/C++ 코드를 C#으로 변환 [3]파일 다운로드2
10920정성태3/15/201627151.NET Framework: 561. null 처리된 객체가 왜 GC에 의해 수집되지 않을까요? [6]파일 다운로드1
10919정성태3/12/201623176.NET Framework: 560. C#에서 return할 때 명시적으로 casting한 것과 안한 것의 차이 [2]파일 다운로드1
10918정성태3/10/201619872.NET Framework: 559. WPF - ICommand.CanExecuteChanged가 해제되지 않는 문제 [2]파일 다운로드1
10917정성태3/10/201640402.NET Framework: 558. WPF - ICommand 동작 방식 [9]파일 다운로드1
10916정성태3/9/201626199.NET Framework: 557. 머신 바이트 배열로부터 역어셈블해주는 라이브러리 - Udis86 Assembler파일 다운로드2
10915정성태3/9/201621790오류 유형: 323. FatalExecutionEngineError was detected
10914정성태3/8/201625188오류 유형: 322. 정적 라이브러리 참조 시 "LNK2019 unresolved external symbol '...' referenced in function" 오류 발생파일 다운로드1
10913정성태3/7/201625112.NET Framework: 556. C#으로 다루는 MBR(Master Boot Record) [9]파일 다운로드1
10912정성태3/2/201621968.NET Framework: 555. List<T>의 Resize 메서드 구현 [2]파일 다운로드1
10911정성태2/29/201625868Math: 15. 그래프 그리기로 알아보는 뉴턴-랩슨(Newton-Raphson's method)법과 제곱근 구하기 - C#파일 다운로드1
10910정성태2/29/201627201Math: 14. HTML에서 수학 관련 기호/수식을 표현하기 위한 방법 - MathJax.js - 두 번째 이야기 [5]
10909정성태2/25/201625438기타: 56. ETW provider 목록 [3]
10908정성태2/25/201622190기타: 55. ETW man 파일 목록
10907정성태2/24/201620634.NET Framework: 554. 인터프리터 - 재귀적 하향 구문 분석 C# 예제파일 다운로드1
10906정성태2/24/201619502.NET Framework: 553. C# 관리 코드에서 IMetaDataDispenserEx, IMetaDataImport 관련 인터페이스를 얻는 방법파일 다운로드1
10905정성태2/24/201623041오류 유형: 321. Hyper-V The operation failed with error code '32791'.
10904정성태2/23/201619431.NET Framework: 552. 인터프리터 - 역폴란드 표기법을 이용한 식의 분석 - C# 예제파일 다운로드1
10903정성태2/22/201620859.NET Framework: 551. 인터프리터 어휘 분석 프로그램 - C# 예제파일 다운로드1
10902정성태2/22/201620756.NET Framework: 550. GetFunctionPointer 호출 시 System.InvalidProgramException 예외 발생
10901정성태2/20/201622888.NET Framework: 549. ContextBoundObject 상속 클래스와 System.Reflection.ReflectionTypeLoadException 예외 [4]파일 다운로드1
10900정성태2/19/201622057.NET Framework: 548. Linq는 결국 메서드 호출! [3]파일 다운로드1
10899정성태2/17/201623386개발 환경 구성: 282. kernel32.dll, kernel32legacy.dll, api-ms-win-core-sysinfo-l1-2-0.dll [1]
... 106  107  108  109  110  111  112  113  114  115  116  117  118  119  [120]  ...