성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] 그냥 RSS Reader 기능과 약간의 UI 편의성 때문에 사용...
[이종효] 오래된 소프트웨어는 보안 위협이 되기도 합니다. 혹시 어떤 기능...
[정성태] @Keystroke IEEE의 문서를 소개해 주시다니... +_...
[손민수 (Keystroke)] 괜히 듀얼채널 구성할 때 한번에 같은 제품 사라고 하는 것이 아...
[정성태] 전각(Full-width)/반각(Half-width) 기능을 토...
[정성태] Vector에 대한 내용은 없습니다. Vector가 닷넷 BCL...
[orion] 글 읽고 찾아보니 디자인 타임에는 InitializeCompon...
[orion] 연휴 전에 재현 프로젝트 올리자 생각해 놓고 여의치 않아서 못 ...
[정성태] 아래의 글에 정리했으니 참고하세요. C# - Typed D...
[정성태] 간단한 재현 프로젝트라도 있을까요? 저런 식으로 설명만 해...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>Win32 Message를 Code로부터 메시지 이름 자체를 텍스트로 구하고 싶다면?</h1> <p> 테스트를 하다 보면 가끔, C#에서 Win32 메시지를 다룰 때 코드값으로부터 메시지의 이름을 텍스트로 구하고 싶을 때가 있습니다.<br /> <br /> 가령, Message 코드가 0x0010인 경우 그것은 "WM_CLOSE"에 해당하는데,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > WM_CLOSE message ; <a target='tab' href='https://docs.microsoft.com/en-us/windows/win32/winmsg/wm-close'>https://docs.microsoft.com/en-us/windows/win32/winmsg/wm-close</a> </pre> <br /> 이 값을 찾으려면 C++ 빈 프로젝트를 하나 열어 Windows.h 헤더 파일을 포함시킨 후 winuser.h 파일로부터 그 값에 해당하는 메시지를 찾는 식으로 작업을 했는데요.<br /> <br /> 오늘은 갑자기 그게 귀찮다는 생각이 드는군요. ^^; 차라리, dictionary 타입으로 미리 구성해 놓고 싶어졌습니다. 가령 WinUser.h 파일을 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > ...[생략]... #define WM_SETTEXT 0x000C #define WM_GETTEXT 0x000D #define WM_GETTEXTLENGTH 0x000E #define WM_PAINT 0x000F #define WM_CLOSE 0x0010 ...[생략]... </pre> <br /> 위와 같은 식인데, 따라서 라인 하나를 읽어서 공백 문자로 분리한 후 3번째 숫자 값을 기준으로 2번째 WM_ 문자열을 매핑시켜 놓으면 되는 것입니다. 실제로 아래와 같은 정도로 간단하게 구할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > internal class Program { static void Main(string[] args) { StringBuilder sb = new StringBuilder(); sb.AppendLine("public class Win32MessageMap"); sb.AppendLine("{"); { sb.AppendLine("\tpublic static Dictionary<uint, string> codes = new ()"); sb.AppendLine("{"); foreach (string line in File.ReadAllLines(@"C:\Program Files (x86)\Windows Kits\10\Include\10.0.22000.0\um\WinUser.h")) { string text = line.Trim(); if (text.StartsWith("#define ") == false) { continue; } string[] tokens = text.Split(' ', StringSplitOptions.RemoveEmptyEntries); if (tokens.Length != 3) { continue; } if (tokens[1].StartsWith("WM_") == false) { continue; } sb.AppendLine($"\t\t[{tokens[2]}] = \"{tokens[1]}\","); } sb.AppendLine("\t};"); } sb.AppendLine("}"); Console.WriteLine(sb.ToString()); } } </pre> <br /> 그럼 다음과 같은 텍스트가 구해지므로,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public class Win32MessageMap { public static Dictionary<int, string> codes = new() { [0x0000] = "WM_NULL", [0x0001] = "WM_CREATE", [0x0002] = "WM_DESTROY", [0x0003] = "WM_MOVE", [0x0005] = "WM_SIZE", [0x0006] = "WM_ACTIVATE", [0x0007] = "WM_SETFOCUS", ...[생략]... [0x0360] = "WM_AFXFIRST", [0x037F] = "WM_AFXLAST", [0x0380] = "WM_PENWINFIRST", [0x038F] = "WM_PENWINLAST", [0x8000] = "WM_APP", [0x0400] = "WM_USER", }; } </pre> <br /> 복사해서 붙여넣기 하면 됩니다. ^^ 단지, 유일하게 하나의 예외 케이스가 나오는데요,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > #if(WINVER >= 0x0400) #define WM_SETTINGCHANGE WM_WININICHANGE #endif /* WINVER >= 0x0400 */ </pre> <br /> 위와 같은 경우 다음과 같이 출력이 되는데,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > [0x001A] = "WM_WININICHANGE", [WM_WININICHANGE] = "WM_SETTINGCHANGE", [0x001B] = "WM_DEVMODECHANGE", </pre> <br /> WINVER이 0x0400이라는 것은 WIN NT 4.0 버전을 의미하는데요, 따라서 근래에는 WM_WININICHANGE의 의미보다는 WM_SETTINGCHANGE가 더 맞으므로 그 부분만 이렇게 바꿔주면 되겠습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // [0x001A] = "WM_WININICHANGE", [0x001A] = "WM_SETTINGCHANGE", </pre> <br /> <hr style='width: 50%' /><br /> <br /> 기왕 하는 김에 Win32 메시지만을 담은 enum도 이런 식으로 나오게 출력할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public enum Win32Message : uint { // ...[생략]... WM_IME_COMPOSITION = 0x010F, WM_IME_KEYLAST = 0x010F, WM_INITDIALOG = 0x0110, // ...[생략]... } </pre> <br /> 위와 같이 enum을 생성하는 경우에는 이런 문제가 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > #if(_WIN32_WINNT >= 0x0501) #define WM_UNICHAR 0x0109 <span style='color: blue; font-weight: bold'>#define WM_KEYLAST 0x0109</span> #define UNICODE_NOCHAR 0xFFFF #else <span style='color: blue; font-weight: bold'>#define WM_KEYLAST 0x0108</span> </pre> <br /> _WIN32_WINNT 값이 0x0501이, 5.1 버전으로 Windows XP 이상을 의미하니까, 역시 WM_KEYLAST 값도 0x0109로 맞춰주시면 됩니다.<br /> <br /> 마찬가지로 WM_MOUSELAST 값도,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > #if (_WIN32_WINNT >= 0x0600) <span style='color: blue; font-weight: bold'>#define WM_MOUSELAST 0x020E</span> #elif (_WIN32_WINNT >= 0x0500) <span style='color: blue; font-weight: bold'>#define WM_MOUSELAST 0x020D</span> #elif (_WIN32_WINNT >= 0x0400) || (_WIN32_WINDOWS > 0x0400) <span style='color: blue; font-weight: bold'>#define WM_MOUSELAST 0x020A</span> #else <span style='color: blue; font-weight: bold'>#define WM_MOUSELAST 0x0209</span> #endif /* (_WIN32_WINNT >= 0x0600) */ </pre> <br /> _WIN32_WINNT 값이 0x0600은, 6.0 버전으로 Windows Vista와 Windows Server 2008에 해당하니까, 역시 근래의 환경을 생각하면 0x020E로 맞추면 되겠습니다.<br /> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1917&boardid=331301885'>첨부 파일은 이 글에서 생성한 enum과 Dictionary 클래스 코드와, 그것을 생성하는 예제 코드를 포함</a>합니다.)<br /> <br /> <hr style='width: 50%' /><br /> <br /> 참고로, 테스트를 하다 보면 Message 코드 값이 C0F4와 같은 값이 나오는 경우가 있습니다. 그런 경우는 RegisterWindowMessage를 이용해 등록한,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > RegisterWindowMessage ; <a target='tab' href='https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-registerwindowmessagea'>https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-registerwindowmessagea</a> </pre> <br /> 0xC000-0xFFFF 범위의 사용자 정의 메시지일 수 있습니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 위의 코드를 담아서 일일이 쓰기 귀찮은 분은, 해당 코드와 관련된 helper 클래스를 nuget 패키지에 추가했으니,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Install-Package CustomMessageLoop -Version 1.0.2 </pre> <br /> 이런 용도로 쓰면 되겠습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > void OnThreadFilterMessage(ref System.Windows.Interop.MSG msg, ref bool handled) { switch (msg.message) { case (int)<span style='color: blue; font-weight: bold'>Win32Message.WM_CLOSE</span>: break; } string text = <span style='color: blue; font-weight: bold'>Win32MessageMap.GetText</span>(msg.message); } </pre> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
7566
(왼쪽의 숫자를 입력해야 합니다.)