Microsoft MVP성태의 닷넷 이야기
.NET Framework: 147. WPF - Binding에 Sibling 요소 지정 [링크 복사], [링크+제목 복사],
조회: 27255
글쓴 사람
정성태 (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

... 181  182  183  184  185  [186]  187  188  189  190  191  192  193  194  195  ...
NoWriterDateCnt.TitleFile(s)
359정성태10/17/200619095오류 유형: 14. VS.NET 빌드 오류 - FxCopCmd.exe returned error code 65.
358정성태10/17/200624452오류 유형: 13. WSE 3.0 서비스 관련 WSE101 오류 / Destination Unreachable
357정성태12/1/200626772.NET Framework: 74. WCF 이야기 [4]
378정성태10/28/200631641    답변글 .NET Framework: 74.1. WCF와 WSE 3.0의 활용 [4]파일 다운로드1
379정성태11/3/200630548    답변글 .NET Framework: 74.2. WCF로 구현하는 .NET Remoting [4]파일 다운로드1
380정성태10/28/200629400    답변글 .NET Framework: 74.3. 웹 서비스와 닷넷 리모팅으로써의 WCF 구현파일 다운로드1
381정성태10/28/200631946    답변글 .NET Framework: 74.4. WCF 서비스 참조 추가 메뉴 [2]
382정성태10/28/200637841    답변글 .NET Framework: 74.5. WCF 서비스를 IIS에서 호스팅하는 방법파일 다운로드1
383정성태10/28/200632726    답변글 .NET Framework: 74.6. IIS 6.0: 다중 Endpoint 제공파일 다운로드1
384정성태10/28/200629746    답변글 .NET Framework: 74.7. IIS 7.0: 다중 Endpoint 제공
389정성태11/11/200632422    답변글 .NET Framework: 74.8. WCF에 SSL 적용 (1) - Httpcfg.exe 도구를 이용한 SSL 설정
390정성태11/6/200629584    답변글 .NET Framework: 74.9. WCF에 SSL 적용 (2) - 서비스 제작파일 다운로드1
356정성태10/7/200625135COM 개체 관련: 19. COM의 Apartment를 이해해 보자. [8]
386light10/30/200620080    답변글 COM 개체 관련: 19.1. [답변]: COM 객체를 글로벌마샬으로 만든후, 사용한다.
355정성태10/9/200627902개발 환경 구성: 19. Internet_Zone 하위에 새로운 코드 그룹을 추가하는 예제 [4]파일 다운로드2
353정성태12/31/200636279개발 환경 구성: 18. 윈도우즈 인증서 서비스 이야기 [3]
354정성태10/23/200638371    답변글 개발 환경 구성: 18.1. 윈도우즈 인증서 서비스 설치
372정성태12/31/200640726    답변글 개발 환경 구성: 18.2. 웹 사이트에 SSL을 적용 [3]
373정성태10/24/200632137    답변글 개발 환경 구성: 18.3. 사용자 입장에서의 HTTPS 접근 (1)
374정성태10/25/200629424    답변글 개발 환경 구성: 18.4. 사용자 입장에서의 HTTPS 접근 (2)
391정성태11/7/200633354    답변글 개발 환경 구성: 18.5. 사용자 인증서 발급
392정성태11/11/200646188    답변글 개발 환경 구성: 18.6. 인증서 관리 (1) - 내보내기/가져오기
394정성태11/9/200630892    답변글 개발 환경 구성: 18.7. 인증서 관리 (2) - 개인키를 내보낼 수 있는 유형의 인증서 발급 [1]
395정성태11/9/200642951    답변글 개발 환경 구성: 18.8. 인증서 관리 (3) - 인증서 MMC 관리자 사용
414정성태12/23/200634833    답변글 개발 환경 구성: 18.9. CRL(Certificate Revocation List) 관리
428정성태12/31/200647489    답변글 개발 환경 구성: 18.10. IIS 7 - SSL 사이트 설정하는 방법 [4]
... 181  182  183  184  185  [186]  187  188  189  190  191  192  193  194  195  ...