Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

왜? Button 컨트롤에는 MouseDown/MouseUp 이벤트가 발생하지 않을까요?


마침, 질문해 주신 분도 있고 ^^

Mouse이벤트 관련해서 질문있습니다.
; https://www.sysnet.pe.kr/3/0/983

오랜만에 WPF에 대한 감각도 되찾을 겸 ^^ 살펴보았답니다.

질문의 요지는, WPF의 "Button" 컨트롤에 MouseDown/MouseUp 이벤트를 걸어두었는데, 왜? 발생하지 않느냐는 것입니다.

다소 허무하긴 하지만, 검색해 보니 ^^ 금방 답이 나오더군요.

MouseDown event not working WPF
; http://social.msdn.microsoft.com/Forums/eu/wpf/thread/61807025-d4c4-41e0-b648-b11183065009

Here MouseDown is a bubbling event so it will be fired only when it won't be handled at child level.
 
To capture the same you need to use PreviewMouseDown event which is a tunneling event.
 
OR
 
Paste below line in constructor of Window1 class:
 
InitializeComponent();
AddHandler(FrameworkElement.MouseDownEvent, new MouseButtonEventHandler(Button_MouseDown), true);

WPF의 이벤트 순서는 다음의 글에서 잘 설명했듯이,

WPF Input Events
; https://docs.microsoft.com/en-us/dotnet/desktop/wpf/advanced/routed-events-overview#how_event_processing_works

제일 바깥쪽의 요소에서부터 안쪽으로 Preview... 유의 이벤트가 발생한 다음, 다시 반대로 안쪽에서부터 바깥쪽으로 Preview가 없는 이벤트가 발생합니다. 전자의 경우를 Tunneling이라고 하고, 후자의 경우를 Bubbling이라고 합니다. (이 외에, MouseEnter/Leave처럼 다른 요소로 전파되지 않는 Direct 유형이 있습니다.)

재미있는 점은, Bubbling 이벤트의 경우 해당 요소에서 "처리되었다는 표시"를 하면 더 이상 이벤트 전달을 하지 않습니다.

실제로 그런지 예제 코드를 한번 구성해 보겠습니다. 우선, WPF 프로젝트를 생성하고, TestElement라는 WPF UserControl을 하나 추가한 후 빌드를 합니다. 그럼, 도구 상자에 빌드된 "TestElement" 사용자 컨트롤이 추가되고 드래그 & 드롭으로 윈도우에 쉽게 추가할 수 있습니다.

mouse_updown_event_1.png

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="168" Width="297" xmlns:my="clr-namespace:WpfApplication1">
    <Grid>
        <my:TestElement Margin="31,23,156,0" x:Name="testElement1" VerticalAlignment="Top" Height="84" MouseDown="testElement1_MouseDown" />
    </Grid>
</Window>

위에 보시면, MouseDown 이벤트도 같이 걸어두었는데요. MainWindow.xaml.cs에서는 간단하게 다음과 같이 이벤트 핸들러를 작성해 줍니다.

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
    }

    private void testElement1_MouseDown(object sender, MouseButtonEventArgs e)
    {
        System.Diagnostics.Trace.WriteLine("testElement1_MouseDown");
    }
}

여기까지 작성하고 실행한 후, TestElement 영역에서 마우스를 클릭하면 정상적으로 MouseDown 이벤트가 발생하는 것을 Debug Output 창을 통해서 확인할 수 있습니다.

자, 이제 그럼 Bubbling을 막아볼까요?

<UserControl x:Class="WpfApplication1.TestElement"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             mc:Ignorable="d" 
             Background="Blue"
             MouseDown="UserControl_MouseDown"
             d:DesignHeight="300" d:DesignWidth="300">
    <Grid>
        <TextBlock Text="test" />
    </Grid>
</UserControl>

public partial class TestElement : UserControl
{
    public TestElement()
    {
        InitializeComponent();
    }

    private void UserControl_MouseDown(object sender, MouseButtonEventArgs e)
    {
        e.Handled = true;
    }
}

