Microsoft MVP성태의 닷넷 이야기
.NET Framework: 107. WPF - RadioButton 데이터 바인딩 해제 현상 [링크 복사], [링크+제목 복사],
조회: 28715
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 166  167  168  169  170  171  172  173  174  175  [176]  177  178  179  180  ...
NoWriterDateCnt.TitleFile(s)
633정성태12/20/200826006기타: 24. RMClock for x64 [2]
632정성태12/19/200834458기타: 23. D820 - 배터리 없이 바이오스 업데이트 방법 [2]파일 다운로드1
631정성태12/10/200843468VC++: 36. Detours 라이브러리를 이용한 Win32 API - Sleep 호출 가로채기 [3]
630정성태12/9/200824319.NET Framework: 117. WPF - TreeView에서 항목이 펼쳐질 때 Cursors.Wait 사용파일 다운로드1
629정성태12/7/200833545.NET Framework: 116. 소켓 연결 시간 제한
628정성태12/6/200822036.NET Framework: 115. Marshal 타입 관련 2가지 자원 해제 메서드파일 다운로드1
627정성태12/6/200824732VS.NET IDE: 58. VS.NET IDE 팁 - 커서 위치 이동 [1]
626정성태12/6/200824805오류 유형: 65. TF53018: The application tier XXXXXXX is attempting to connect to a data tier with an incompatible version
625정성태12/6/200824969오류 유형: 64. TFS 2008 SP1 설치 - MsiApplyMultiplePatches returned 0x643
624정성태12/5/200825917.NET Framework: 114. WPF 이벤트에 속한 핸들러 확인 [2]파일 다운로드1
623정성태12/4/200830405디버깅 기술: 22. VS.NET SP1 + .NET Framework 소스 코드 디버깅 [2]파일 다운로드1
622정성태12/1/200832686오류 유형: 63. WPF - XamlParseException 대응 방법 [2]
621정성태11/30/200824707Team Foundation Server: 27. TeamBuild + VDPROJ 셋업 프로젝트 [1]
620정성태11/30/200823620디버깅 기술: 21. 올바른 이벤트 예외 정보 출력
619정성태11/30/200823723디버깅 기술: 20. 예외 처리를 방해하는 WPF Modal 대화창파일 다운로드1
618정성태11/29/200824074.NET Framework: 113. 이벤트에 속한 이벤트 핸들러 확인파일 다운로드1
617정성태11/26/200830236.NET Framework: 112. How to Interop DISPPARAMS [2]파일 다운로드2
616정성태11/26/200823513디버깅 기술: 19. C++/CLI - F11 디버깅 시의 변수 초기화파일 다운로드1
615정성태11/9/200833117.NET Framework: 111. WPF - Window, UserControl 클래스 상속 [1]
614정성태11/9/200833094.NET Framework: 110. WPF - 전역 예외 처리 [4]파일 다운로드1
613정성태11/8/200822625.NET Framework: 109. WPF - SystemColors 색상표파일 다운로드1
612정성태11/1/200827971.NET Framework: 108. WPF + WCF 환경에서는 DataContract를 권장 [1]
611정성태10/31/200822002오류 유형: 62. WPF - Visual Studio 2008 비정상 종료
610정성태10/24/200824596Team Foundation Server: 26. TFS 2008 SP1 + SQL Server 2008 설치
609정성태10/24/200828715.NET Framework: 107. WPF - RadioButton 데이터 바인딩 해제 현상파일 다운로드2
608정성태10/23/200822864오류 유형: 61. TFS 연결 오류: TF31003, TF30331 오류
... 166  167  168  169  170  171  172  173  174  175  [176]  177  178  179  180  ...