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

... 91  92  93  94  95  96  97  [98]  99  100  101  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11484정성태4/11/201824733.NET Framework: 737. C# - async를 Task 타입이 아닌 사용자 정의 타입에 적용하는 방법파일 다운로드1
11483정성태4/10/201828034개발 환경 구성: 358. "Let's Encrypt"에서 제공하는 무료 SSL 인증서를 IIS에 적용하는 방법 (2) [1]
11482정성태4/10/201820479VC++: 126. CUDA Core 수를 알아내는 방법
11481정성태4/10/201832096개발 환경 구성: 357. CUDA의 인덱싱 관련 용어 - blockIdx, threadIdx, blockDim, gridDim
11480정성태4/9/201822143.NET Framework: 736. C# - API를 사용해 Azure에 접근하는 방법 [2]파일 다운로드1
11479정성태4/9/201817760.NET Framework: 735. Azure - PowerShell로 Access control(IAM)에 새로운 계정 만드는 방법
11478정성태11/8/201920004디버깅 기술: 115. windbg - 덤프 파일로부터 PID와 환경변수 등의 정보를 구하는 방법 [1]
11477정성태4/8/201817459오류 유형: 460. windbg - sos 명령어 수행 시 c0000006 오류 발생
11476정성태4/8/201819025디버깅 기술: 114. windbg - !threads 출력 결과로부터 닷넷 관리 스레드(System.Threading.Thread) 객체를 구하는 방법
11475정성태3/28/201821325디버깅 기술: 113. windbg - Thread.Suspend 호출 시 응용 프로그램 hang 현상에 대한 덤프 분석
11474정성태3/27/201819438오류 유형: 459. xperf: error: TEST.Event: Invalid flags. (0x3ec).
11473정성태3/22/201824590.NET Framework: 734. C# - Thread.Suspend 호출 시 응용 프로그램 hang 현상파일 다운로드2
11472정성태3/22/201818560개발 환경 구성: 356. GTX 1070, GTX 960, GT 640M의 cudaGetDeviceProperties 출력 결과
11471정성태3/20/201821941VC++: 125. CUDA로 작성한 RGB2RGBA 성능 [1]파일 다운로드1
11470정성태3/20/201824092오류 유형: 458. Visual Studio - CUDA 프로젝트 빌드 시 오류 C1189, expression must have a constant value
11469정성태3/19/201817114오류 유형: 457. error MSB3103: Invalid Resx file. Could not load file or assembly 'System.Windows.Forms, ...' or one of its dependencies.
11468정성태3/19/201816638오류 유형: 456. 닷넷 응용 프로그램 실행 시 0x80131401 예외 발생
11467정성태3/19/201816082오류 유형: 455. Visual Studio Installer - 업데이트 실패
11466정성태3/18/201817230개발 환경 구성: 355. 한 대의 PC에서 2개 이상의 DirectX 게임을 실행하는 방법
11463정성태3/15/201819569.NET Framework: 733. 스레드 간의 read/write 시에도 lock이 필요 없는 경우파일 다운로드1
11462정성태3/14/201822428개발 환경 구성: 354. HTTPS 호출에 대한 TLS 설정 확인하는 방법 [1]
11461정성태3/13/201825054오류 유형: 454. 윈도우 업데이트 설치 오류 - 0x800705b4 [1]
11460정성태3/13/201817537디버깅 기술: 112. windbg - 닷넷 메모리 덤프에서 전역 객체의 내용을 조사하는 방법
11459정성태3/13/201818359오류 유형: 453. Debug Diagnostic Tool에서 mscordacwks.dll을 찾지 못하는 문제
11458정성태2/21/201819334오류 유형: 452. This share requires the obsolete SMB1 protocol, which is unsafe and could expose your system to attack. [1]
11457정성태2/17/201824052.NET Framework: 732. C# - Task.ContinueWith 설명 [1]파일 다운로드1
... 91  92  93  94  95  96  97  [98]  99  100  101  102  103  104  105  ...