Microsoft MVP성태의 닷넷 이야기
기타: 25. 가상 키보드 관련 정리 [링크 복사], [링크+제목 복사]
조회: 24502
글쓴 사람
홈페이지
첨부 파일
 

가상 키보드 관련 정리


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

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


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

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

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

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

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

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개 요소의 크기



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

[연관 글]





[최초 등록일: ]
[최종 수정일: 12/29/2008 ]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer@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!!!");
}
정성태

1  [2]  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
12179정성태3/10/202067개발 환경 구성: 481. docker - PostgreSQL 컨테이너 실행
12178정성태3/10/202049개발 환경 구성: 480. Linux 운영체제의 docker를 위한 tcp 바인딩 추가
12177정성태3/9/202080개발 환경 구성: 479. docker - MySQL 컨테이너 실행
12176정성태3/9/202041개발 환경 구성: 478. 파일의 (sha256 등의) 해시 값(checksum) 확인하는 방법
12175정성태3/8/202096개발 환경 구성: 477. "Docker Desktop for Windows"의 "Linux Container" 모드를 위한 tcp 바인딩 추가
12174정성태3/8/2020126개발 환경 구성: 476. DockerDesktopVM의 파일 시스템 접근 [1]
12173정성태3/8/2020121개발 환경 구성: 475. docker - SQL Server 2019 컨테이너 실행 [1]
12172정성태3/8/202074개발 환경 구성: 474. docker - container에서 root 권한 명령어 실행(sudo)
12171정성태3/6/2020122VS.NET IDE: 143. Visual Studio - ASP.NET Core Web Application의 "Enable Docker Support" 옵션으로 달라지는 점
12170정성태3/6/202067오류 유형: 599. "Docker Desktop is switching..." 메시지와 DockerDesktopVM CPU 소비 현상
12169정성태3/5/2020123개발 환경 구성: 473. Windows nanoserver에 대한 docker pull의 태그 사용
12168정성태3/8/2020154개발 환경 구성: 472. 윈도우 환경에서의 dockerd.exe("Docker Engine" 서비스)가 Linux의 것과 다른 점
12167정성태3/5/202056개발 환경 구성: 471. C# - 닷넷 응용 프로그램에서 DB2 Express-C 데이터베이스 사용 (3) - ibmcom/db2express-c 컨테이너 사용
12166정성태3/14/202096개발 환경 구성: 470. Windows Server 컨테이너 - DockerMsftProvider 모듈을 이용한 docker 설치
12165정성태3/2/2020101.NET Framework: 900. 실행 시에 메서드 가로채기 - CLR Injection: Runtime Method Replacer 개선 - 세 번째 이야기(Monitor.Enter 후킹)파일 다운로드1
12164정성태2/29/202069오류 유형: 598. Surface Pro 6 - Windows Hello Face Software Device가 인식이 안 되는 문제
12163정성태2/27/2020134.NET Framework: 899. 익명 함수를 가리키는 delegate 필드에 대한 직렬화 문제
12162정성태2/28/2020143디버깅 기술: 166. C#에서 만든 COM 객체를 C/C++로 P/Invoke Interop 시 메모리 누수(Memory Leak) 발생파일 다운로드2
12161정성태2/26/202044오류 유형: 597. manifest - The value "x64" of attribute "processorArchitecture" in element "assemblyIdentity" is invalid.
12160정성태2/26/202082개발 환경 구성: 469. Reg-free COM 개체 사용을 위한 manifest 파일 생성 도구 - COMRegFreeManifest
12159정성태2/26/202046오류 유형: 596. Visual Studio - The project needs to include ATL support
12158정성태2/26/202095디버깅 기술: 165. C# - Marshal.GetIUnknownForObject/GetIDispatchForObject 사용 시 메모리 누수(Memory Leak) 발생파일 다운로드1
12157정성태2/27/2020103디버깅 기술: 164. C# - Marshal.GetNativeVariantForObject 사용 시 메모리 누수(Memory Leak) 발생 및 해결 방법파일 다운로드1
12156정성태2/25/202066오류 유형: 595. LINK : warning LNK4098: defaultlib 'nafxcw.lib' conflicts with use of other libs; use /NODEFAULTLIB:library
12155정성태2/25/202069오류 유형: 594. Warning NU1701 - This package may not be fully compatible with your project
12154정성태2/25/202065오류 유형: 593. warning LNK4070: /OUT:... directive in .EXP differs from output filename
1  [2]  3  4  5  6  7  8  9  10  11  12  13  14  15  ...