Microsoft MVP성태의 닷넷 이야기
.NET Framework: 147. WPF - Binding에 Sibling 요소 지정 [링크 복사], [링크+제목 복사],
조회: 24506
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

WPF - Binding에 Sibling 요소 지정


MVVM 구조로 개발을 하다 보니, xaml 안에 정의되는 요소 내에 Name을 지정하는 것이 필요 없게 되었습니다. 물론, 일반적으로는 그렇지만 가끔씩 필요한 경우가 발생하게 되는데, 그 한 예가 요소 간에 바인딩을 지정할 때입니다.

그나마 상위 요소에 있을 때는 역시 이름을 제거하고 RelactiveSource.Mode=FindAncestor 식으로 탐색이 가능하지만, 형제 노드에 대해서는 탐색할 방법이 제공되지 않아 대상이 되는 요소에 이름을 지정해야만 합니다. 바로 이것! 형제 노드 탐색을 위한 바인딩을 제공하고 싶다면 어떻게 해야 할까요?




처음에 생각했던 방법은 Binding.Source에 마크업 확장을 이용하여 다음과 같은 식으로 지정하려고 했었습니다.

<Border Margin="20"
        BorderBrush="{Binding Source={ext:Sibling}}"
        BorderThickness="2" Grid.Row="1">

그런 후 SiblingExtension.ProvideValue 메서드에서 자신이 바인딩된 요소, 위의 경우에서는 Border 인스턴스를 찾아내어 그것의 형제 노드를 반환하는 식으로 구현하려고 했는데... ^^; 이게 웬일입니까?

public class SiblingExtension : MarkupExtension
{
    public override object  ProvideValue(IServiceProvider serviceProvider)
    {
        IProvideValueTarget target = serviceProvider.GetService(typeof(IProvideValueTarget)) as IProvideValueTarget;

        Binding binding = target.TargetObject as Binding;
        
        return null;
    }
}

위의 코드를 실행해보면,

IProvideValueTarget.TargetObject가 의도한 대로 제가 원하는 Border - Dependency Object가 아닌 Binding 타입의 인스턴스를 반환합니다. 계층상으로 보면 Binding.Source에 ext:Sibling이 쓰였기 때문에 TargetObject가 Binding이 되는 것이 맞는 것 같습니다. 그렇긴 해도 Binding 자체에서는 그것의 요소를 구해낼 길이 없어서 위의 방법은 결국 포기를 해야 했습니다.

그렇다면, Binding 요소와 동일한 수준에서 마크업 확장을 구현해야만 가능하다는 것인데요. 다행히 Custom Binding을 구현한 소스코드를 웹에서 쉽게 찾을 수 있었습니다.

A base class for custom WPF binding markup extensions
; http://www.hardcodet.net/2008/04/wpf-custom-binding-class

그럼, 이를 응용해서 진행해볼까요? ^^




역시나 생각은 간단합니다. 사용법은 대략 다음과 같이 구성할 것입니다.

<Border BorderBrush="Blue">
</Border>

<Border BorderBrush="{ext:Sibling Path=BorderBrush}">
</Border>

위에서 의도하는 것은, 두 번째 Border의 BorderBrush가 첫 번째 Border.BorderBrush 값과 같게 하자는 것입니다.

이를 위해 BindingDecoratorBase를 상속받은 SiblingExtension을 정의합니다. 보시는 것처럼 굳이 Binding.Source를 지정할 필요는 없습니다. 왜냐하면 자기 자신을 기준으로 상위 노드로 올라간 후, 다시 자식 노드로 내려와서 형제 노드를 찾아 실행 시에 Binding.Source를 지정해 줄 것이기 때문입니다.

소스 코드도 제법 간단합니다. ^^
"A base class for custom WPF binding markup extensions" 글에서 공개한 BindingDecoratorBase를 상속받은 SiblingExtension을 정의합니다.

[그림 1: SiblingExtension]
custom_binding_siblingextension_1.png

다행히, Binding.ProvideValue 호출에서는 IServiceProvider로부터 데이터 바인딩이 되는 개체의 인스턴스를 얻어올 수가 있습니다.

IProvideValueTarget service = (IProvideValueTarget)provider.GetService(typeof(IProvideValueTarget));
DependencyObject target = service.TargetObject as DependencyObject;

즉, 위에서 최종 결과인 target은 ext:Sibling을 담고 있는 Border 인스턴스가 됩니다. 그럼 ^^ 이걸로 게임 끝이군요.

SiblingExtension.ProvideValue에서는 다음과 같이 구현해 주어 Source에 형제 노드를 담아서 ProvideValue를 호출해 주면 됩니다.

public override object ProvideValue(IServiceProvider provider)
{   
    DependencyObject targetObject;
    DependencyProperty targetProperty;
    bool status = TryGetTargetItems(provider, out targetObject, out targetProperty);

    DependencyObject parent = VisualTreeHelper.GetParent(targetObject);

    int thisIndex = 0;
    int count = VisualTreeHelper.GetChildrenCount(parent);
    for (int childIndex = 0; childIndex < count; childIndex++)
    {
        DependencyObject item = VisualTreeHelper.GetChild(parent, childIndex);
        if (item == targetObject)
        {
            thisIndex = childIndex;
        }
    }

    int targetIndex = Math.Max(0, thisIndex - 1);
    DependencyObject sibling = VisualTreeHelper.GetChild(parent, targetIndex);

    this.Source = sibling;

    return base.ProvideValue(provider);
}

