Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 10개 있습니다.)
Windows: 148. Windows - Raw Input의 Top level collection 의미
; https://www.sysnet.pe.kr/2/0/11612

.NET Framework: 788. RawInput을 이용한 키보드/마우스 입력 모니터링
; https://www.sysnet.pe.kr/2/0/11615

개발 환경 구성: 488. (User-mode 코드로 가상 USB 장치를 만들 수 있는) USB/IP PROJECT 소개
; https://www.sysnet.pe.kr/2/0/12213

개발 환경 구성: 490. C# - (Wireshark의) USBPcap을 이용한 USB 패킷 모니터링
; https://www.sysnet.pe.kr/2/0/12215

.NET Framework: 904. USB/IP PROJECT를 이용해 C#으로 USB Keyboard 가상 장치 만들기
; https://www.sysnet.pe.kr/2/0/12216

.NET Framework: 905. C# - DirectX 게임 클라이언트 실행 중 키보드 입력을 감지하는 방법
; https://www.sysnet.pe.kr/2/0/12218

.NET Framework: 917. C# - USB 관련 ETW(Event Tracing for Windows)를 이용한 키보드 입력을 감지하는 방법
; https://www.sysnet.pe.kr/2/0/12246

.NET Framework: 990. C# - SendInput Win32 API를 이용한 가상 키보드/마우스
; https://www.sysnet.pe.kr/2/0/12469

.NET Framework: 1062. Windows Forms - 폼 내에서 발생하는 마우스 이벤트를 자식 컨트롤 영역에 상관없이 수신하는 방법
; https://www.sysnet.pe.kr/2/0/12660

개발 환경 구성: 607. 로컬의 USB 장치를 원격 머신에 제공하는 방법 - usbip-win
; https://www.sysnet.pe.kr/2/0/12858




Windows Forms - 폼 내에서 발생하는 마우스 이벤트를 자식 컨트롤 영역에 상관없이 수신하는 방법

아래와 같은 질문과 그 대답이 있군요. ^^

마우스 이벤트 관련 질문 좀 드리겠습니다.
; https://www.sysnet.pe.kr/3/0/5508

마우스 이벤트 관련 질문 좀 드리겠습니다
; https://forum.dotnetdev.kr/t/topic/1061

그래서 다음과 같이 코딩하면 아주 잘 동작합니다.

using System;
using System.Windows.Forms;

namespace WindowsFormsApp1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        protected override void WndProc(ref Message m)
        {
            if (m.Msg == Win32Interop.WM_PARENTNOTIFY  // WM_PARENTNOTIFY == 0x210
                && m.WParam.ToInt32() == Win32Interop.WM_LBUTTONDOWN) // WM_LBUTTONDOWN == 0x0201
            {
                System.Diagnostics.Trace.WriteLine($"{DateTime.Now} WndProc - LeftDown");
            }

            if (m.Msg == Win32Interop.WM_LBUTTONDOWN) // WM_LBUTTONDOWN == 0x0201
            {
                System.Diagnostics.Trace.WriteLine($"{DateTime.Now} WndProc - LeftDown");
            }

            base.WndProc(ref m);
        }
    }
}

첫 번째 if 문의 코드는 자식 컨트롤의 영역에서 마우스 좌 클릭을 했을 때 발생하고, 두 번째 if 문은 (자식 컨트롤이 없는) Form 위에서 했을 때 발생합니다.




기왕 하는 김에, 전에 알아봤던 RawInput도 함께 실습을 해봤습니다. (물론, WndProc을 이용해 구현하는 것이 훨씬 더 좋습니다. ^^)

RawInput을 이용한 키보드/마우스 입력 모니터링
; https://www.sysnet.pe.kr/2/0/11615

이번에는 전역 모니터링이 아닌, 해당 윈도우가 전경으로 실행 중일 때만 마우스 입력을 받기만 하면 되므로 초기화를 다음과 같은 식으로 할 수 있습니다.

private unsafe void Form1_Load(object sender, EventArgs e)
{
    RAWINPUTDEVICE[] Rid = new RAWINPUTDEVICE[1];

    Rid[0].UsagePage = HIDUsagePage.Generic;
    Rid[0].Usage = HIDUsage.Mouse;
    Rid[0].Flags = RawInputDeviceFlags.None;
    Rid[0].WindowHandle = IntPtr.Zero;

    if (Win32Interop.RegisterRawInputDevices(Rid, 1, sizeof(RAWINPUTDEVICE)) == false)
    {
        MessageBox.Show("Failed to register");
    }
}

그런 다음, IMessageFilter 구현 코드를 통해 다음과 같이 WM_LBUTTONDOWN을 감지할 수 있습니다.


using System;
using System.Windows.Forms;

namespace WindowsFormsApp1
{
    public partial class Form1 : Form, IMessageFilter
    {
        public Form1()
        {
            InitializeComponent();
            Application.AddMessageFilter(this);
        }

        private unsafe void Form1_Load(object sender, EventArgs e)
        {
            // ...[초기화 생략]...
        }

