Microsoft MVP성태의 닷넷 이야기
.NET Framework: 107. WPF - RadioButton 데이터 바인딩 해제 현상 [링크 복사], [링크+제목 복사],
조회: 30606
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

WPF - RadioButton 데이터 바인딩 해제 현상


개인적으로, WPF에서 실망스러운 컨트롤이 RadioButton입니다. 제 개인적인 WPF에 대한 이해도가 깊지 않아서든, 혹은 "by design"이란 변명이 아무리 그럴싸할지라도! 이 현상은 "일반적인 개발 상식"으로 XAML을 다루다 보면 누구나 만날 수 있는 현상이기 때문입니다.

일단, 간단하게 재현을 해볼까요!

아래와 같이, 평범한 RadioButton을 나열해 놓는 xaml 파일을 만듭니다.

 <Window x:Class="WpfApplication1.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:my="clr-namespace:WpfApplication1" x:Name="My">
    <Canvas>
        <RadioButton IsChecked="{Binding Path=DisplayData.Radio1, ElementName=My}"
                             Name="radioButton1">test1</RadioButton>
        <RadioButton IsChecked="{Binding Path=DisplayData.Radio2, ElementName=My}" 
                             Name="radioButton2" Height="16" Canvas.Top="20">test2</RadioButton>
        <RadioButton IsChecked="{Binding Path=DisplayData.Radio3, ElementName=My}" 
                             Name="radioButton3" Height="16" Canvas.Top="40">test3</RadioButton>
    </Canvas>
</Window>

xaml.cs 파일에는 아래와 같이 데이터 바인딩을 위한 코드를 구성합니다.

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

    private DisplayT displayData = new DisplayT();
    public DisplayT DisplayData
    {
        get { return this.displayData; }
    }
}

DisplayT는 다음과 같이 간단하게 정의되어 있습니다.

public partial class DisplayT
{
    private bool radio1;
    private bool radio2;
    private bool radio3;

    public bool Radio1 { get { return this.radio1; } set { this.radio1 = value; } }
    public bool Radio2 { get { return this.radio2; } set { this.radio2 = value; } }
    public bool Radio3 { get { return this.radio3; } set { this.radio3 = value; } }

    public DisplayT()
    {
    }

    public override string ToString()
    {
        return string.Format("radio1 == {0}, radio2 == {1}, radio3 == {2}",
            this.radio1, this.radio2, this.radio3);
    }
}    

빌드하고, 실행을 해보시면 뭐 그다지 표면상으로는 이상한 현상이 없어 보입니다. 하지만, 문제점을 보여주기 위한 테스트 코드를 1초마다 발생하는 타이머 이벤트에 다음과 같이 걸어보면,

public Window1()
{
    InitializeComponent();

    DispatcherTimer timer = new DispatcherTimer(new TimeSpan(0, 0, 1), DispatcherPriority.Normal,
        callback, App.Current.Dispatcher);
}

public void callback(object sender, EventArgs e)
{
    Debug.WriteLine(displayData);
}

뜻밖에도, 실행 후에 Radio 버튼 1,2,3을 모두 한번씩이라도 눌러본 경우, 데이터바인딩이 된 DisplayT 타입의 변수에는 Radio1 == True, Radio2 == True, Radio3 == True라는 엉뚱한 결과값이 실려서 출력이 됩니다.

사실, RadioButton이 그룹 설정이 되어 있는 경우에만 서로간의 배타적인 선택권이 가능하다는 점을 고려하면 위의 현상이 아주 이해못할 것도 아닙니다. 하지만, 그걸 감안하면 각각의 RadioButton들이 모두 Checked == True 상태로 남은 체로 있어야 하는 것이 정상입니다. 그런데, 결과는 UI 따로, 데이터 바인딩 따로입니다.

조금 더 테스트를 위해 callback 메서드에 다음과 같이 데이터 바인딩이 살아 있는지에 대한 테스트 코드를 추가해 보겠습니다.

public void callback(object sender, EventArgs e)
{
    CheckBinding(this.radioButton1);
    CheckBinding(this.radioButton2);
    CheckBinding(this.radioButton3);
    Debug.WriteLine(displayData);
}

private void CheckBinding(RadioButton button)
{
    Binding test = BindingOperations.GetBinding(button, RadioButton.IsCheckedProperty);
    if (test == null)
    {
        Debug.WriteLine(string.Format("{0} Binding == null", button.Content));
    }
}

"F5" 키를 이용해서 디버깅 모드로 응용 프로그램을 실행한 후, 각각의 Radio 버튼들을 누르면 방금 전에 선택된 Radio 버튼들에 대해서 바인딩 인스턴스가 null로 되는 것을 확인할 수 있습니다. 3개 모두 누르면 다음과 같이 3개 모두 바인딩이 끊기게 됩니다.

==== Output 내용 ====

test1 Binding == null
test2 Binding == null
test3 Binding == null
radio1 == True, radio2 == True, radio3 == True

분명히, 이러한 결과는 우리가 원한 것이 아니지요.