이렇게 해주고 나면, MainWindow에서 걸어두었던 MouseDown 이벤트 핸들러는 더 이상 동작하지 않습니다. 즉, Button의 MouseDown/Up 이벤트가 발생하지 않는 것은 버튼의 특성이 아니라, 위와 같이 '마우스 이벤트를 처리했다'라고 표시해두기 때문에 일어난 상황입니다.

참고로, MouseLeftButtonDown이 MouseDown 이전에 발생하며 LeftButtonDown을 "e.Handled = true" 처리하는 경우 MouseDown도 발생하지 않도록 되어 있군요. ^^

(첨부된 파일은 위의 코드를 포함한 예제 프로젝트입니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 9/7/2021]

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

비밀번호

댓글 작성자
 




... 166  167  168  169  170  [171]  172  173  174  175  176  177  178  179  180  ...
NoWriterDateCnt.TitleFile(s)
771정성태8/28/200922209.NET Framework: 160. WPF - 입력 포커스 외곽선 없애는 방법
770정성태8/26/200924568.NET Framework: 159. WCF - 같은 컴퓨터에서만 WCF 요청을 서비스하도록 설정
769정성태8/25/200927654개발 환경 구성: 49. GAC와 같은 Namespace Extension에 의해서 보여지는 폴더의 원본 확인 방법
768정성태8/24/200927039오류 유형: 85. WCF 연결 오류: MessageSecurityException
767정성태8/23/200935344.NET Framework: 158. 닷넷 프로파일러 - IL 코드 재작성 [14]
766정성태8/23/200936274.NET Framework: 157. C# 4.0 - dynamic 키워드 [4]파일 다운로드1
765정성태8/22/200929933.NET Framework: 156. XamDataGrid의 UnboundField 사용파일 다운로드1
764정성태8/21/200924298Windows: 47. Windows Virtual PC에 설치된 Windows 7 VPC에서 Aero 효과 사용 [3]
763정성태8/20/200927811Windows: 46. Windows 7 - XP 모드 응용 프로그램 바로가기 만드는 방법 [2]
762정성태8/18/200933464개발 환경 구성: 48. 개발자 PC 환경 - 유니코드(Unicode)를 위한 설정 [3]
760정성태8/17/200939884개발 환경 구성: 47. XmlCodeGenerator 1.0.0.4 업데이트 [2]
759정성태8/16/200931696.NET Framework: 155. 닷넷 프로파일러의 또 다른 응용: Visual Studio 2010 Historical Debugging
758정성태8/15/200925315VS.NET IDE: 65. WPF 프로젝트용 Visual Studio 패치들 [2]
757정성태8/12/200924681오류 유형: 84. TFS 작업 항목 보기 오류 - WorkItemTypeDeniedOrNotExistException
756정성태8/9/200924075오류 유형: 83. A revocation check could not be performed for the certificate.
755정성태8/6/200921830.NET Framework: 154. 이벤트 2중 구독
754정성태7/16/200934039VS.NET IDE: 64. Visual Studio 2010 - 64bit 혼합 모드 디버깅 지원
753정성태7/15/200932650.NET Framework: 153. WPF와 WinForm의 Shown 이벤트 시점
752정성태7/14/200928342개발 환경 구성: 46. .NET Service Bus 응용 사례: SocketShifter [2]파일 다운로드1
751정성태7/9/200929508.NET Framework: 152. 순환 참조와 XmlSerializer파일 다운로드1
750정성태7/7/200929418.NET Framework: 151. Team Explorer가 설치되지 않은 PC에서 System.InvalidProgramException 예외 발생파일 다운로드1
748정성태7/2/200927760.NET Framework: 150. WPF - Property Element 사용 의미파일 다운로드2
747정성태7/1/200947088.NET Framework: 149. WPF - UI 업데이트를 바로 반영하고 싶다면? [3]파일 다운로드1
746정성태6/25/200934062.NET Framework: 148. WPF - 데이터 바인딩 시의 예외 처리 방법 [1]파일 다운로드1
745정성태6/22/200926019.NET Framework: 147. WPF - Binding에 Sibling 요소 지정 [2]파일 다운로드1
744정성태6/21/200924717.NET Framework: 146. WPF - 중첩된 ScrollViewer의 크기 제어 [2]파일 다운로드1
... 166  167  168  169  170  [171]  172  173  174  175  176  177  178  179  180  ...