        protected override void OnFormClosing(FormClosingEventArgs e)
        {
            Application.RemoveMessageFilter(this);
            base.OnFormClosing(e);
        }

        public bool PreFilterMessage(ref Message m)
        {
            if (m.Msg == Win32Interop.WM_INPUT)
            {
                RawInput raw = Win32Interop.GetDeviceID(m);

                switch (raw.Header.Type)
                {
                    case RawInputType.Mouse:
                        if (raw.Mouse.ButtonFlags == RawMouseButtons.LeftDown)
                        {
                            System.Diagnostics.Trace.WriteLine($"{DateTime.Now} RawInput - LeftDown");
                        }
                        break;
                }
            }

            return false;
        }
    }
}

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/3/2021]

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

비밀번호

댓글 작성자
 



2021-06-03 11시12분
감사합니다.
많은 공부와 가르침 받았습니다.
초심으로

... 91  92  93  94  95  96  97  98  99  100  [101]  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11440정성태1/17/201820606.NET Framework: 727. ASP.NET의 HttpContext.Current 구현에 대응하는 ASP.NET Core의 IHttpContextAccessor/HttpContextAccessor 사용법파일 다운로드1
11439정성태1/17/201826286기타: 69. C# - CPU 100% 부하 주는 프로그램파일 다운로드1
11438정성태1/17/201820196오류 유형: 446. Error CS0234 The type or namespace name 'ITuple' does not exist in the namespace
11437정성태1/17/201819766VS.NET IDE: 124. Platform Toolset 설정에 따른 Visual C++의 헤더 파일 기본 디렉터리
11436정성태1/16/201822176개발 환경 구성: 352. ASP.NET Core (EXE) 프로세스가 IIS에서 호스팅되는 방법 - ASP.NET Core Module(AspNetCoreModule) [4]
11435정성태1/16/201823260개발 환경 구성: 351. OWIN 웹 서버(EXE)를 IIS에서 호스팅하는 방법 - HttpPlatformHandler (Reverse Proxy)파일 다운로드2
11434정성태1/15/201823906개발 환경 구성: 350. 사용자 정의 웹 서버(EXE)를 IIS에서 호스팅하는 방법 - HttpPlatformHandler (Reverse Proxy)파일 다운로드2
11433정성태1/15/201822070개발 환경 구성: 349. dotnet ef 명령어 사용을 위한 준비
11432정성태1/11/201828486.NET Framework: 726. WPF + Direct2D + SharpDX 출력 C# 예제파일 다운로드2
11431정성태1/11/201825635.NET Framework: 725. C# - 동기 방식이면서 비동기 메서드(awaitable)처럼 구현한 사례 [9]
11430정성태1/10/201829106.NET Framework: 724. WPF + Direct2D 출력 C# 예제 [2]파일 다운로드1
11429정성태1/9/201819662개발 환경 구성: 348. ASP.NET Core 2.1 Preview 버전 적용 방법
11428정성태1/6/201822874개발 환경 구성: 347. WinForm 프로젝트를 WPF 프로젝트 유형으로 변경하는 방법파일 다운로드1
11427정성태1/5/201820217오류 유형: 445. vcpkg 빌드 오류 - Starting the CLR failed with HRESULT 80040153
11426정성태1/5/201830431오류 유형: 444. curl로 호출할 때 발생하는 오류 정리
11425정성태1/4/201821071개발 환경 구성: 346. ASP.NET Core Web Application을 IIS에서 호스팅하는 방법 (2)
11424정성태1/4/201820512개발 환경 구성: 345. ASP.NET Core 프로젝트를 명령행에서 빌드하는 방법
11423정성태1/3/201838956VC++: 123. 내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유 [8]파일 다운로드2
11422정성태1/2/201829063.NET Framework: 723. C# - OpenCvSharp 사용 시 C/C++을 이용한 속도 향상 (for 루프 연산) [4]파일 다운로드1
11421정성태1/2/201821293오류 유형: 443. Visual Studio - nuget configuration is invalid
11420정성태12/30/201725386.NET Framework: 722. C# - Windows 10 운영체제의 데스크톱 앱에서 음성인식(SpeechRecognizer) 사용하는 방법 [3]파일 다운로드1
11419정성태12/23/201727554.NET Framework: 721. WebClient 타입의 ...Async 메서드 호출은 왜 await + 동기 호출 시 hang 현상이 발생할까요? [2]파일 다운로드1
11418정성태12/23/201737270.NET Framework: 720. 비동기 메서드 내에서 await 시 ConfigureAwait 호출 의미 [2]파일 다운로드1
11417정성태12/22/201723190.NET Framework: 719. Task를 포함하는 async 메서드의 동작 방식 [2]
11416정성태12/21/201720301.NET Framework: 718. AsyncTaskMethodBuilder.Create() 메서드 동작 방식 [2]
11415정성태12/21/201722166.NET Framework: 717. Task를 포함하지 않는 async 메서드의 동작 방식 [6]
... 91  92  93  94  95  96  97  98  99  100  [101]  102  103  104  105  ...