그런대로 잘 동작하지만, 몇 가지 아쉬운 점이 있지요? 대개의 경우 VisualTree는 복잡하기 때문에 LogicalTree를 사용하는 것이 더 편할 때가 있습니다. 게다가 형제 노드를 앞 뒤로 +/- 값을 지정해서 인덱스를 지정할 수 있으면 더 좋을 것 같습니다. 즉, 아래와 같은 식으로도 사용하고 싶다는 것이지요.

=== 논리적 트리 구조로, 아래로 2번째의 형제 노드를 지정 ===
<Border BorderBrush="{ext:Sibling Path=BorderBrush, OnVisualTree=False, Level=2}">
</Border>

=== Visual 트리 구조로, 위로 3번째의 형제 노드를 지정 ===
<Border BorderBrush="{ext:Sibling Path=BorderBrush, OnVisualTree=True, Level=-3}">
</Border>

위의 코드는 부가적인 기능에 불과하기 때문에 소스 코드는 첨부한 프로젝트를 참조하시고 설명은 생략합니다.



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







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

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

비밀번호

댓글 작성자
 



2009-06-23 08시42분
[짜두] RSS 가 먼가 변경된듯 한데요~ㅎ 깔끔함다~ ^^
[guest]
2009-06-23 09시41분
원래 이랬는데요. ^^;
kevin25

... 166  167  168  169  170  171  172  173  174  [175]  176  177  178  179  180  ...
NoWriterDateCnt.TitleFile(s)
624정성태12/5/200824596.NET Framework: 114. WPF 이벤트에 속한 핸들러 확인 [2]파일 다운로드1
623정성태12/4/200828971디버깅 기술: 22. VS.NET SP1 + .NET Framework 소스 코드 디버깅 [2]파일 다운로드1
622정성태12/1/200831196오류 유형: 63. WPF - XamlParseException 대응 방법 [2]
621정성태11/30/200823408Team Foundation Server: 27. TeamBuild + VDPROJ 셋업 프로젝트 [1]
620정성태11/30/200822260디버깅 기술: 21. 올바른 이벤트 예외 정보 출력
619정성태11/30/200822322디버깅 기술: 20. 예외 처리를 방해하는 WPF Modal 대화창파일 다운로드1
618정성태11/29/200822851.NET Framework: 113. 이벤트에 속한 이벤트 핸들러 확인파일 다운로드1
617정성태11/26/200828368.NET Framework: 112. How to Interop DISPPARAMS [2]파일 다운로드2
616정성태11/26/200822137디버깅 기술: 19. C++/CLI - F11 디버깅 시의 변수 초기화파일 다운로드1
615정성태11/9/200831840.NET Framework: 111. WPF - Window, UserControl 클래스 상속 [1]
614정성태11/9/200831795.NET Framework: 110. WPF - 전역 예외 처리 [4]파일 다운로드1
613정성태11/8/200821385.NET Framework: 109. WPF - SystemColors 색상표파일 다운로드1
612정성태11/1/200826644.NET Framework: 108. WPF + WCF 환경에서는 DataContract를 권장 [1]
611정성태10/31/200821023오류 유형: 62. WPF - Visual Studio 2008 비정상 종료
610정성태10/24/200823201Team Foundation Server: 26. TFS 2008 SP1 + SQL Server 2008 설치
609정성태10/24/200826944.NET Framework: 107. WPF - RadioButton 데이터 바인딩 해제 현상파일 다운로드2
608정성태10/23/200821669오류 유형: 61. TFS 연결 오류: TF31003, TF30331 오류
607정성태10/18/200819552Windows: 32. 자동 실행 경로에 ".." 가 포함된 경우
606정성태10/18/200821545.NET Framework: 106. WCF - 다중 서비스 호스트파일 다운로드1
605정성태10/18/200825000.NET Framework: 105. WPF - 닫기 버튼을 없애려면.
604정성태10/18/200825571오류 유형: 60. System.Management.ManagementException - Generic failure [1]
602정성태10/15/200827478오류 유형: 59. WPF - XAML 로딩 시 Visual Studio 비정상 종료 [1]
600정성태10/9/200832172디버깅 기술: 18. TFS Team Build + Source Server = 소스 코드 디버깅 [3]
603정성태10/15/200824094    답변글 디버깅 기술: 18.1. 소스 서버 구성, 그 외의 이야기
599정성태10/5/200830049디버깅 기술: 17. TFS Team Build + Symbol Server [1]
598정성태10/3/200820024VS.NET IDE: 57. VS.NET 2008 - 다중 프로젝트에서 단일 SNK를 사용하는 방법
... 166  167  168  169  170  171  172  173  174  [175]  176  177  178  179  180  ...