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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  [11]  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13359정성태5/19/20233499오류 유형: 860. Docker Desktop - k8s 초기화 무한 반복한다면?
13358정성태5/17/20233827.NET Framework: 2125. C# - Semantic Kernel의 Semantic Memory 사용 예제 [1]파일 다운로드1
13357정성태5/16/20233611.NET Framework: 2124. C# - Semantic Kernel의 Planner 사용 예제파일 다운로드1
13356정성태5/15/20233938DDK: 10. Device Driver 테스트 설치 관련 오류 (Code 37, Code 31) 및 인증서 관련 정리
13355정성태5/12/20233878.NET Framework: 2123. C# - Semantic Kernel의 ChatGPT 대화 구현 [1]파일 다운로드1
13354정성태5/12/20234131.NET Framework: 2122. C# - "Use Unicode UTF-8 for worldwide language support" 설정을 한 경우, 한글 입력이 '\0' 문자로 처리
13352정성태5/12/20233754.NET Framework: 2121. C# - Semantic Kernel의 대화 문맥 유지파일 다운로드1
13351정성태5/11/20234261VS.NET IDE: 185. Visual Studio - 원격 Docker container 내에 실행 중인 응용 프로그램에 대한 디버깅 [1]
13350정성태5/11/20233528오류 유형: 859. Windows Date and Time - Unable to continue. You do not have permission to perform this task
13349정성태5/11/20233833.NET Framework: 2120. C# - Semantic Kernel의 Skill과 Function 사용 예제파일 다운로드1
13348정성태5/10/20233743.NET Framework: 2119. C# - Semantic Kernel의 "Basic Loading of the Kernel" 예제
13347정성태5/10/20234169.NET Framework: 2118. C# - Semantic Kernel의 Prompt chaining 예제파일 다운로드1
13346정성태5/10/20234004오류 유형: 858. RDP 원격 환경과 로컬 PC 간의 Ctrl+C, Ctrl+V 복사가 안 되는 문제
13345정성태5/9/20235398.NET Framework: 2117. C# - (OpenAI 기반의) Microsoft Semantic Kernel을 이용한 자연어 처리 [1]파일 다운로드1
13344정성태5/9/20236539.NET Framework: 2116. C# - OpenAI API 사용 - 지원 모델 목록 [1]파일 다운로드1
13343정성태5/9/20234435디버깅 기술: 192. Windbg - Hyper-V VM으로 이더넷 원격 디버깅 연결하는 방법
13342정성태5/8/20234346.NET Framework: 2115. System.Text.Json의 역직렬화 시 필드/속성 주의
13341정성태5/8/20234033닷넷: 2114. C# 12 - 모든 형식의 별칭(Using aliases for any type)
13340정성태5/8/20234134오류 유형: 857. Microsoft.Data.SqlClient.SqlException - 0x80131904
13339정성태5/6/20234891닷넷: 2113. C# 12 - 기본 생성자(Primary Constructors)
13338정성태5/6/20234324닷넷: 2112. C# 12 - 기본 람다 매개 변수파일 다운로드1
13337정성태5/5/20234799Linux: 59. dockerfile - docker exec로 container에 접속 시 자동으로 실행되는 코드 적용
13336정성태5/4/20234618.NET Framework: 2111. C# - 바이너리 출력 디렉터리와 연관된 csproj 설정
13335정성태4/30/20234700.NET Framework: 2110. C# - FFmpeg.AutoGen 라이브러리를 이용한 기본 프로젝트 구성 - Windows Forms파일 다운로드1
13334정성태4/29/20234355Windows: 250. Win32 C/C++ - Modal 메시지 루프 내에서 SetWindowsHookEx를 이용한 Thread 메시지 처리 방법
13333정성태4/28/20233773Windows: 249. Win32 C/C++ - 대화창 템플릿을 런타임에 코딩해서 사용파일 다운로드1
1  2  3  4  5  6  7  8  9  10  [11]  12  13  14  15  ...