물론, 해결책은 있습니다. ^^
WPF를 초기 시절부터 공부해오신 분들은 RadioButtonList 컨트롤에 대해 알고 계실 텐데요. 이것이 나중에 Obsolete으로 바뀌면서 아예 WPF에서 빠지게 됩니다. 대신에, 그에 대한 대안으로 ListBox의 ContentTemplate을 변경하는 방법으로 - 지극히 WPF스러운 방식이죠. - 제공이 됩니다. 이에 대해서는 다음의 토픽에서도 잘 설명되고 있습니다.

Data-binding to a radio button group
; https://social.msdn.microsoft.com/Forums/en-US/323d067a-efef-4c9f-8d99-fecf45522395/databinding-to-a-radio-button-group?forum=wpf

RadioButton unchecked bindings issue still not resolved?
; https://social.msdn.microsoft.com/Forums/en-US/8eb8280a-19c4-4502-8260-f74633a9e2f2/radiobutton-unchecked-bindings-issue-still-not-resolved?forum=wpf

첨부된 파일인 "WpfApplication1.zip"은, 위에서 문제점을 짚은 예제 소스 코드이고, "WpfApplication2.zip"은 이를 해결한 간단한 예제 소스 코드입니다.



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

[연관 글]






[최초 등록일: ]
[최종 수정일: 5/21/2021]

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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...
NoWriterDateCnt.TitleFile(s)
11010정성태8/6/201624981기타: 58. Outlook에 설정한 SMTP/POP3(예:천리안 메일) 계정 암호를 잊어버린 경우
11009정성태8/3/201630222개발 환경 구성: 289. 2016-08-02부터 시작된 윈도우 10 1주년 업데이트에서 Bash Shell 사용 [8]
11008정성태8/1/201624708오류 유형: 345. 2의 30승 이상의 원소를 갖는 경우 버그가 발생하는 이진 검색(Binary Search) 코드
11007정성태8/1/201626204오류 유형: 344. RDP ActiveX 컨트롤로 특정 PC에 연결할 수 없을 때, 오류 상황을 해결하기 위한 팁파일 다운로드1
11006정성태7/22/201629672개발 환경 구성: 288. SSL 인증서를 Azure Cloud Service에 적용하는 방법
11005정성태7/22/201628403개발 환경 구성: 287. Let's Encrypt 인증서 업데이트 주기: 90일
11004정성태7/22/201622847오류 유형: 343. Invalid service definition or service configuration. Please see the Error List for more details.
11003정성태7/20/201630606VS.NET IDE: 110. Visual Studio 2015에서 .NET Core 응용 프로그램 개발 [1]
11002정성태7/20/201623948개발 환경 구성: 286. Microsoft Azure 서비스의 구독은 반드시 IE로!
11001정성태7/19/201635018.NET Framework: 599. .NET Core/SDK 설치 및 기본 사용법 [6]
11000정성태7/16/201623646오류 유형: 342. Microsoft Visual Studio 2010 Tools for Office Runtime (x86 and x64) 설치 시 오류
10999정성태7/16/201623771오류 유형: 341. .NET Framework 4.5.2가 설치 안 되는 경우
10998정성태7/16/201623454.NET Framework: 598. C# - Excel 시트에 윈도우 폼 기능을 추가하는 방법 [1]파일 다운로드1
10997정성태7/16/201624502오류 유형: 340. HTTP Error 500.23 - Internal Server Error파일 다운로드1
10996정성태7/14/201629381Windows: 118. 유선 접속 상태에서 재부팅하면 무선 연결이 자동 연결 안되는 문제 [4]파일 다운로드1
10995정성태6/27/201622546VS.NET IDE: 109. Visual Studio 유료 버전 사용자의 주기적인 온라인 인증을 없애는 방법
10994정성태6/23/201621816개발 환경 구성: 285. 알고스팟(https://algospot.com)을 위한 Visual C++ 답안 작성 요령파일 다운로드1
10993정성태6/23/201623224.NET Framework: 597. 닷넷 메타데이터에 struct/class(값/참조 형식)의 구분이 있을까요?
10992정성태6/13/201619500오류 유형: 339. vbs 스크립트 실행 시 항상 실행 여부를 묻는 질문 창이 뜬다면?
10991정성태6/13/201625391오류 유형: 338. octave-gui 실행 시 "octave-gui.exe has stopped working" 오류
10990정성태6/13/201626850오류 유형: 337. missing type specifier - [type] assumed. Note: C++ does not support default-[type]
10989정성태6/7/201623231.NET Framework: 596. C# - WCF wsDualHttpBinding의 ClientBaseAddress 속성 - 두 번째 이야기
10988정성태6/3/201624676기타: 57. Outlook blocked access to the following potentially unsafe attachments
10987정성태6/2/201625452.NET Framework: 595. XLL 파일에 포함된 .NET 어셈블리를 추출하는 방법
10986정성태6/1/201626160.NET Framework: 594. C# - WCF wsDualHttpBinding의 ClientBaseAddress 속성
10985정성태6/1/201624328오류 유형: 336. An error occurred while ejecting 'DVD RW drive ...'
... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...