WPF-퀵 가이드

WPF는 Windows Presentation Foundation을 나타냅니다. Windows 응용 프로그램을 빌드하기위한 강력한 프레임 워크입니다. 이 자습서에서는 WPF 응용 프로그램을 빌드하는 데 필요한 기능과 Windows 응용 프로그램에 근본적인 변화를 가져 오는 방법에 대해 설명합니다.

WPF는 .NET Framework 3.0 버전에서 처음 도입 된 후 후속 .NET Framework 버전에 다른 많은 기능이 추가되었습니다.

WPF 아키텍처

WPF 이전에는 MFC 및 Windows 양식과 같이 Microsoft에서 제공하는 다른 사용자 인터페이스 프레임 워크가 User32 및 GDI32 DLL을 둘러싼 래퍼 였지만 WPF는 User32를 최소한으로 만 사용합니다. 그래서,

  • WPF는 단순한 래퍼 그 이상입니다.
  • .NET 프레임 워크의 일부입니다.
  • 여기에는 관리 코드와 비 관리 코드가 혼합되어 있습니다.

WPF 아키텍처의 주요 구성 요소는 아래 그림과 같습니다. WPF의 가장 중요한 코드 부분은-

  • 프레젠테이션 프레임 워크
  • 프레젠테이션 코어
  • Milcore

그만큼 presentation framework 그리고 presentation core 관리 코드로 작성되었습니다. Milcore DirectX와의 긴밀한 통합을 허용하는 관리되지 않는 코드의 일부입니다 (디스플레이 및 렌더링 담당). CLR 메모리 관리, 오류 처리 등과 같은 많은 기능을 제공하여 개발 프로세스의 생산성을 높입니다.

WPF – 장점

이전 GUI 프레임 워크에서는 애플리케이션의 모양과 작동 방식 사이에 실질적인 분리가 없었습니다. GUI와 동작은 모두 동일한 언어로 만들어졌습니다. 예를 들어 C # 또는 VB.Net은 UI 및 이와 관련된 동작을 모두 구현하기 위해 개발자의 더 많은 노력이 필요합니다.

WPF에서 UI 요소는 XAML로 디자인되고 동작은 C # 및 VB.Net과 같은 절차 언어로 구현 될 수 있습니다. 따라서 디자이너 코드에서 동작을 분리하는 것은 매우 쉽습니다.

XAML을 사용하면 프로그래머가 디자이너와 동시에 작업 할 수 있습니다. GUI와 그 동작을 구분하면 스타일과 템플릿을 사용하여 컨트롤의 모양을 쉽게 변경할 수 있습니다.

WPF – 기능

WPF는 Windows 응용 프로그램을 만드는 강력한 프레임 워크입니다. 그것은 많은 훌륭한 기능을 지원하며, 그중 일부는 아래에 나열되어 있습니다.

특색 기술
컨트롤 내부의 컨트롤 다른 컨트롤 내부의 컨트롤을 콘텐츠로 정의 할 수 있습니다.
데이터 바인딩 사용자 인터페이스에서 UI 요소와 데이터 개체 간의 데이터를 표시하고 상호 작용하는 메커니즘입니다.
미디어 서비스 이미지, 오디오 및 비디오와 같은 일반적인 미디어 요소로 사용자 인터페이스를 구축하기위한 통합 시스템을 제공합니다.
템플릿 WPF에서는 템플릿을 사용하여 요소의 모양을 직접 정의 할 수 있습니다.
애니메이션 사용자 인터페이스에서 상호 작용 및 이동 구축
대체 입력 Windows 7 이상에서 멀티 터치 입력을 지원합니다.
Direct3D 더 복잡한 그래픽과 사용자 정의 테마를 표시 할 수 있습니다.

Microsoft는 WPF 응용 프로그램 개발을위한 두 가지 중요한 도구를 제공합니다.

  • 비주얼 스튜디오
  • 익스프레션 블렌드

두 도구 모두 WPF 프로젝트를 만들 수 있지만 실제로 Visual Studio는 개발자가 더 많이 사용하는 반면 Blend는 디자이너가 더 자주 사용합니다. 이 자습서에서는 주로 Visual Studio를 사용합니다.

설치

Microsoft는 VisualStudio 에서 다운로드 할 수있는 무료 버전의 Visual Studio를 제공합니다 .

파일을 다운로드하고 아래 단계에 따라 시스템에 WPF 애플리케이션 개발 환경을 설정하십시오.

  • 다운로드가 완료되면 installer. 다음 대화 상자가 표시됩니다.

  • 클릭 Install 버튼을 누르면 설치 프로세스가 시작됩니다.

  • 설치 과정이 성공적으로 완료되면 다음 대화 상자가 나타납니다.

  • 이 대화 상자를 닫고 필요한 경우 컴퓨터를 다시 시작하십시오.

  • 이제 시작 메뉴에서 Visual Studio를 열면 다음 대화 상자가 열립니다.

  • 모든 작업이 완료되면 Visual Studio의 기본 창이 표시됩니다.

이제 첫 번째 WPF 애플리케이션을 빌드 할 준비가되었습니다.

이 장에서는 간단한 Hello World WPF 애플리케이션을 개발합니다. 이제 아래 단계에 따라 간단한 구현을 시작하겠습니다.

  • 파일> 새로 만들기> 프로젝트 메뉴 옵션을 클릭합니다.
  • 다음 대화 상자가 표시됩니다.
  • 템플릿에서 Visual C #을 선택하고 가운데 패널에서 WPF 애플리케이션을 선택합니다.

  • 프로젝트 이름을 지정하십시오. 유형HelloWorld 이름 필드에서 확인 버튼을 클릭합니다.

  • 기본적으로 두 개의 파일이 생성되며 하나는 XAML 파일 (mainwindow.xaml)이고 다른 하나는 CS 파일 (mainwindow.cs)

  • mainwindow.xaml에는 두 개의 하위 창이 표시됩니다. 하나는 design window 그리고 다른 하나는 source (XAML) window.

  • WPF 애플리케이션에는 애플리케이션의 UI를 디자인하는 두 가지 방법이 있습니다. 하나는 도구 상자에서 디자인 창으로 UI 요소를 드래그 앤 드롭하는 것입니다. 두 번째 방법은 UI 요소에 대한 XAML 태그를 작성하여 UI를 디자인하는 것입니다. Visual Studio는 UI 디자인에 끌어서 놓기 기능을 사용할 때 XAML 태그를 처리합니다.

  • mainwindow.xaml 파일에는 기본적으로 다음 XAML 태그가 작성됩니다.

<Window x:Class = "HelloWorld.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid> 
   </Grid> 
	
</Window>
  • 기본적으로 그리드는 페이지 다음의 첫 번째 요소로 설정됩니다.
  • 도구 상자로 이동하여 TextBlock을 디자인 창으로 드래그 해 보겠습니다.
  • 디자인 창에 TextBlock이 표시됩니다.
  • 소스 창을 보면 Visual Studio에서 TextBlock의 XAML 코드를 생성했음을 알 수 있습니다.

  • XAML 코드에서 TextBlock의 Text 속성을 TextBlock에서 Hello World로 변경해 보겠습니다.

<Window x:Class = "HelloWorld.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <TextBlock x:Name = "textBlock" HorizontalAlignment = "Left"
         Margin = "235,143,0,0" TextWrapping = "Wrap" Text = "Hello World!"
         VerticalAlignment = "Top" Height = "44" Width = "102" /> 
   </Grid> 
	
</Window>
  • 이제 디자인 창에서도 변경 사항을 볼 수 있습니다.

위의 코드가 컴파일되고 실행되면 다음과 같은 창이 나타납니다.

축하합니다! 첫 번째 WPF 애플리케이션을 설계하고 만들었습니다.

WPF로 작업하는 동안 가장 먼저 접하게되는 것 중 하나는 XAML입니다. XAML은 Extensible Application Markup Language를 나타냅니다. XML을 기반으로하는 단순하고 선언적인 언어입니다.

  • XAML에서는 계층 적 관계가있는 개체의 속성을 매우 쉽게 만들고, 초기화하고, 설정할 수 있습니다.

  • 주로 GUI 디자인에 사용되지만 Workflow Foundation에서 워크 플로를 선언하는 등 다른 용도로도 사용할 수 있습니다.

기본 구문

새 WPF 프로젝트를 만들 때 아래와 같이 MainWindow.xaml에 기본적으로 일부 XAML 코드가 표시됩니다.

<Window x:Class = "Resources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
         
   </Grid> 
	
</Window>

위의 XAML 파일에는 다양한 종류의 정보가 포함되어 있습니다. 다음 표는 각 정보의 역할을 간략하게 설명합니다.

정보 기술
<창 여는 개체 요소 또는 루트의 컨테이너입니다.
x : Class = "Resources.MainWindow" 마크 업을 뒤에 정의 된 부분 클래스 코드에 연결하는 부분 클래스 선언입니다.
xmlns = "http://schemas.microsoft.com/win fx / 2006 / xaml / presentation" WPF 클라이언트 / 프레임 워크에 대한 기본 XAML 네임 스페이스 매핑
xmlns : x = "http://schemas.microsoft.com/w infx / 2006 / xaml" x : 접두사에 매핑하는 XAML 언어 용 XAML 네임 스페이스
> 루트의 개체 요소 끝

<그리드>

</ 그리드>

빈 그리드 객체의 시작 및 종료 태그입니다.
</ 창> 개체 요소 닫기

XAML의 구문 규칙은 XML과 거의 유사합니다. XAML 문서를 보면 실제로 유효한 XML 파일임을 알 수 있지만 XML 파일이 반드시 XAML 파일은 아닙니다. XML에서는 특성 값이 문자열이어야하는 반면 XAML에서는 속성 요소 구문으로 알려진 다른 개체 일 수 있기 때문입니다.

  • Object 요소의 구문은 왼쪽 꺾쇠 괄호 (<)와 그 뒤에 개체 이름 (예 : Button)으로 시작합니다.

  • 해당 개체 요소의 몇 가지 속성 및 특성을 정의합니다.

  • Object 요소는 슬래시 (/)와 오른쪽 꺾쇠 괄호 (>)로 닫아야합니다.

자식 요소가없는 단순 개체의 예

<Button/>

일부 속성이있는 개체 요소의 예

<Button Content = "Click Me" Height = "30" Width = "60" />

대체 구문의 예는 속성을 정의합니다 (속성 요소 구문).

<Button> 
   <Button.Content>Click Me</Button.Content> 
   <Button.Height>30</Button.Height> 
   <Button.Width>60</Button.Width> 
</Button>

자식 요소가있는 객체의 예 : StackPanel은 자식 요소로 Textblock을 포함합니다.

<StackPanel Orientation = "Horizontal"> 
   <TextBlock Text = "Hello"/> 
</StackPanel>

WPF에서 XAML을 사용해야하는 이유

XAML은 WPF의 가장 널리 알려진 기능 일뿐만 아니라 가장 오해를받는 기능 중 하나이기도합니다. WPF에 노출되어 있다면 XAML에 대해 들어 보셨을 것입니다. 그러나 XAML에 대한 다음 두 가지 덜 알려진 사실에 유의하십시오.

  • WPF에는 XAML이 필요하지 않습니다.
  • XAML에는 WPF가 필요하지 않습니다.

실제로 분리 가능한 기술입니다. 그 방법을 이해하기 위해 XAML의 일부 속성을 사용하여 단추를 만드는 간단한 예제를 살펴 보겠습니다.

<Window x:Class = "WPFXAMLOverview.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button x:Name = "button" Content = "Click Me" HorizontalAlignment = "Left"  
         Margin = "150" VerticalAlignment = "Top" Width = "75" /> 
   </StackPanel> 
	
</Window>

WPF에서 XAML을 사용하지 않기로 선택한 경우 절차 언어로도 동일한 GUI 결과를 얻을 수 있습니다. 같은 예제를 보지만 이번에는 C #에서 버튼을 생성합니다.

using System.Windows; 
using System.Windows.Controls;  

namespace WPFXAMLOverview { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
			
         // Create the StackPanel 
         StackPanel stackPanel = new StackPanel(); 
         this.Content = stackPanel; 
			
         // Create the Button 
         Button button = new Button();
         button.Content = "Click Me"; 
         button.HorizontalAlignment = HorizontalAlignment.Left; 
         button.Margin = new Thickness(150); 
         button.VerticalAlignment = VerticalAlignment.Top; 
         button.Width = 75; 
         stackPanel.Children.Add(button);  
      } 
   } 
}

XAML 코드 또는 C # 코드를 컴파일하고 실행하면 아래와 같은 출력이 표시됩니다.

위의 예에서 XAML에서 개체의 속성을 만들고, 초기화하고, 설정하기 위해 수행 할 수있는 작업은 코드를 사용하여 동일한 작업을 수행 할 수도 있습니다.

  • XAML은 UI 요소를 디자인하는 또 다른 간단하고 쉬운 방법입니다.

  • XAML을 사용한다고해서 UI 요소를 디자인하기 위해 수행 할 수있는 작업이 유일한 방법은 아닙니다. XAML에서 개체를 선언하거나 코드를 사용하여 정의 할 수 있습니다.

  • XAML은 선택 사항이지만 그럼에도 불구하고 WPF 디자인의 핵심입니다.

  • XAML의 목표는 비주얼 디자이너가 사용자 인터페이스 요소를 직접 만들 수 있도록하는 것입니다.

  • WPF는 마크 업에서 사용자 인터페이스의 모든 시각적 측면을 제어 할 수 있도록하는 것을 목표로합니다.

프로그래머가 쉽게 개체를 처리하고 응용 프로그램의 동작을 변경할 수 있도록 요소와 구성 요소가 트리 구조로 정렬되는 많은 기술이 있습니다. WPF (Windows Presentation Foundation)에는 개체 형태의 포괄적 인 트리 구조가 있습니다. WPF에서는 완전한 객체 트리를 개념화하는 두 가지 방법이 있습니다.

  • 논리 트리 구조
  • 시각적 트리 구조

이러한 트리 구조의 도움으로 UI 요소 간의 관계를 쉽게 만들고 식별 할 수 있습니다. 대부분 WPF 개발자와 디자이너는 절차 언어를 사용하여 응용 프로그램을 만들거나 XAML에서 응용 프로그램의 UI 부분을 디자인하여 개체 트리 구조를 염두에 둡니다.

논리 트리 구조

WPF 애플리케이션에서 XAML의 UI 요소 구조는 논리적 트리 구조를 나타냅니다. XAML에서 UI의 기본 요소는 개발자가 선언합니다. WPF의 논리 트리는 다음을 정의합니다.

  • 종속성 속성
  • 정적 및 동적 리소스
  • 이름 등에 요소 바인딩

버튼과 목록 상자가 생성 된 다음 예제를 살펴 보겠습니다.

<Window x:Class = "WPFElementsTree.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button x:Name = "button" Height = "30" Width = "70" Content = "OK" Margin = "20" /> 
		
      <ListBox x:Name = "listBox" Height = "100" Width = "100" Margin = "20"> 
         <ListBoxItem Content = "Item 1" /> 
         <ListBoxItem Content = "Item 2" /> 
         <ListBoxItem Content = "Item 3" /> 
      </ListBox> 
		
   </StackPanel> 
	
</Window>

XAML 코드를 보면 트리 구조를 볼 수 있습니다. 즉, 루트 노드는 Window이고 루트 노드 내부에는 StackPanel이라는 자식이 하나뿐입니다. 그러나 StackPanel에는 두 개의 자식 요소, 버튼 및 목록 상자가 있습니다. 목록 상자에는 세 개의 하위 목록 상자 항목이 더 있습니다.

시각적 트리 구조

WPF에서 시각적 트리의 개념은 Visual Base Class로 표현되는 시각적 개체의 구조를 설명합니다. 출력 화면에 렌더링되는 모든 UI 요소를 의미합니다.

프로그래머가 특정 컨트롤에 대한 템플릿을 만들려고 할 때 실제로 해당 컨트롤의 시각적 트리를 렌더링합니다. 비주얼 트리는 성능 및 최적화를 위해 낮은 수준의 컨트롤을 그리려는 사용자에게도 매우 유용합니다.

WPF 응용 프로그램에서 시각적 트리는 다음에 사용됩니다.

  • 시각적 개체를 렌더링합니다.
  • 레이아웃 렌더링.
  • 라우트 된 이벤트는 대부분 논리 트리가 아닌 시각적 트리를 따라 이동합니다.

버튼과 목록 상자가 포함 된 위의 간단한 애플리케이션의 시각적 트리를보기 위해 XAML 코드를 컴파일하고 실행하면 다음 창이 표시됩니다.

응용 프로그램이 실행 중일 때 아래와 같이이 응용 프로그램의 전체 계층을 보여주는 라이브 비주얼 트리 창에서 실행중인 응용 프로그램의 시각적 트리를 볼 수 있습니다.

시각적 트리는 일반적으로 논리 트리의 상위 집합입니다. 여기에서 모든 논리적 요소가 시각적 트리에도 있음을 알 수 있습니다. 따라서이 두 트리는 UI를 구성하는 동일한 개체 집합에 대한 두 가지 다른보기 일뿐입니다.

  • 논리적 트리는 사용자 인터페이스의 핵심 구조에 초점을 맞추고 정확히 어떻게 표현되었는지에 대한 세부 사항을 무시할 수 있도록 많은 세부 사항을 생략합니다.

  • 논리 트리는 사용자 인터페이스의 기본 구조를 만드는 데 사용합니다.

  • 프레젠테이션에 집중하는 경우 시각적 트리가 흥미로울 것입니다. 예를 들어 UI 요소의 모양을 사용자 지정하려면 시각적 트리를 사용해야합니다.

WPF 응용 프로그램에서 종속성 속성은 CLR 속성을 확장하는 특정 유형의 속성입니다. WPF 속성 시스템에서 사용할 수있는 특정 기능을 활용합니다.

종속성 속성을 정의하는 클래스는 DependencyObject수업. XAML에서 사용되는 많은 UI 컨트롤 클래스는DependencyObject 클래스는 종속성 속성을 지원합니다. 예를 들어 Button 클래스는 IsMouseOver 종속성 속성.

다음 XAML 코드는 몇 가지 속성이있는 단추를 만듭니다.

<Window x:Class = "WPFDependencyProperty.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFDependencyProperty"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button  Height = "40" Width = "175" Margin = "10" Content = "Dependency Property"> 
         <Button.Style> 
            <Style TargetType = "{x:Type Button}"> 
               <Style.Triggers> 
					
                  <Trigger Property = "IsMouseOver" Value = "True"> 
                     <Setter Property = "Foreground" Value = "Red" /> 
                  </Trigger>
						
               </Style.Triggers>
            </Style> 
         </Button.Style> 
      </Button> 
   </Grid> 
	
</Window>

XAML의 x : Type 태그 확장에는 C #의 typeof ()와 유사한 기능이 있습니다. <Style TargetType = "{x : Type Button}">과 같은 개체의 유형을 취하는 속성이 지정 될 때 사용됩니다.

위의 코드가 컴파일되고 실행되면 다음과 같은 결과가 나타납니다. MainWindow. 마우스가 버튼 위에 있으면 버튼의 전경색이 변경됩니다. 마우스가 버튼을 떠나면 원래 색상으로 다시 변경됩니다.

종속성 속성이 필요한 이유

종속성 속성은 응용 프로그램에서 사용할 때 모든 종류의 이점을 제공합니다. 종속성 속성은 다음 시나리오에서 CLR 속성을 통해 사용할 수 있습니다-

  • 스타일을 설정하려면
  • 데이터 바인딩을 원하는 경우
  • 리소스 (정적 또는 동적 리소스)로 설정하려는 경우
  • 애니메이션을 지원하고 싶다면

기본적으로 종속성 속성은 CLR 속성을 사용하여 얻을 수없는 많은 기능을 제공합니다.

주요 차이점 dependency properties 및 기타 CLR properties 아래에 나열되어 있습니다-

  • CLR 속성은 다음을 사용하여 클래스의 전용 멤버에서 직접 읽고 쓸 수 있습니다. gettersetter. 반대로 종속성 속성은 로컬 개체에 저장되지 않습니다.

  • 종속성 속성은 DependencyObject 클래스에서 제공하는 키 / 값 쌍의 사전에 저장됩니다. 또한 변경 될 때 속성을 저장하기 때문에 많은 메모리를 절약합니다. XAML에서도 바인딩 할 수 있습니다.

사용자 지정 종속성 속성

.NET Framework에서 사용자 지정 종속성 속성도 정의 할 수 있습니다. 아래 단계에 따라 C #에서 사용자 지정 종속성 속성을 정의합니다.

  • 신고 및 등록 dependency property 시스템 호출 레지스터로.

  • 제공 settergetter 재산을 위해.

  • 정의 static handler which will handle any changes that occur globally

  • Define an instance handler which will handle any changes that occur to that particular instance.

The following C# code defines a dependency property to set the SetText property of the user control.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks; 

using System.Windows; 
using System.Windows.Controls; 
using System.Windows.Data; 
using System.Windows.Documents; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Media.Imaging; 
using System.Windows.Navigation; 
using System.Windows.Shapes;  

namespace WpfApplication3 { 
   /// <summary> 
      /// Interaction logic for UserControl1.xaml 
   /// </summary> 
	
   public partial class UserControl1 : UserControl { 
	
      public UserControl1() { 
         InitializeComponent(); 
      }
		
      public static readonly DependencyProperty SetTextProperty = 
         DependencyProperty.Register("SetText", typeof(string), typeof(UserControl1), new 
            PropertyMetadata("", new PropertyChangedCallback(OnSetTextChanged))); 
				
      public string SetText { 
         get { return (string)GetValue(SetTextProperty); } 
         set { SetValue(SetTextProperty, value); } 
      } 
		
      private static void OnSetTextChanged(DependencyObject d,
         DependencyPropertyChangedEventArgs e) { 
         UserControl1 UserControl1Control = d as UserControl1; 
         UserControl1Control.OnSetTextChanged(e); 
      } 
		
      private void OnSetTextChanged(DependencyPropertyChangedEventArgs e) { 
         tbTest.Text = e.NewValue.ToString(); 
      }  
   } 
}

Here is the XAML file in which the TextBlock is defined as a user control and the Text property will be assigned to it by the SetText dependency property.

The following XAML code creates a user control and initializes its SetText dependency property.

<Window x:Class = "WpfApplication3.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:views = "clr-namespace:WpfApplication3"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <views:UserControl1 SetText = "Hellow World"/> 
   </Grid> 
	
</Window>

Let's run this application. You can immediately observe that in our MainWindow, the dependency property for user control has been successfully used as a Text.

A routed event is a type of event that can invoke handlers on multiple listeners in an element tree rather than just the object that raised the event. It is basically a CLR event that is supported by an instance of the Routed Event class. It is registered with the WPF event system. RoutedEvents have three main routing strategies which are as follows −

  • Direct Event
  • Bubbling Event
  • Tunnel Event

Direct Event

A direct event is similar to events in Windows forms which are raised by the element in which the event is originated.

Unlike a standard CLR event, direct routed events support class handling and they can be used in Event Setters and Event Triggers within your style of your Custom Control.

A good example of a direct event would be the MouseEnter event.

Bubbling Event

A bubbling event begins with the element where the event is originated. Then it travels up the visual tree to the topmost element in the visual tree. So, in WPF, the topmost element is most likely a window.

Tunnel Event

Event handlers on the element tree root are invoked and then the event travels down the visual tree to all the children nodes until it reaches the element in which the event originated.

The difference between a bubbling and a tunneling event is that a tunneling event will always start with a preview.

In a WPF application, events are often implemented as a tunneling/bubbling pair. So, you'll have a preview MouseDown and then a MouseDown event.

Given below is a simple example of a Routed event in which a button and three text blocks are created with some properties and events.

<Window x:Class = "WPFRoutedEvents.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "450" Width = "604" ButtonBase.Click  = "Window_Click" >
	
   <Grid> 
      <StackPanel Margin = "20" ButtonBase.Click = "StackPanel_Click">
		
         <StackPanel Margin = "10"> 
            <TextBlock Name = "txt1" FontSize = "18" Margin = "5" Text = "This is a TextBlock 1" /> 
            <TextBlock Name = "txt2" FontSize = "18" Margin = "5" Text = "This is a TextBlock 2" /> 
            <TextBlock Name = "txt3" FontSize = "18" Margin = "5" Text = "This is a TextBlock 3" /> 
         </StackPanel> 
			
         <Button Margin = "10" Content = "Click me" Click = "Button_Click" Width = "80"/> 
      </StackPanel> 
   </Grid> 
	
</Window>

Here is the C# code for the Click events implementation for Button, StackPanel, and Window.

using System.Windows; 
 
namespace WPFRoutedEvents { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary>
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         txt1.Text = "Button is Clicked"; 
      } 
		
      private void StackPanel_Click(object sender, RoutedEventArgs e) { 
         txt2.Text = "Click event is bubbled to Stack Panel"; 
      } 
		
      private void Window_Click(object sender, RoutedEventArgs e) { 
         txt3.Text = "Click event is bubbled to Window"; 
      }
		
   } 
}

When you compile and execute the above code, it will produce the following window −

When you click on the button, the text blocks will get updated, as shown below.

If you want to stop the routed event at any particular level, then you will need to set the e.Handled = true;

Let’s change the StackPanel_Click event as shown below −

private void StackPanel_Click(object sender, RoutedEventArgs e) { 
   txt2.Text = "Click event is bubbled to Stack Panel"; 
   e.Handled = true; 
}

When you click on the button, you will observe that the click event will not be routed to the window and will stop at the stackpanel and the 3rd text block will not be updated.

Custom Routed Events

In .NET framework, custom routed event can also be defined. You need to follow the steps given below to define a custom routed event in C#.

  • Declare and register your routed event with system call RegisterRoutedEvent.

  • Specify the Routing Strategy, i.e. Bubble, Tunnel, or Direct.

  • Provide the event handler.

Let’s take an example to understand more about custom routed events. Follow the steps given below −

  • Create a new WPF project with WPFCustomRoutedEvent

  • Right click on your solution and select Add > New Item...

  • The following dialog will open, now select Custom Control (WPF) and name it MyCustomControl.

  • Click the Add button and you will see that two new files (Themes/Generic.xaml and MyCustomControl.cs) will be added in your solution.

The following XAML code sets the style for the custom control in Generic.xaml file.

<ResourceDictionary 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomRoutedEvent">
	
   <Style TargetType = "{x:Type local:MyCustomControl}"> 
      <Setter Property = "Margin" Value = "50"/> 
      <Setter Property = "Template"> 
         <Setter.Value> 
            <ControlTemplate TargetType = "{x:Type local:MyCustomControl}">
				
               <Border Background = "{TemplateBinding Background}" 
                  BorderBrush = "{TemplateBinding BorderBrush}" 
                  BorderThickness = "{TemplateBinding BorderThickness}"> 
                  <Button x:Name = "PART_Button" Content = "Click Me" /> 
               </Border> 
					
            </ControlTemplate> 
         </Setter.Value> 
      </Setter> 
   </Style> 
	
</ResourceDictionary>

Given below is the C# code for the MyCustomControl class which inherits from the Control class in which a custom routed event Click is created for the custom control.

using System.Windows; 
using System.Windows.Controls;  

namespace WPFCustomRoutedEvent { 

   public class MyCustomControl : Control { 
	
      static MyCustomControl() { 
         DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), 
            new FrameworkPropertyMetadata(typeof(MyCustomControl))); 
      } 
		
      public override void OnApplyTemplate() { 
         base.OnApplyTemplate();
			
         //demo purpose only, check for previous instances and remove the handler first 
         var button  =  GetTemplateChild("PART_Button") as Button; 
         if (button ! =  null) 
         button.Click + =  Button_Click;  
      } 
		
      void Button_Click(object sender, RoutedEventArgs e) { 
         RaiseClickEvent(); 
      } 
		
      public static readonly RoutedEvent ClickEvent  =  
         EventManager.RegisterRoutedEvent("Click", RoutingStrategy.Bubble, 
         typeof(RoutedEventHandler), typeof(MyCustomControl)); 
			
      public event RoutedEventHandler Click { 
         add { AddHandler(ClickEvent, value); } 
         remove { RemoveHandler(ClickEvent, value); } 
      } 
		
      protected virtual void RaiseClickEvent() { 
         RoutedEventArgs args = new RoutedEventArgs(MyCustomControl.ClickEvent); 
         RaiseEvent(args); 
      }
		
   } 
}

Here is the custom routed event implementation in C# which will display a message box when the user clicks it.

using System.Windows;  

namespace WPFCustomRoutedEvent { 
   // <summary> 
      // Interaction logic for MainWindow.xaml
   // </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void MyCustomControl_Click(object sender, RoutedEventArgs e) { 
         MessageBox.Show("It is the custom routed event of your custom control"); 
      } 
		
   } 
}

Here is the implementation in MainWindow.xaml to add the custom control with a routed event Click.

<Window x:Class = "WPFCustomRoutedEvent.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomRoutedEvent"
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <local:MyCustomControl Click = "MyCustomControl_Click" /> 
   </Grid> 
	
</Window>

When the above code is compiled and executed, it will produce the following window which contains a custom control.

When you click on the custom control, it will produce the following message.

Windows Presentation Foundation (WPF) allows developers to easily build and create visually enriched UI based applications.

  • The classical UI elements or controls in other UI frameworks are also enhanced in WPF applications.

  • All of the standard WPF controls can be found in the Toolbox which is a part of the System.Windows.Controls.

  • These controls can also be created in XAML markup language.

The complete inheritance hierarchy of WPF controls are as follows −

The following table contains a list of controls which we will discuss in the subsequent chapters.

Sr. No. Controls & Description
1 Button

A control that responds to user input

2 Calendar

Represents a control that enables a user to select a date by using a visual calendar display.

3 CheckBox

A control that a user can select or clear.

4 ComboBox

A drop-down list of items a user can select from.

5 ContextMenu

Gets or sets the context menu element that should appear whenever the context menu is requested through user interface (UI) from within this element.

6 DataGrid

Represents a control that displays data in a customizable grid.

7 DatePicker

A control that lets a user select a date.

8 Dialogs

An application may also display additional windows to help the user gather or display important information.

9 GridView

A control that presents a collection of items in rows and columns that can scroll horizontally.

10 Image

A control that presents an image.

11 Label

Displays text on a form. Provides support for access keys.

12 ListBox

A control that presents an inline list of items that the user can select from.

13 Menus

Represents a Windows menu control that enables you to hierarchically organize elements associated with commands and event handlers.

14 PasswordBox

A control for entering passwords.

15 Popup

Displays content on top of existing content, within the bounds of the application window.

16 ProgressBar

A control that indicates progress by displaying a bar.

17 RadioButton

A control that allows a user to select a single option from a group of options.

18 ScrollViewer

A container control that lets the user pan and zoom its content.

19 Slider

A control that lets the user select from a range of values by moving a Thumb control along a track.

20 TextBlock

A control that displays text.

21 ToggleButton

A button that can be toggled between 2 states.

22 ToolTip

A pop-up window that displays information for an element.

23 Window

The root window which provides minimize/maximize option, Title bar, border and close button

24 3rd Party Controls

Use third-party controls in your WPF applications.

We will discuss all these controls one by one with their implementation.

The layout of controls is very important and critical for application usability. It is used to arrange a group of GUI elements in your application. There are certain important things to consider while selecting layout panels −

  • Positions of the child elements
  • Sizes of the child elements
  • Layering of overlapping child elements on top of each other

Fixed pixel arrangement of controls doesn’t work when the application is to be sed on different screen resolutions. XAML provides a rich set of built-in layout panels to arrange GUI elements in an appropriate way. Some of the most commonly used and popular layout panels are as follows −

Sr. No. Panels & Description
1 Stack Panel

Stack panel is a simple and useful layout panel in XAML. In stack panel, child elements can be arranged in a single line, either horizontally or vertically, based on the orientation property.

2 Wrap Panel

In WrapPanel, child elements are positioned in sequential order, from left to right or from top to bottom based on the orientation property.

3 Dock Panel

DockPanel defines an area to arrange child elements relative to each other, either horizontally or vertically. With DockPanel you can easily dock child elements to top, bottom, right, left and center using the Dock property.

4 Canvas Panel

Canvas panel is the basic layout panel in which the child elements can be positioned explicitly using coordinates that are relative to the Canvas any side such as left, right, top and bottom.

5 Grid Panel

A Grid Panel provides a flexible area which consists of rows and columns. In a Grid, child elements can be arranged in tabular form.

Nesting of layout means the use layout panel inside another layout, e.g. define stack panels inside a grid. This concept is widely used to take the advantages of multiple layouts in an application. In the following example, we will be using stack panels inside a grid.

Let’s have a look at the following XAML code.

<Window x:Class = "WPFNestingLayouts.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFNestingLayouts" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid Background = "AntiqueWhite"> 
      <Grid.RowDefinitions> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "*" /> 
      </Grid.ColumnDefinitions> 
		
      <Label Content = "Employee Info" FontSize = "15"
         FontWeight = "Bold" Grid.Column = "0" Grid.Row = "0"/> 
			
      <StackPanel Grid.Column = "0" Grid.Row = "1" Orientation = "Horizontal"> 
         <Label Content = "Name"  VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtName" Text = "Muhammad Ali" VerticalAlignment = "Center"
            Width = "200">
         </TextBox> 
      </StackPanel>
		
      <StackPanel Grid.Column = "0" Grid.Row = "2" Orientation = "Horizontal"> 
         <Label Content = "ID" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtCity" Text = "421" VerticalAlignment = "Center"
            Width = "50">
         </TextBox> 
      </StackPanel>
		
      <StackPanel Grid.Column = "0" Grid.Row = "3" Orientation = "Horizontal"> 
         <Label Content = "Age" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtState" Text = "32" VerticalAlignment = "Center"
            Width = "50"></TextBox> 
      </StackPanel> 
		
      <StackPanel Grid.Column = "0" Grid.Row = "4" Orientation = "Horizontal"> 
         <Label Content = "Title" VerticalAlignment = "Center" Width = "70"/> 
         <TextBox Name = "txtCountry" Text = "Programmer" VerticalAlignment = "Center"
            Width = "200"></TextBox> 
      </StackPanel> 
		
   </Grid> 
	
</Window>

When you compile and execute the above code, it will produce the following window.

We recommend that you execute the above example code and try other nesting layouts.

Windows Presentation Foundation (WPF) provides a powerful API with the help of which applications can get input from various devices such as mouse, keyboard, and touch panels. In this chapter, we will discuss the following types of input which can be handled in WPF applications −

Sr. No. Inputs & Description
1 Mouse

There are different types of mouse inputs such as MouseDown, MouseEnter, MouseLeave, etc.

2 Keyboard

There are many types of keyboard inputs such as KeyDown, KeyUp, TextInput, etc.

3 ContextMenu or RoutedCommands

RoutedCommands enable input handling at a more semantic level. These are actually simple instructions as New, Open, Copy, Cut, and Save.

4 Multi Touch

Windows 7 and its higher versions have the ability to receive input from multiple touchsensitive devices. WPF applications can also handle touch input as other input, such as the mouse or keyboard, by raising events when a touch occurs.

Command line argument is a mechanism where a user can pass a set of parameters or values to a WPF application when it is executed. These arguments are very important to control an application from outside, for example, if you want to open a Word document from the command prompt, then you can use this command “C:\> start winword word1.docx” and it will open word1.docx document.

Command line arguments are handled in Startup function. Following is a simple example which shows how to pass command line arguments to a WPF application. Let’s create a new WPF application with the name WPFCommandLine.

  • Drag one textbox from the toolbox to the design window.

  • In this example, we will pass a txt file path to our application as command line parameter.

  • The program will read the txt file and then write all the text on the text box.

  • The following XAML code creates a textbox and initializes it with some properties.

<Window x:Class = "WPFCommandLine.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFCommandLine" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"  
         Height = "180" Margin = "100" TextWrapping = "Wrap" 
         VerticalAlignment = "Top" Width = "300"/> 
   </Grid> 
	
</Window>
  • Now subscribe the Startup event in App.xaml file as shown below.
<Application x:Class = "WPFCommandLine.App" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   xmlns:local = "clr-namespace:WPFCommandLine" 
   StartupUri = "MainWindow.xaml" Startup = "app_Startup"> 
	
   <Application.Resources> 
          
   </Application.Resources>
	
</Application>
  • 다음은 명령 줄 인수를 가져 오는 App.xaml.cs의 app_Startup 이벤트 구현입니다.

using System.Windows;
  
namespace WPFCommandLine { 
   /// <summary> 
      /// Interaction logic for App.xaml 
   /// </summary> 
	
   public partial class App : Application { 
      public static string[] Args;
		
      void app_Startup(object sender, StartupEventArgs e) { 
         // If no command line arguments were provided, don't process them 
         if (e.Args.Length == 0) return;
			
         if (e.Args.Length > 0) { 
            Args = e.Args; 
         } 
      } 
   } 
}
  • 이제 MainWindow 클래스에서 프로그램은 txt 파일을 열고 모든 텍스트를 텍스트 상자에 씁니다.

  • 오류가 발견되면 프로그램은 텍스트 상자에 오류 메시지를 표시합니다.

using System; 
using System.IO; 
using System.Windows;  

namespace WPFCommandLine { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         String[] args = App.Args;
			
         try {
            // Open the text file using a stream reader. 
            using (StreamReader sr = new StreamReader(args[0])) { 
               // Read the stream to a string, and write  
               // the string to the text box 
               String line = sr.ReadToEnd(); 
               textBox.AppendText(line.ToString()); 
               textBox.AppendText("\n"); 
            } 
         } 
         catch (Exception e) { 
            textBox.AppendText("The file could not be read:"); 
            textBox.AppendText("\n"); 
            textBox.AppendText(e.Message); 
         } 
      } 
   } 
}
  • 위의 코드가 컴파일되고 실행되면이 프로그램에는 명령 줄 인수가 필요하기 때문에 텍스트 상자가있는 빈 창이 생성됩니다. 따라서 Visual Studio는 명령 줄 매개 변수를 사용하여 애플리케이션을 쉽게 실행할 수있는 방법을 제공합니다.

  • 솔루션 탐색기에서 WPF 프로젝트를 마우스 오른쪽 버튼으로 클릭하고 속성을 선택하면 다음 창이 표시됩니다.

  • 디버그 옵션을 선택하고 명령 줄 인수에 파일 경로를 작성합니다.

  • Test.txt로 txt 파일을 만들고 해당 파일에 텍스트를 작성하고 원하는 위치에 저장합니다. 이 경우 txt 파일은“D:\”하드 드라이브.

  • 프로젝트의 변경 사항을 저장하고 지금 애플리케이션을 컴파일하고 실행하십시오. 프로그램이 Text.txt 파일에서 읽는 TextBox의 텍스트를 볼 수 있습니다.

이제 컴퓨터의 파일 이름을 다음에서 변경해 보겠습니다. Test.txt ...에 Test1.txt 프로그램을 다시 실행하면 텍스트 상자에 해당 오류 메시지가 표시됩니다.

위 코드를 실행하고 모든 단계에 따라 애플리케이션을 성공적으로 실행하는 것이 좋습니다.

데이터 바인딩은 Windows 런타임 앱이 데이터를 표시하고 상호 작용할 수있는 간단하고 쉬운 방법을 제공하는 WPF 응용 프로그램의 메커니즘입니다. 이 메커니즘에서 데이터 관리는 데이터 방식과 완전히 분리됩니다.

데이터 바인딩을 사용하면 UI 요소와 사용자 인터페이스의 데이터 개체 간의 데이터 흐름이 가능합니다. 바인딩이 설정되고 데이터 또는 비즈니스 모델이 변경되면 업데이트가 UI 요소에 자동으로 반영되고 그 반대의 경우도 마찬가지입니다. 표준 데이터 소스가 아니라 페이지의 다른 요소에 바인딩 할 수도 있습니다.

데이터 바인딩은 두 가지 유형입니다- one-way data bindingtwo-way data binding.

단방향 데이터 바인딩

단방향 바인딩에서 데이터는 원본 (데이터를 보유하는 개체)에서 대상 (데이터를 표시하는 개체)으로 바인딩됩니다.

  • 단방향 데이터 바인딩을 자세히 이해하는 간단한 예를 들어 보겠습니다. 먼저, 이름으로 새 WPF 프로젝트를 만듭니다.WPFDataBinding.

  • 다음 XAML 코드는 레이블 2 개, 텍스트 상자 2 개, 단추 1 개를 만들고 일부 속성을 사용하여 초기화합니다.

<Window x:Class = "WPFDataBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataBinding" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
	
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "Auto" /> 
         <ColumnDefinition Width = "200" /> 
      </Grid.ColumnDefinitions>
		
      <Label Name = "nameLabel" Margin = "2">_Name:</Label> 
		
      <TextBox Name = "nameText" Grid.Column = "1" Margin = "2" 
         Text = "{Binding Name, Mode = OneWay}"/>  
			
      <Label Name = "ageLabel" Margin = "2" Grid.Row = "1">_Age:</Label> 
		
      <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "2" 
         Text = "{Binding Age, Mode = OneWay}"/>  
			
      <StackPanel Grid.Row = "2" Grid.ColumnSpan = "2"> 
         <Button Content = "_Show..." Click="Button_Click" /> 
      </StackPanel> 
		
   </Grid> 
</Window>
  • 두 텍스트 상자의 텍스트 속성은 아래와 같이 Person 클래스의 클래스 변수 인 "Name"과 "Age"에 바인딩됩니다.

  • Person 클래스에는 두 개의 변수 만 있습니다. NameAge, 그 객체는 다음에서 초기화됩니다. MainWindow 수업.

  • XAML 코드에서 우리는 속성 Name 및 Age에 바인딩하지만 속성이 속한 개체를 선택하지 않았습니다.

  • 더 쉬운 방법은 객체를 DataContext 다음 C # 코드에서 바인딩하는 속성 MainWindowconstructor.

using System.Windows;  
namespace WPFDataBinding { 

   public partial class MainWindow : Window {
	
      Person person = new Person { Name = "Salman", Age = 26 };
		
      public MainWindow() { 
         InitializeComponent(); 
         this.DataContext = person; 
      } 
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         string message = person.Name + " is " + person.Age; 
         MessageBox.Show(message); 
      } 
   } 
	
   public class Person { 
	
      private string nameValue;
		
      public string Name { 
         get { return nameValue; } 
         set { nameValue = value; } 
      } 
		
      private double ageValue; 
		
      public double Age { 
         get { return ageValue; } 
				
         set { 
            if (value != ageValue) { 
               ageValue = value; 
            } 
         } 
      }
		
   } 
}
  • 이 응용 프로그램을 실행하면 MainWindow에서 해당 Person 개체의 이름과 나이에 성공적으로 바인딩되었음을 즉시 확인할 수 있습니다.

누를 때 Show 버튼을 누르면 메시지 상자에 이름과 나이가 표시됩니다.

대화 상자에서 이름과 나이를 변경해 보겠습니다.

이제 표시 버튼을 클릭하면 동일한 메시지가 다시 표시됩니다.

이는 데이터 바인딩 모드가 XAML 코드에서 단방향으로 설정되기 때문입니다. 업데이트 된 데이터를 표시하려면 양방향 데이터 바인딩을 이해해야합니다.

양방향 데이터 바인딩

양방향 바인딩에서 사용자는 사용자 인터페이스를 통해 데이터를 수정하고 해당 데이터를 소스에서 업데이트 할 수 있습니다. 사용자가보기를 보는 동안 소스가 변경되면보기를 업데이트하려고합니다.

같은 예를 들어 보지만 여기서는 XAML 코드에서 바인딩 모드를 단방향에서 양방향으로 변경합니다.

<Window x:Class = "WPFDataBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataBinding" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
	
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <Grid.ColumnDefinitions> 
         <ColumnDefinition Width = "Auto" /> 
         <ColumnDefinition Width = "200" /> 
      </Grid.ColumnDefinitions> 
		
      <Label Name = "nameLabel" Margin = "2">_Name:</Label> 
      <TextBox Name = "nameText" Grid.Column = "1" Margin = "2" 
         Text = "{Binding Name, Mode = TwoWay}"/>  
      <Label Name = "ageLabel" Margin = "2" Grid.Row = "1">_Age:</Label> 
      <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "2" 
         Text = "{Binding Age, Mode = TwoWay}"/> 
			
      <StackPanel Grid.Row = "2" Grid.ColumnSpan = "2"> 
         <Button Content = "_Show..." Click = "Button_Click" /> 
      </StackPanel>
		
   </Grid>
	
</Window>

이 애플리케이션을 다시 실행 해 보겠습니다.

동일한 출력을 생성합니다-

이제 이름과 나이 값을 변경해 보겠습니다.

지금 표시 버튼을 클릭하면 업데이트 된 메시지가 표시됩니다.

개념을 더 잘 이해하려면 두 경우 모두 위의 코드를 실행하는 것이 좋습니다.

리소스는 일반적으로 두 번 이상 사용할 것으로 예상되는 일부 개체와 연결된 정의입니다. 컨트롤 또는 현재 창에 대해 로컬로 데이터를 저장하거나 전체 응용 프로그램에 대해 전역 적으로 데이터를 저장할 수있는 기능입니다.

객체를 리소스로 정의하면 다른 위치에서 객체에 액세스 할 수 있습니다. 그 의미는 객체를 재사용 할 수 있다는 것입니다. 리소스는 리소스 사전에 정의되며 모든 개체를 리소스로 정의하여 효과적으로 공유 가능한 자산으로 만들 수 있습니다. 고유 키는 XAML 리소스에 지정되며 해당 키와 함께 StaticResource 태그 확장을 사용하여 참조 할 수 있습니다.

자원은 두 가지 유형이 있습니다-

  • StaticResource
  • DynamicResource

StaticResource는 일회성 조회 인 반면 DynamicResource는 데이터 바인딩처럼 작동합니다. 속성이 특정 리소스 키와 연결되어 있음을 기억합니다. 해당 키와 연결된 개체가 변경되면 동적 리소스가 대상 속성을 업데이트합니다.

다음은 SolidColorBrush 리소스에 대한 간단한 응용 프로그램입니다.

  • 이름으로 새 WPF 프로젝트를 만들어 보겠습니다. WPFResouces.

  • 두 개의 Rectangle을 드래그하고 다음 XAML 코드와 같이 속성을 설정합니다.

<Window x:Class = "WPFResources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFResources" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Window.Resources> 
      <SolidColorBrush x:Key = "brushResource" Color = "Blue" /> 
   </Window.Resources> 
	
   <StackPanel> 
      <Rectangle Height = "50" Margin = "20" Fill = "{StaticResource brushResource}" /> 
      <Rectangle Height = "50" Margin = "20" Fill = "{DynamicResource brushResource}" /> 
      <Button x:Name = "changeResourceButton"
         Content = "_Change Resource" Click = "changeResourceButton_Click" /> 
   </StackPanel> 
	
</Window>
  • 위의 XAML 코드에서 한 사각형에는 StaticResource가 있고 다른 사각형에는 DynamicResource가 있고 brushResource의 색상은 Bisque입니다.

  • 코드를 컴파일하고 실행하면 다음 MainWindow가 생성됩니다.

"자원 변경"버튼을 클릭하면 DynamicResource가있는 사각형의 색상이 빨간색으로 변경되는 것을 볼 수 있습니다.

리소스 범위

리소스는 resource dictionaries하지만 리소스 사전을 정의 할 수있는 곳은 많습니다. 위의 예에서 리소스 사전은 창 / 페이지 수준에서 정의됩니다. 어떤 사전에서 리소스가 정의되면 해당 리소스의 범위가 즉시 제한됩니다. 따라서 범위, 즉 리소스를 사용할 수있는 위치는 정의한 위치에 따라 다릅니다.

  • 그리드의 리소스 사전에 리소스를 정의하면 해당 그리드 및 하위 요소에서만 액세스 할 수 있습니다.

  • 창 / 페이지에서 정의하면 해당 창 / 페이지의 모든 요소에서 액세스 할 수 있습니다.

  • 앱 루트는 App.xaml 리소스 사전에서 찾을 수 있습니다. 애플리케이션의 루트이므로 여기에 정의 된 리소스는 전체 애플리케이션으로 범위가 지정됩니다.

리소스의 범위에 관한 한 가장 자주 응용 프로그램 수준, 페이지 수준 및 Grid, StackPanel 등과 같은 특정 요소 수준입니다.

위의 응용 프로그램은 창 / 페이지 수준에 리소스가 있습니다.

리소스 사전

XAML 앱의 리소스 사전은 리소스 사전이 별도의 파일에 보관됨을 의미합니다. 거의 모든 XAML 앱에서 뒤 따릅니다. 별도의 파일에 리소스를 정의하면 다음과 같은 이점이 있습니다.

  • 리소스 사전에 정의 된 리소스와 UI 관련 코드를 구분합니다.

  • App.xaml과 같은 별도의 파일에 모든 리소스를 정의하면 앱에서 사용할 수 있습니다.

그렇다면 별도의 파일에있는 리소스 사전의 리소스를 어떻게 정의할까요? 글쎄, 그것은 매우 쉽습니다, 아래 주어진 단계에 따라 Visual Studio를 통해 새 리소스 사전을 추가하십시오.

  • 솔루션에서 새 폴더를 추가하고 이름을 ResourceDictionaries.

  • 이 폴더를 마우스 오른쪽 버튼으로 클릭하고 하위 메뉴 항목 추가에서 리소스 사전을 선택하고 이름을 지정합니다. DictionaryWithBrush.xaml

이제 동일한 예제를 보겠습니다. 여기서는 앱 수준에서 리소스 사전을 정의합니다. MainWindow.xaml의 XAML 코드는 다음과 같습니다.

<Window x:Class = "WPFResources.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFResources" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <StackPanel> 
      <Rectangle Height = "50" Margin = "20" Fill = "{StaticResource brushResource}" /> 
      <Rectangle Height = "50" Margin = "20" Fill = "{DynamicResource brushResource}" /> 
      <Button x:Name = "changeResourceButton"
         Content = "_Change Resource" Click = "changeResourceButton_Click" /> 
   </StackPanel> 
	
</Window>

다음은 DictionaryWithBrush.xaml의 구현입니다.

<ResourceDictionary xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"> 
   <SolidColorBrush x:Key = "brushResource" Color = "Blue" /> 
</ResourceDictionary>

다음은 app.xaml의 구현입니다.

<Application x:Class="WPFResources.App" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   StartupUri = "MainWindow.xaml"> 
	
   <Application.Resources> 
      <ResourceDictionary Source = " XAMLResources\ResourceDictionaries\DictionaryWithBrush.xaml"/> 
   </Application.Resources> 
	
</Application>

위의 코드가 컴파일되고 실행되면 다음과 같은 출력이 생성됩니다.

리소스 변경 버튼을 클릭하면 사각형의 색상이 빨간색으로 변경됩니다.

위의 코드를 실행하고 더 많은 리소스 (예 : 배경색)를 시도하는 것이 좋습니다.

템플릿은 컨트롤의 전체적인 모양과 시각적 모양을 설명합니다. 각 컨트롤에는 컨트롤의 모양을 제공하는 연결된 기본 템플릿이 있습니다. WPF 응용 프로그램에서 컨트롤의 시각적 동작과 시각적 모양을 사용자 지정하려는 경우 고유 한 템플릿을 쉽게 만들 수 있습니다.

논리와 템플릿 간의 연결은 데이터 바인딩을 통해 얻을 수 있습니다. 주요 차이점stylestemplates 아래에 나열되어 있습니다-

  • 스타일은 해당 컨트롤의 기본 속성으로 만 컨트롤의 모양을 변경할 수 있습니다.

  • 템플릿을 사용하면 스타일보다 컨트롤의 더 많은 부분에 액세스 할 수 있습니다. 컨트롤의 기존 동작과 새 동작을 모두 지정할 수도 있습니다.

가장 일반적으로 사용되는 두 가지 유형의 템플릿이 있습니다.

  • 컨트롤 템플릿
  • 데이터 템플릿

컨트롤 템플릿

컨트롤 템플릿은 컨트롤의 시각적 모양을 정의합니다. 모든 UI 요소는 동작뿐만 아니라 모양도 가지고 있습니다. 예를 들어 버튼에는 모양과 동작이 있습니다. 클릭 이벤트 또는 마우스 호버 이벤트는 클릭 및 호버에 대한 응답으로 실행되는 동작이며 컨트롤 템플릿에 의해 변경 될 수있는 버튼의 기본 모양도 있습니다.

간단한 예를 들어 보겠습니다. 두 개의 버튼 (하나는 템플릿을 사용하고 다른 하나는 기본 버튼)을 만들고 일부 속성을 사용하여 초기화합니다.

<Window x:Class = "TemplateDemo.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Window.Resources> 
      <ControlTemplate x:Key = "ButtonTemplate" TargetType = "Button">
		
         <Grid> 
            <Ellipse x:Name = "ButtonEllipse" Height = "100" Width = "150" > 
               <Ellipse.Fill> 
                  <LinearGradientBrush StartPoint = "0,0.2" EndPoint = "0.2,1.4"> 
                     <GradientStop Offset = "0" Color = "Red" /> 
                     <GradientStop Offset = "1" Color = "Orange" /> 
                  </LinearGradientBrush> 
               </Ellipse.Fill> 
            </Ellipse> 
				
            <ContentPresenter Content = "{TemplateBinding Content}" 
               HorizontalAlignment = "Center" VerticalAlignment = "Center" /> 
         </Grid> 
			
         <ControlTemplate.Triggers> 
			
            <Trigger Property = "IsMouseOver" Value = "True"> 
               <Setter TargetName = "ButtonEllipse" Property = "Fill" > 
                  <Setter.Value> 
                     <LinearGradientBrush StartPoint = "0,0.2" EndPoint = "0.2,1.4"> 
                        <GradientStop Offset = "0" Color = "YellowGreen" /> 
                        <GradientStop Offset = "1" Color = "Gold" /> 
                     </LinearGradientBrush> 
                  </Setter.Value> 
               </Setter> 
            </Trigger> 
				
            <Trigger Property = "IsPressed" Value = "True"> 
               <Setter Property = "RenderTransform"> 
                  <Setter.Value> 
                     <ScaleTransform ScaleX = "0.8" ScaleY = "0.8" 
                        CenterX = "0" CenterY = "0"  /> 
                  </Setter.Value> 
               </Setter> 
               <Setter Property = "RenderTransformOrigin" Value = "0.5,0.5" /> 
            </Trigger> 
				
         </ControlTemplate.Triggers> 
			
      </ControlTemplate> 
   </Window.Resources> 
	
   <StackPanel> 
      <Button Content = "Round Button!"
         Template = "{StaticResource ButtonTemplate}" 
         Width = "150" Margin = "50" /> 
      <Button Content = "Default Button!" Height = "40" 
         Width = "150" Margin = "5" />
   </StackPanel> 
	
</Window>

위 코드를 컴파일하고 실행하면 아래와 같은 MainWindow가 표시됩니다.

사용자 정의 템플릿이있는 버튼 위로 마우스를 이동하면 아래와 같이 색상이 변경됩니다.

데이터 템플릿

데이터 템플릿은 데이터 컬렉션의 모양과 구조를 정의하고 지정합니다. 모든 UI 요소에서 데이터 표시 형식을 지정하고 정의 할 수있는 유연성을 제공합니다. 주로 ComboBox, ListBox 등과 같은 데이터 관련 항목 컨트롤에 사용됩니다.

  • 데이터 템플릿의 개념을 이해하기 위해 간단한 예를 들어 보겠습니다. 이름으로 새 WPF 프로젝트 만들기WPFDataTemplates.

  • 다음 XAML 코드에서는 레이블 및 텍스트 상자를 보관할 리소스로 데이터 템플릿을 만듭니다. 데이터를 표시하는 버튼과 목록 상자도 있습니다.

<Window x:Class = "WPFDataTemplates.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDataTemplates" 
   xmlns:loc = "clr-namespace:WPFDataTemplates" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Window.Resources> 
      <DataTemplate DataType = "{x:Type loc:Person}"> 
		
         <Grid> 
            <Grid.RowDefinitions> 
               <RowDefinition Height = "Auto" /> 
               <RowDefinition Height = "Auto" /> 
            </Grid.RowDefinitions> 
				
            <Grid.ColumnDefinitions> 
               <ColumnDefinition Width = "Auto" /> 
               <ColumnDefinition Width = "200" /> 
            </Grid.ColumnDefinitions>
				
            <Label Name = "nameLabel" Margin = "10"/> 
            <TextBox Name = "nameText" Grid.Column = "1" Margin = "10" 
               Text = "{Binding Name}"/>  
            <Label Name = "ageLabel" Margin = "10" Grid.Row = "1"/> 
            <TextBox Name = "ageText" Grid.Column = "1" Grid.Row = "1" Margin = "10" 
               Text = "{Binding Age}"/> 
         </Grid> 
			
      </DataTemplate> 
   </Window.Resources> 
	
   <Grid> 
      <Grid.RowDefinitions> 
         <RowDefinition Height = "Auto" /> 
         <RowDefinition Height = "*" /> 
      </Grid.RowDefinitions> 
		
      <ListBox ItemsSource = "{Binding}" />  
      <StackPanel Grid.Row = "1" > 
         <Button Content = "_Show..." Click = "Button_Click" Width = "80" HorizontalAlignment = "Left" Margin = "10"/> 
      </StackPanel> 
		
   </Grid> 
	
</Window>

여기 있습니다 implementation in C# 여기서 Person 객체의 목록이 DataContext에 할당되고, Person 클래스의 구현과 버튼 클릭 이벤트가 있습니다.

using System.Collections.Generic; 
using System.Windows;
  
namespace WPFDataTemplates { 

   public partial class MainWindow : Window { 
	
      Person src = new Person { Name = "Ali", Age = 27 }; 
      List<Person> people = new List<Person>(); 
		
      public MainWindow() { 
         InitializeComponent(); 
         people.Add(src); 
         people.Add(new Person { Name = "Mike", Age = 62 }); 
         people.Add(new Person { Name = "Brian", Age = 12 });  
         this.DataContext = people; 
      } 
		
      private void Button_Click(object sender, RoutedEventArgs e) { 
         string message = src.Name + " is " + src.Age; 
         MessageBox.Show(message); 
      } 
   } 
	
   public class Person { 
      private string nameValue; 
		
      public string Name { 
         get { return nameValue; } 
         set { nameValue = value; } 
      }  
		
      private double ageValue; 
		
      public double Age { 
         get { return ageValue; } 
         set { 
            if (value != ageValue) { 
            ageValue = value; 
            } 
         } 
      } 
   } 
	
}

위 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다. 하나의 목록을 포함하고 목록 상자 내부에 각 목록 상자 항목에는 레이블 및 텍스트 상자에 표시되는 Person 클래스 개체 데이터가 포함됩니다.

.NET 프레임 워크는 응용 프로그램의 모양을 개인화하고 사용자 지정하는 몇 가지 전략을 제공합니다. 스타일은 객체의 일부 속성을 설정하고 일관된 모양을 위해 여러 객체에서 이러한 특정 설정을 재사용 할 수있는 유연성을 제공합니다.

  • 스타일에서는 높이, 너비, 글꼴 크기 등과 같은 개체의 기존 속성 만 설정할 수 있습니다.

  • 컨트롤의 기본 동작 만 지정할 수 있습니다.

  • 단일 스타일에 여러 속성을 추가 할 수 있습니다.

스타일은 컨트롤 집합에 균일 한 모양이나 모양을 제공하는 데 사용됩니다. 암시 적 스타일은 지정된 형식의 모든 컨트롤에 모양을 적용하고 응용 프로그램을 단순화하는 데 사용됩니다. 세 개의 버튼이 모두 같고 너비와 높이가 같고 글꼴 크기가 같고 전경색이 같아야한다고 상상해보십시오. 모든 속성을 버튼 요소 자체에 설정할 수 있으며 모든 버튼에 대해 여전히 괜찮습니다. 다음 다이어그램을 살펴보십시오.

그러나 실제 응용 프로그램에서는 일반적으로 정확히 동일하게 표시되어야하는 이러한 항목이 많이 있습니다. 물론 버튼뿐만 아니라 일반적으로 텍스트 블록, 텍스트 상자 및 콤보 상자 등이 애플리케이션 전체에서 동일하게 보이기를 원할 것입니다. 분명히 이것을 달성하는 더 나은 방법이 있어야하며styling. 스타일은 속성 값 집합을 둘 이상의 요소에 적용하는 편리한 방법으로 생각할 수 있습니다. 다음 다이어그램을 살펴보십시오.

이 개념을 이해하기 위해 간단한 예를 들어 보겠습니다. 새 WPF 프로젝트를 만들어 시작합니다.

  • 도구 상자에서 디자인 창으로 세 개의 버튼을 드래그합니다.

  • 다음 XAML 코드는 세 개의 단추를 만들고 일부 속성을 사용하여 초기화합니다.

<Window x:Class = "WPFStyle.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace: WPFStyle" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <Button Content = "Button1" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
      <Button Content = "Button2" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
      <Button Content = "Button3" Height = "30" Width = "80" 
         Foreground = "Blue" FontSize = "12" Margin = "10"/> 
   </StackPanel> 
	
</Window>

위의 코드를 보면 모든 버튼의 높이, 너비, 전경색, 글꼴 크기 및 여백 속성이 동일하다는 것을 알 수 있습니다. 이제 위의 코드가 컴파일되고 실행되면 다음과 같은 창이 표시됩니다.

이제 같은 예를 살펴 보겠습니다.하지만 이번에는 style.

<Window x:Class = "XAMLStyle.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"
   xmlns:local = "clr-namespace:XAMLStyle" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Window.Resources> 
      <Style x:Key = "myButtonStyle" TargetType = "Button"> 
         <Setter Property = "Height" Value = "30" /> 
         <Setter Property = "Width" Value = "80" /> 
         <Setter Property = "Foreground" Value = "Blue" /> 
         <Setter Property = "FontSize" Value = "12" /> 
         <Setter Property = "Margin" Value = "10" /> 
      </Style> 
   </Window.Resources> 
	
   <StackPanel> 
      <Button Content = "Button1" Style = "{StaticResource myButtonStyle}" /> 
      <Button Content = "Button2" Style = "{StaticResource myButtonStyle}" /> 
      <Button Content = "Button3" Style="{StaticResource myButtonStyle}" /> 
   </StackPanel> 
	
</Window>

스타일은 리소스 사전에 정의되어 있으며 각 스타일에는 고유 한 키 식별자와 대상 유형이 있습니다. <style> 안에는 스타일에 포함될 각 속성에 대해 여러 setter 태그가 정의되어있는 것을 볼 수 있습니다.

위의 예에서 각 버튼의 모든 공통 속성은 이제 스타일로 정의 된 다음 StaticResource 태그 확장을 통해 스타일 속성을 설정하여 고유 키로 각 ​​버튼에 스타일이 할당됩니다.

위 코드를 컴파일하고 실행하면 다음과 같은 화면 (동일한 출력)이 표시됩니다.

이렇게하는 것의 장점은 즉시 명백합니다. 우리는 그 스타일을 범위의 어느 곳에서나 재사용 할 수 있습니다. 변경해야하는 경우 각 요소 대신 스타일 정의에서 한 번만 변경하면됩니다.

어떤 수준에서 스타일이 정의되면 해당 스타일의 범위가 즉시 제한됩니다. 따라서 범위, 즉 스타일을 사용할 수있는 위치는 정의한 위치에 따라 다릅니다. 스타일은 다음 수준에서 정의 할 수 있습니다-

Sr. 아니요 레벨 및 설명
1 제어 수준

컨트롤 수준에서 스타일을 정의하는 것은 해당 특정 컨트롤에만 적용 할 수 있습니다. 다음은 버튼과 TextBlock에 고유 한 스타일이있는 제어 수준의 예입니다.

2 레이아웃 수준

모든 레이아웃 수준에서 스타일을 정의하면 해당 레이아웃 및 하위 요소에서만 액세스 할 수 있습니다.

창 수준

창 수준에서 스타일을 정의하면 해당 창의 모든 요소에서 액세스 할 수 있습니다.

4 신청 수준

앱 수준에서 스타일을 정의하면 전체 애플리케이션에서 액세스 할 수 있습니다. 동일한 예를 들어 보겠습니다.하지만 여기서는 app.xaml 파일에 스타일을 넣어 애플리케이션 전체에서 액세스 할 수 있도록합니다.

기본적으로 트리거를 사용하면 속성 값을 변경하거나 속성 값에 따라 작업을 수행 할 수 있습니다. 따라서 새 컨트롤을 만들지 않고도 컨트롤의 모양 및 / 또는 동작을 동적으로 변경할 수 있습니다.

트리거는 특정 조건이 충족 될 때 특정 속성의 값을 변경하는 데 사용됩니다. 트리거는 일반적으로 특정 컨트롤에 적용되는 문서의 스타일 또는 루트에서 정의됩니다. 트리거에는 세 가지 유형이 있습니다.

  • 속성 트리거
  • 데이터 트리거
  • 이벤트 트리거

속성 트리거

속성 트리거에서 한 속성에서 변경이 발생하면 다른 속성에서 즉시 또는 애니메이션 변경을 가져옵니다. 예를 들어 속성 ​​트리거를 사용하여 마우스를 단추 위로 가져갈 때 단추 모양을 변경할 수 있습니다.

다음 예제 코드는 마우스가 버튼 위에있을 때 버튼의 전경색을 변경하는 방법을 보여줍니다.

<Window x:Class = "WPFPropertyTriggers.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Window.Resources> 
      <Style x:Key = "TriggerStyle" TargetType = "Button"> 
         <Setter Property = "Foreground" Value = "Blue" /> 
         <Style.Triggers> 
            <Trigger Property = "IsMouseOver" Value = "True"> 
               <Setter Property = "Foreground" Value = "Green" /> 
            </Trigger> 
         </Style.Triggers> 
      </Style> 
   </Window.Resources> 
	
   <Grid> 
      <Button Width = "100" Height = "70"
         Style = "{StaticResource TriggerStyle}" Content = "Trigger"/> 
   </Grid> 
	
</Window>

위의 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다.

마우스를 버튼 위로 가져 가면 전경색이 녹색으로 바뀝니다.

데이터 트리거

데이터 트리거는 바인딩 된 데이터가 일부 조건을 충족 할 때 일부 작업을 수행합니다. 몇 가지 속성으로 확인란과 텍스트 블록이 생성되는 다음 XAML 코드를 살펴 보겠습니다. 확인란을 선택하면 전경색이 빨간색으로 변경됩니다.

<Window x:Class = "WPFDataTrigger.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "Data Trigger" Height = "350" Width = "604">
	
   <StackPanel HorizontalAlignment = "Center"> 
      <CheckBox x:Name = "redColorCheckBox" 
         Content = "Set red as foreground color" Margin = "20"/> 
			
      <TextBlock Name = "txtblock" VerticalAlignment = "Center" 
         Text = "Event Trigger" FontSize = "24" Margin = "20"> 
         <TextBlock.Style> 
            <Style> 
               <Style.Triggers> 
                  <DataTrigger Binding = "{Binding ElementName = redColorCheckBox, Path = IsChecked}" 
                     Value = "true"> 
                     <Setter Property = "TextBlock.Foreground" Value = "Red"/> 
                     <Setter Property = "TextBlock.Cursor" Value = "Hand" /> 
                  </DataTrigger> 
               </Style.Triggers> 
            </Style> 
         </TextBlock.Style> 
      </TextBlock> 
		
   </StackPanel> 
	
</Window>

위의 코드가 컴파일되고 실행되면 다음과 같은 출력이 생성됩니다.

확인란을 선택하면 텍스트 블록의 전경색이 빨간색으로 변경됩니다.

이벤트 트리거

이벤트 트리거는 특정 이벤트가 발생하면 몇 가지 작업을 수행합니다. 일반적으로 DoubleAnumatio, ColorAnimation 등과 같은 제어에 대한 일부 애니메이션을 수행하는 데 사용됩니다. 다음 예제에서는 간단한 버튼을 생성합니다. 클릭 이벤트가 시작되면 버튼 너비와 높이가 확장됩니다.

<Window x:Class = "WPFEventTrigger.MainWindow"
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button Content = "Click Me" Width = "60" Height = "30">
		
         <Button.Triggers> 
            <EventTrigger RoutedEvent = "Button.Click"> 
               <EventTrigger.Actions> 
                  <BeginStoryboard> 
                     <Storyboard> 
							
                        <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = 
                           "Width" Duration = "0:0:4"> 
                           <LinearDoubleKeyFrame Value = "60" KeyTime = "0:0:0"/> 
                           <LinearDoubleKeyFrame Value = "120" KeyTime = "0:0:1"/> 
                           <LinearDoubleKeyFrame Value = "200" KeyTime = "0:0:2"/> 
                           <LinearDoubleKeyFrame Value = "300" KeyTime = "0:0:3"/> 
                        </DoubleAnimationUsingKeyFrames>
								
                        <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = "Height" 
                           Duration = "0:0:4"> 
                           <LinearDoubleKeyFrame Value = "30" KeyTime = "0:0:0"/> 
                           <LinearDoubleKeyFrame Value = "40" KeyTime = "0:0:1"/> 
                           <LinearDoubleKeyFrame Value = "80" KeyTime = "0:0:2"/> 
                           <LinearDoubleKeyFrame Value = "150" KeyTime = "0:0:3"/> 
                        </DoubleAnimationUsingKeyFrames>
								
                     </Storyboard> 
                  </BeginStoryboard> 
               </EventTrigger.Actions> 
            </EventTrigger> 
         </Button.Triggers> 
			
      </Button> 
   </Grid> 
	
</Window>

위의 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다.

버튼을 클릭하면 두 차원 모두에서 확장되기 시작하는 것을 볼 수 있습니다.

위의 예제를 컴파일 및 실행하고 다른 속성에도 트리거를 적용하는 것이 좋습니다.

이것은 여러분이 기대하는 것과 똑같이 동작하지 않는 코드 조각의 버그 나 결함을 식별하고 수정하는 체계적인 메커니즘입니다. 한 하위 시스템의 버그를 수정하면 다른 하위 시스템에 버그가 발생할 수 있기 때문에 하위 시스템이 긴밀하게 결합 된 복잡한 응용 프로그램을 디버깅하는 것은 쉽지 않습니다.

C #에서 디버깅

WPF 애플리케이션에서 프로그래머는 C # 및 XAML과 같은 두 가지 언어를 다룹니다. C # 또는 C / C ++와 같은 절차 적 언어의 디버깅에 익숙하고 중단 점의 사용법도 알고 있다면 애플리케이션의 C # 부분을 쉽게 디버깅 할 수 있습니다.

C # 코드를 디버그하는 방법을 보여주는 간단한 예제를 살펴 보겠습니다. 이름으로 새 WPF 프로젝트 만들기WPFDebuggingDemo. 도구 상자에서 레이블 4 개, 텍스트 상자 3 개, 버튼 1 개를 드래그합니다. 다음 XAML 코드를 살펴보십시오.

<Window x:Class = "WPFDebuggingDemo.Window1" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "Window1" Height = "400" Width = "604"> 
	
   <Grid> 
      <TextBox Height = "23" Margin = "0,44,169,0" Name = "textBox1"  
         VerticalAlignment = "Top" HorizontalAlignment = "Right" Width = "120" /> 
			
      <TextBox Height = "23" Margin = "0,99,169,0" Name = "textBox2"  
         VerticalAlignment = "Top" HorizontalAlignment = "Right" Width = "120" /> 
			
      <TextBox HorizontalAlignment = "Right" Margin = "0,153,169,0"  
         Name = "textBox3" Width = "120" Height = "23" VerticalAlignment = "Top" /> 
			
      <Label Height = "28" Margin = "117,42,0,0" Name = "label1"  
         VerticalAlignment = "Top" HorizontalAlignment = "Left" Width = "120">
         Item 1</Label> 
			
      <Label Height = "28" HorizontalAlignment = "Left"  
         Margin = "117,99,0,0" Name = "label2" VerticalAlignment = "Top" Width = "120">
         Item 2</Label> 
			
      <Label HorizontalAlignment = "Left" Margin = "117,153,0,181"  
         Name = "label3" Width = "120">Item 3</Label>
			
      <Button Height = "23" HorizontalAlignment = "Right" Margin = "0,0,214,127"
         Name = "button1" VerticalAlignment = "Bottom" Width = "75"  
         Click = "button1_Click">Total</Button> 
			
      <Label Height = "28" HorizontalAlignment = "Right"  
         Margin = "0,0,169,66" Name = "label4" VerticalAlignment = "Bottom" Width = "120"/> 
			
   </Grid> 
	
</Window>

다음은 버튼 클릭 이벤트가 구현 된 C # 코드입니다.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text;
 
using System.Windows; 
using System.Windows.Controls; 
using System.Windows.Data; 
using System.Windows.Documents; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Media.Imaging; 
using System.Windows.Navigation; 
using System.Windows.Shapes;  

namespace WPFDebuggingDemo { 
   /// <summary> 
      /// Interaction logic for Window1.xaml 
   /// </summary> 
	
   public partial class Window1 : Window { 
	
      public Window1() {
         InitializeComponent();
      }
		
      private void button1_Click(object sender, RoutedEventArgs e) {
		
         if (textBox1.Text.Length > 0 && textBox2.Text.Length > 0 && textBox2.Text.Length > 0) {
            double total = Convert.ToDouble(textBox1.Text) + 
            Convert.ToDouble(textBox2.Text) + Convert.ToDouble(textBox3.Text); 
            label4.Content = total.ToString(); 
         } 
         else { 
            MessageBox.Show("Enter the value in all field.");
         } 
      } 
   } 
}

위 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다. 이제 텍스트 상자에 값을 입력하고 합계 버튼을 누릅니다. 텍스트 상자에 입력 된 모든 값을 합산 한 후 총 값을 얻을 수 있습니다.

실제 값이 아닌 값을 입력하려고하면 위의 응용 프로그램이 충돌합니다. 문제 (충돌 원인)를 찾아 해결하려면 버튼 클릭 이벤트에 중단 점을 삽입 할 수 있습니다.

아래 그림과 같이 항목 1에 "abc"를 작성해 보겠습니다.

합계 버튼을 클릭하면 프로그램이 중단 점에서 중지되는 것을 볼 수 있습니다.

이제 커서를 textbox1.Text쪽으로 이동하면 프로그램이 추가하려고하는 것을 볼 수 있습니다. abc 값을 다른 값과 함께 사용하면 프로그램이 충돌합니다.

XAML에서 디버깅

XAML에서 동일한 종류의 디버깅이 예상되는 경우 다른 절차 언어 코드를 디버깅하는 것처럼 XAML 코드를 아직 디버깅 할 수 없다는 사실에 놀랄 것입니다. XAML 코드에서 디버깅이라는 용어를 들으면 오류를 찾아보십시오.

  • 데이터 바인딩에서 데이터가 화면에 표시되지 않고 이유를 모릅니다.

  • 또는 복잡한 레이아웃과 관련된 문제입니다.

  • 또는 ListBox 및 콤보 상자와 같은 광범위한 템플릿을 사용하여 정렬 문제 또는 여백 색상, 오버레이 등의 문제.

XAML 프로그램 디버깅은 일반적으로 바인딩이 작동하는지 확인하기 위해 수행하는 작업입니다. 작동하지 않으면 무엇이 잘못되었는지 확인합니다. Silverlight를 제외하고 XAML 바인딩에서 중단 점을 설정하는 것은 불가능하지만 출력 창을 사용하여 데이터 바인딩 오류를 확인할 수 있습니다. 데이터 바인딩에서 오류를 찾기 위해 다음 XAML 코드를 살펴 보겠습니다.

<Window x:Class = "DataBindingOneWay.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <StackPanel Name = "Display"> 
         <StackPanel Orientation = "Horizontal" Margin = "50, 50, 0, 0"> 
            <TextBlock Text = "Name: " Margin = "10" Width = "100"/> 
            <TextBlock Margin = "10" Width = "100" Text = "{Binding FirstName}"/> 
         </StackPanel> 
			
         <StackPanel Orientation = "Horizontal" Margin = "50,0,50,0"> 
            <TextBlock Text = "Title: " Margin = "10" Width = "100"/> 
            <TextBlock Margin = "10" Width = "100" Text = "{Binding Title}" /> 
         </StackPanel> 
			
      </StackPanel> 
   </Grid> 
	
</Window>

두 텍스트 블록의 텍스트 속성은 정적으로 "Name"및 "Title"로 설정되고 다른 두 텍스트 블록 텍스트 속성은 "FirstName"및 "Title"에 바인딩되지만 클래스 변수는 아래 표시된 Employee 클래스의 Name 및 Title입니다.

원하는 출력이 표시되지 않을 때 이러한 유형의 오류를 찾을 수있는 위치를 이해하기 위해 의도적으로 잘못된 변수 이름을 작성했습니다.

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks;  

namespace DataBindingOneWay { 

   public class Employee { 
      public string Name { get; set; } 
      public string Title { get; set; }  
		
      public static Employee GetEmployee() { 
		
         var emp = new Employee() { 
            Name = "Ali Ahmed", Title = "Developer" 
         }; 
			
         return emp; 
      }  
   } 
}

다음은 C # 코드에서 MainWindow 클래스의 구현입니다.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace DataBindingOneWay { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         DataContext = Employee.GetEmployee(); 
      } 
   } 
}

이 응용 프로그램을 실행하면 MainWindow에서 Employee 개체의 Title에 성공적으로 바인딩되었지만 이름이 바인딩되지 않았 음을 즉시 확인할 수 있습니다.

이름에 무슨 일이 있었는지 확인하기 위해 많은 로그가 생성되는 출력 창을 살펴 보겠습니다.

오류를 쉽게 찾을 수있는 방법은 오류를 검색하는 것뿐입니다. "BindingExpression 경로 오류 : 'Object' ''Employe에서 'FirstName'속성을 찾을 수 없습니다. '라는 다음 오류가 표시됩니다.

System.Windows.Data Error: 40 : BindingExpression path error: 'FirstName'
   property not found on 'object' ''Employee' (HashCode=11611730)'.
   BindingExpression:Path = FirstName; DataItem = 'Employee' (HashCode = 11611730);
   target element is 'TextBlock' (Name=''); target property is 'Text' (type 'String')

이는 FirstName이 Employee 클래스의 구성원이 아님을 명확하게 나타내므로 응용 프로그램에서 이러한 유형의 문제를 해결하는 데 도움이됩니다.

FirstName을 다시 Name으로 변경하면 원하는 출력이 표시됩니다.

XAML 용 UI 디버깅 도구

런타임에 XAML 코드를 검사하기 위해 Visual Studio 2015가 포함 된 XAML 용 UI 디버깅 도구가 도입되었습니다. 이러한 도구의 도움으로 XAML 코드는 실행중인 WPF 응용 프로그램의 시각적 트리 형태와 트리의 다양한 UI 요소 속성으로 표시됩니다. 이러한 도구를 활성화하려면 아래 단계를 따르십시오.

  • 도구 메뉴로 이동하여 도구 메뉴에서 옵션을 선택합니다.
  • 다음 대화 상자가 열립니다.
  • 왼쪽의 디버깅 항목에서 일반 옵션으로 이동하십시오.
  • 강조 표시된 옵션, 즉 "XAML 용 UI 디버깅 도구 사용"을 선택하고 확인 버튼을 클릭합니다.

이제 XAML 애플리케이션을 실행하거나 다음 XAML 코드를 사용합니다.

<Window x:Class = "XAMLTestBinding.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <ComboBox Name = "comboBox"  Margin = "50" Width = "100"> 
         <ComboBoxItem Content = "Green" /> 
         <ComboBoxItem  Content = "Yellow" IsSelected = "True" />
         <ComboBoxItem Content = "Orange" /> 
      </ComboBox> 
		
      <TextBox  Name = "textBox" Margin = "50" Width = "100" Height = "23"
         VerticalAlignment = "Top" Text  =
         "{Binding ElementName = comboBox, Path = SelectedItem.Content, Mode = TwoWay, UpdateSourceTrigger = PropertyChanged}" 
         Background = "{Binding ElementName = comboBox, Path = SelectedItem.Content}"> 
      </TextBox> 
		
   </StackPanel> 
	
</Window>

응용 프로그램을 실행하면 모든 요소가 트리에 표시되는 라이브 비주얼 트리가 표시됩니다.

이 라이브 비주얼 트리는 UI 요소의 위치를 ​​이해하기위한 전체 레이아웃 구조를 보여줍니다. 그러나이 옵션은 Visual Studio 2015에서만 사용할 수 있습니다. Visual Studio의 이전 옵션을 사용하는 경우이 도구를 사용할 수 없지만 Visual Studio 용 XAML Spy와 같이 Visual Studio와 통합 할 수있는 다른 도구가 있습니다. . xamlspy 에서 다운로드 할 수 있습니다.

WPF 응용 프로그램을 사용하면 사용자 지정 컨트롤을 만들 수 있으므로 기능이 풍부하고 사용자 지정 가능한 컨트롤을 매우 쉽게 만들 수 있습니다. 사용자 지정 컨트롤은 Microsoft에서 제공하는 모든 기본 제공 컨트롤이 기준을 충족하지 않거나 타사 컨트롤에 대해 비용을 지불하지 않으려는 경우에 사용됩니다.

이 장에서는 사용자 지정 컨트롤을 만드는 방법을 배웁니다. 사용자 지정 컨트롤을 살펴보기 전에 먼저 사용자 컨트롤에 대해 간략히 살펴 보겠습니다.

사용자 제어

사용자 컨트롤은 서로 다른 기본 제공 컨트롤을 수집 및 결합하고 재사용 가능한 XAML로 패키징하는 방법을 제공합니다. 사용자 컨트롤은 다음 시나리오에서 사용됩니다.

  • 컨트롤이 기존 컨트롤로 구성된 경우, 즉 이미 존재하는 여러 컨트롤의 단일 컨트롤을 만들 수 있습니다.

  • 컨트롤에 테마 지원이 필요하지 않은 경우. 사용자 컨트롤은 복잡한 사용자 지정, 컨트롤 템플릿 및 스타일을 지원하지 않습니다.

  • 개발자가보기가있는 코드 숨김 모델을 사용하여 컨트롤을 작성하고 이벤트 처리기에 대한 직접 코드를 작성하려는 경우.

  • 응용 프로그램간에 제어권을 공유하지 않습니다.

사용자 제어의 예로 이동하여 아래 단계를 따르십시오.

  • 새 WPF 프로젝트를 만든 다음 솔루션을 마우스 오른쪽 단추로 클릭하고 추가> 새 항목 ...을 선택합니다.

  • 다음 창이 열립니다. 이제 선택User Control (WPF) 이름을 MyUserControl로 지정합니다.

  • 추가 버튼을 클릭하면 두 개의 새 파일 (MyUserControl.xaml 및 MyUserControl.cs)이 솔루션에 추가되는 것을 볼 수 있습니다.

다음은 MyUserControl.xaml 파일의 일부 속성을 사용하여 단추와 텍스트 상자를 만드는 XAML 코드입니다.

<UserControl x:Class = "WPFUserControl.MyUserControl" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"  
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"  
   mc:Ignorable = "d" d:DesignHeight = "300" d:DesignWidth = "300"> 
	
   <Grid> 
      <TextBox Height = "23"  
         HorizontalAlignment = "Left"  
         Margin = "80,49,0,0" Name = "txtBox"  
         VerticalAlignment = "Top" Width = "200" /> 
			
      <Button Content = "Click Me"  
         Height = "23" HorizontalAlignment = "Left"  
         Margin = "96,88,0,0" Name = "button"  
         VerticalAlignment = "Top" Click = "button_Click" />    
   </Grid>
	
</UserControl>

다음은 텍스트 상자를 업데이트하는 MyUserControl.cs 파일의 버튼 클릭 이벤트에 대한 C # 코드입니다.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace WPFUserControl {
   /// <summary>
      /// Interaction logic for MyUserControl.xaml 
   /// </summary> 
	
   public partial class MyUserControl : UserControl { 
	
      public MyUserControl() { 
         InitializeComponent(); 
      }  
		
      private void button_Click(object sender, RoutedEventArgs e) { 
         txtBox.Text = "You have just clicked the button"; 
      } 
   } 
}

다음은 사용자 컨트롤을 추가하기위한 MainWindow.xaml의 구현입니다.

<Window x:Class = "XAMLUserControl.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:control = "clr-namespace:WPFUserControl" 
   Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid> 
      <control:MyUserControl/> 
   </Grid> 
	
</Window>

위 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다.

"Click Me"버튼을 클릭하면 텍스트 상자 안의 텍스트가 업데이트되는 것을 알 수 있습니다.

사용자 지정 컨트롤

사용자 지정 컨트롤은 일반적으로 generic.xaml에 정의 된 자체 스타일과 템플릿을 제공하는 클래스입니다. 사용자 지정 컨트롤은 다음 시나리오에서 사용됩니다.

  • 컨트롤이 존재하지 않고 처음부터 만들어야하는 경우.

  • 특정 시나리오에 맞게 추가 속성 또는 추가 기능을 추가하여 기존 컨트롤에 기능을 확장하거나 추가하려는 경우.

  • 컨트롤이 테마 및 스타일을 지원해야하는 경우.

  • 응용 프로그램간에 제어 권한을 공유하려는 경우.

사용자 지정 컨트롤의 작동 방식을 이해하기 위해 예제를 살펴 보겠습니다. 새 WPF 프로젝트를 만든 다음 솔루션을 마우스 오른쪽 단추로 클릭하고 추가> 새 항목 ...을 선택합니다.

다음 창이 열립니다. 이제 선택Custom Control (WPF) 이름을 MyCustomControl.

추가 버튼을 클릭하면 두 개의 새 파일 (Themes / Generic.xaml 및 MyCustomControl.cs)이 솔루션에 추가되는 것을 볼 수 있습니다.

다음은 Generic.xaml 파일의 사용자 지정 컨트롤에 대해 스타일이 설정된 XAML 코드입니다.

<ResourceDictionary 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFCustomControls">  
	
   <Style TargetType = "{x:Type local:MyCustomControl}"
      BasedOn = "{StaticResource {x:Type Button}}"> 
      <Setter Property = "Background" Value = "LightSalmon" /> 
      <Setter Property = "Foreground" Value = "Blue"/> 
   </Style> 
	
</ResourceDictionary>

다음은 단추 클래스에서 상속되고 생성자에서 메타 데이터를 재정의하는 MyCustomControl 클래스에 대한 C # 코드입니다.

using System; 
using System.Windows; 
using System.Windows.Controls; 
 
namespace WPFCustomControls { 

   public class MyCustomControl : Button { 
	
      static MyCustomControl() { 
         DefaultStyleKeyProperty.OverrideMetadata(typeof(MyCustomControl), new
            FrameworkPropertyMetadata(typeof(MyCustomControl))); 
      } 
		
   } 
}

다음은 텍스트 블록의 텍스트를 업데이트하는 C #의 사용자 지정 컨트롤 클릭 이벤트 구현입니다.

using System; 
using System.Windows; 
using System.Windows.Controls;

namespace WPFCustomControls { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void customControl_Click(object sender, RoutedEventArgs e) { 
         txtBlock.Text = "You have just click your custom control"; 
      }
		
   } 
}

다음은 사용자 지정 컨트롤과 TextBlock을 추가하는 MainWindow.xaml의 구현입니다.

<Window x:Class = "WPFCustomControls.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:control = "clr-namespace:WPFCustomControls" 
   Title = "MainWindow" Height = "350" Width = "604"> 
	
   <StackPanel> 
      <control:MyCustomControl x:Name = "customControl"  
         Content = "Click Me" Width = "70" 
         Margin = "10" Click = "customControl_Click"/> 
			
      <TextBlock Name = "txtBlock"  
         Width = "250" Height = "30"/> 
   </StackPanel>
	
</Window>

위 코드를 컴파일하고 실행하면 사용자 정의 버튼 인 사용자 정의 컨트롤이있는 다음과 같은 창이 생성됩니다.

사용자 정의 버튼을 클릭하면 텍스트 블록 내부의 텍스트가 업데이트되는 것을 볼 수 있습니다.

예외는 프로그램 실행 중에 발생하는 오류 조건 또는 예기치 않은 동작입니다. 여러 가지 이유로 예외가 발생할 수 있으며, 그중 일부는 다음과 같습니다.

  • 코드 또는 호출하는 코드 (예 : 공유 라이브러리)의 오류,

  • 사용할 수없는 운영 체제 리소스,

  • 공용 언어 런타임에서 발생하는 예기치 않은 조건 (예 : 확인할 수없는 코드)

통사론

예외에는 프로그램의 흐름을 한 부분에서 다른 부분으로 전송할 수있는 기능이 있습니다. .NET 프레임 워크에서 예외 처리에는 다음 네 가지 키워드가 있습니다.

  • try −이 블록에서 프로그램은 일부 예외를 발생시키는 특정 조건을 식별합니다.

  • catch− catch 키워드는 예외 포착을 나타냅니다. ㅏtry 블록 뒤에 하나 이상의 catch 문제를 처리하려는 프로그램의 위치에서 예외 처리기로 예외를 포착하는 블록.

  • finally− finally 블록은 예외 발생 여부에 관계없이 주어진 명령문 집합을 실행하는 데 사용됩니다. 예를 들어 파일을 열면 예외 발생 여부에 관계없이 파일을 닫아야합니다.

  • throw− 문제가 발생하면 프로그램에서 예외가 발생합니다. 이것은 throw 키워드를 사용하여 수행됩니다.

이 네 가지 키워드를 사용하는 구문은 다음과 같습니다.

try { 
   ///This will still trigger the exception 
} 
catch (ExceptionClassName e) { 
   // error handling code 
} 
catch (ExceptionClassName e) { 
   // error handling code
}
catch (ExceptionClassName e) { 
   // error handling code 
} 
finally { 
   // statements to be executed 
}

try 블록이 프로그램 흐름의 상황에 따라 둘 이상의 예외를 발생시킬 수있는 경우 여러 catch 문이 사용됩니다.

계층

.NET 프레임 워크의 거의 모든 예외 클래스는 Exception 클래스에서 직접 또는 간접적으로 파생됩니다. Exception 클래스에서 파생 된 가장 중요한 예외 클래스는 다음과 같습니다.

  • ApplicationException class− 프로그램에 의해 생성되는 예외를 지원합니다. 개발자가 예외를 정의하려면이 클래스에서 클래스를 파생시켜야합니다.

  • SystemException class− 미리 정의 된 모든 런타임 시스템 예외에 대한 기본 클래스입니다. 다음 계층은 런타임에서 제공하는 표준 예외를 보여줍니다.

다음 표에는 런타임에서 제공하는 표준 예외와 파생 클래스를 만들어야하는 조건이 나와 있습니다.

예외 유형 기본 유형 기술
Exception 목적 모든 예외에 대한 기본 클래스입니다.
SystemException 예외 모든 런타임 생성 오류의 기본 클래스입니다.
IndexOutOfRangeException SystemException 배열이 부적절하게 인덱싱 된 경우에만 런타임에서 throw됩니다.
NullReferenceException SystemException null 개체가 참조되는 경우에만 런타임에서 throw됩니다.
AccessViolationException SystemException 유효하지 않은 메모리에 액세스 할 때만 런타임에서 발생합니다.
InvalidOperationException SystemException 유효하지 않은 상태 일 때 메소드에 의해 발생합니다.
ArgumentException SystemException 모든 인수 예외에 대한 기본 클래스입니다.
ArgumentNullException ArgumentException 인수가 널이되는 것을 허용하지 않는 메소드에 의해 발생합니다.
ArgumentOutOfRangeException ArgumentException 인수가 주어진 범위에 있는지 확인하는 메서드에 의해 throw됩니다.
ExternalException SystemException 발생하거나 런타임 외부의 환경을 대상으로하는 예외에 대한 기본 클래스입니다.
SEHException ExternalException Win32 구조적 예외 처리 정보를 캡슐화하는 예외입니다.

개념을 더 잘 이해하기 위해 간단한 예를 들어 보겠습니다. 이름으로 새 WPF 프로젝트를 만들어 시작합니다.WPFExceptionHandling.

도구 상자에서 디자인 창으로 텍스트 상자 하나를 끕니다. 다음 XAML 코드는 텍스트 상자를 만들고 일부 속성으로 초기화합니다.

<Window x:Class = "WPFExceptionHandling.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFExceptionHandling"
   mc:Ignorable = "d" 
   Title = "MainWindow" Height = "350" Width = "604">
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"
         Height = "241" Margin = "70,39,0,0" TextWrapping = "Wrap" 
         VerticalAlignment = "Top" Width = "453"/> 
   </Grid> 
	
</Window>

다음은 C #에서 예외 처리로 읽는 파일입니다.

using System; 
using System.IO; 
using System.Windows;

namespace WPFExceptionHandling { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         ReadFile(0); 
      }
		
      void ReadFile(int index) { 
         string path = @"D:\Test.txt"; 
         StreamReader file = new StreamReader(path); 
         char[] buffer = new char[80]; 
			
         try { 
            file.ReadBlock(buffer, index, buffer.Length); 
            string str = new string(buffer); 
            str.Trim(); 
            textBox.Text = str; 
         }
         catch (Exception e) {
            MessageBox.Show("Error reading from "+ path + "\nMessage = "+ e.Message);
         } 
         finally { 
            if (file != null) { 
               file.Close(); 
            } 
         } 
      } 
   } 
}

위의 코드를 컴파일하고 실행하면 텍스트 상자 안에 텍스트가 표시되는 다음과 같은 창이 생성됩니다.

예외가 발생하거나 수동으로 던지면 (다음 코드에서와 같이) 오류가있는 메시지 상자가 표시됩니다.

using System; 
using System.IO; 
using System.Windows;

namespace WPFExceptionHandling {
 
   public partial class MainWindow : Window {
	
      public MainWindow() { 
         InitializeComponent(); 
         ReadFile(0); 
      } 
		
      void ReadFile(int index) { 
         string path = @"D:\Test.txt"; 
         StreamReader file = new StreamReader(path); 
         char[] buffer = new char[80]; 
			
         try { 
            file.ReadBlock(buffer, index, buffer.Length); 
            string str = new string(buffer); 
            throw new Exception(); 
            str.Trim(); 
            textBox.Text = str; 
         }
         catch (Exception e) { 
            MessageBox.Show("Error reading from "+ path + "\nMessage = "+ e.Message); 
         } 
         finally { 
            if (file != null) { 
               file.Close(); 
            } 
         } 
      } 
   } 
}

위 코드를 실행하는 동안 예외가 발생하면 다음과 같은 메시지가 표시됩니다.

위의 코드를 실행하고 그 기능을 실험 해 보는 것이 좋습니다.

지역화는 응용 프로그램이 지원하는 특정 문화권에 맞게 응용 프로그램 리소스를 지역화 된 버전으로 변환하는 것입니다.

애플리케이션을 개발하고 애플리케이션이 하나의 언어로만 제공되면 고객 수와 비즈니스 규모가 제한됩니다. 고객 기반을 늘리고 비즈니스도 증가 시키려면 제품을 사용할 수 있고 전 세계 고객에게 도달 할 수 있어야합니다. 비용 효율적localization 더 많은 고객에게 다가 갈 수있는 가장 경제적이고 가장 경제적 인 방법 중 하나입니다.

WPF에서 지역화 가능한 응용 프로그램은 다음을 사용하여 매우 쉽게 만들 수 있습니다. resx가장 간단한 현지화 솔루션입니다. 작동 원리를 이해하기 위해 간단한 예를 들어 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPFLocalization.

  • 솔루션 탐색기에서 Properties 폴더 아래 Resources.resx 파일을 볼 수 있습니다.

  • XAML 파일에서 액세스 할 수 있도록 액세스 한정자를 내부에서 공용으로 변경합니다.

  • 이제 애플리케이션에서 사용할 다음 문자열의 이름과 값을 추가합니다.

  • Resources.en.resx 및 Resources.ru-RU.resx라는 이름의 Resources.resx 파일 사본 두 개를 만듭니다. 이는 언어 및 국가 / 지역 이름과 관련된 명명 규칙이며 NLS (National Language Support) API 참조 (https://msdn.microsoft.com/en-us/goglobal/bb896001.aspx ) 페이지.

  • Resources.ru-RU.resx의 값을 아래와 같이 러시아어 단어로 변경합니다.

  • 디자인 창으로 이동하여 텍스트 상자 3 개, 레이블 3 개, 버튼 3 개를 드래그 해 보겠습니다.

  • XAML 파일에서 먼저 네임 스페이스 선언을 추가하여 리소스 지역화 xmlns : p = "clr-namespace : WPFLocalization.Properties"를 사용합니다.

  • 아래와 같이 모든 컨트롤의 속성을 설정합니다. 이 예제에서는 XAML 파일에있는 창 제목, 단추 및 레이블의 내용에 대해 하드 코딩 된 문자열을 사용하지 않습니다. * .resx 파일에 정의 된 문자열을 사용합니다. 예를 들어, 창의 제목에는 “Title = "{x : Static p : Resources.Title}" " 과 같이 * .resx 파일에 정의 된 제목 문자열을 사용합니다 .

  • 다음은 다른 속성을 사용하여 컨트롤을 만들고 초기화하는 XAML 파일입니다.

<Window x:Class = "WPFLocalization.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:local = "clr-namespace:WPFLocalization" 
   xmlns:p = "clr-namespace:WPFLocalization.Properties"
   Title = "{x:Static p:Resources.Title}" Height = "350" Width = "604">
	
   <Grid> 
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,45,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "304"/>
			
      <Label x:Name = "label" Content = "{x:Static p:Resources.Name}"
          HorizontalAlignment = "Left" Margin = "52,45,0,0" VerticalAlignment = "Top" Width = "86"/>
			 
      <TextBox x:Name = "textBox1" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,102,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "304"/> 
			
      <Label x:Name = "label1" Content = "{x:Static p:Resources.Address}" 
         HorizontalAlignment = "Left" Margin = "52,102,0,0" VerticalAlignment = "Top" Width = "86"/>
			
      <TextBox x:Name = "textBox2" HorizontalAlignment = "Left" Height = "23" 
         Margin = "128,157,0,0" TextWrapping = "Wrap" VerticalAlignment = "Top" Width = "80"/>
			
      <Label x:Name = "label2" Content = "{x:Static p:Resources.Age}" 
         HorizontalAlignment = "Left" Margin = "52,157,0,0" VerticalAlignment = "Top" Width = "86"/>
			
      <Button x:Name = "button" Content = "{x:Static p:Resources.OK_Button}" 
         HorizontalAlignment = "Left" Margin = "163,241,0,0" VerticalAlignment = "Top" Width = "75"/> 
			
      <Button x:Name = "button1" Content = "{x:Static p:Resources.Cancel_Button}" 
         HorizontalAlignment = "Left" Margin = "282,241,0,0" VerticalAlignment = "Top" Width = "75"/>
			
      <Button x:Name = "button2" Content = "{x:Static p:Resources.Help_Button}" 
         HorizontalAlignment = "Left" Margin = "392,241,0,0" VerticalAlignment = "Top" Width = "75"/> 
   </Grid> 
	
 </Window>
  • 위의 코드가 컴파일되고 실행되면 다른 컨트롤이 포함 된 다음 창이 표시됩니다.

  • 기본적으로 프로그램은 기본 Resources.resx를 사용합니다. Resources.ru-RU.resx 파일에 정의 된 러시아어로 텍스트를 표시하려면 아래와 같이 App.xaml 파일에서 프로그램을 시작할 때 문화권을 명시 적으로 설정해야합니다.

using System.Windows;

namespace WPFLocalization {
   /// <summary> 
      /// Interaction logic for App.xaml 
   /// </summary> 
	
   public partial class App : Application {
	
      App() { 
         System.Threading.Thread.CurrentThread.CurrentUICulture = new System.Globalization.CultureInfo("ru-RU");
         //System.Threading.Thread.CurrentThread.CurrentUICulture = new System.Globalization.CultureInfo("en"); 
      } 
   } 
}

응용 프로그램을 실행하면 모든 텍스트가 러시아어로 표시됩니다.

위의 코드를 실행하고 다른 문화권에 대해서도 resx 파일을 만드는 것이 좋습니다.

WPF에서 상호 작용은 뷰가 해당 뷰에있는 컨트롤과 상호 작용하는 방식을 보여줍니다. 가장 일반적으로 알려진 상호 작용은 두 가지 유형입니다.

  • Behaviors
  • 끌어서 놓기

행동

일부 기능을 재사용 가능한 구성 요소로 캡슐화 할 수있는 Expression Blend 3에서 동작이 도입되었습니다. 추가 동작을 추가하려면 이러한 구성 요소를 컨트롤에 연결할 수 있습니다. 동작은 복잡한 사용자 상호 작용을 쉽게 설계 할 수있는 유연성을 제공합니다.

ControlStoryBoardAction 비헤이비어가 컨트롤에 연결된 간단한 예제를 살펴 보겠습니다.

  • 이름이 WPFBehavior 인 새 WPF 프로젝트를 만듭니다.

  • 다음 XAML 코드는 타원의 이동을 제어하기 위해 타원과 두 개의 단추를 만듭니다.

<Window 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFBehaviors" 
   xmlns:i = "http://schemas.microsoft.com/expression/2010/interactivity" 
   xmlns:ei = "http://schemas.microsoft.com/expression/2010/interactions" 
   x:Class = "WPFBehaviors.MainWindow" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604">
	
   <Window.Resources> 
      <Storyboard x:Key = "Storyboard1" RepeatBehavior = "Forever" AutoReverse = "True"> 
		
         <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty =
            "(UIElement.RenderTransform).(TransformGroup.Children )[3].(TranslateTransform.X)"
            Storyboard.TargetName = "ellipse"> 
            <EasingDoubleKeyFrame KeyTime = "0:0:1" Value = "301.524"/> 
            <EasingDoubleKeyFrame KeyTime = "0:0:2" Value = "2.909"/> 
         </DoubleAnimationUsingKeyFrames>
			
         <DoubleAnimationUsingKeyFrames Storyboard.TargetProperty = 
            "(UIElement.RenderTransform).(TransformGroup.Children )[3].(TranslateTransform.Y)"
            Storyboard.TargetName = "ellipse"> 
            <EasingDoubleKeyFrame KeyTime = "0:0:1" Value = "-0.485"/> 
            <EasingDoubleKeyFrame KeyTime = "0:0:2" Value = "0"/> 
         </DoubleAnimationUsingKeyFrames> 
			
         <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty = "(ContentControl.Content)"
            Storyboard.TargetName = "button"> 
            <DiscreteObjectKeyFrame KeyTime = "0" Value = "Play"/> 
         </ObjectAnimationUsingKeyFrames>
			
         <ObjectAnimationUsingKeyFrames Storyboard.TargetProperty = "(ContentControl.Content)"
            Storyboard.TargetName = "button1"> 
            <DiscreteObjectKeyFrame KeyTime = "0" Value = "Stop"/> 
            <DiscreteObjectKeyFrame KeyTime = "0:0:2" Value = "Stop"/> 
         </ObjectAnimationUsingKeyFrames> 
      </Storyboard> 
   </Window.Resources> 
	
   <Window.Triggers> 
      <EventTrigger RoutedEvent = "FrameworkElement.Loaded"> 
         <BeginStoryboard Storyboard = "{StaticResource Storyboard1}"/> 
      </EventTrigger> 
   </Window.Triggers> 
	
   <Grid> 
      <Ellipse x:Name = "ellipse" Fill = "#FFAAAAC5" HorizontalAlignment = "Left"
         Height = "50.901" Margin = "49.324,70.922,0,0" Stroke = "Black"
         VerticalAlignment = "Top" Width = "73.684" RenderTransformOrigin = "0.5,0.5"> 
         <Ellipse.RenderTransform> 
            <TransformGroup> 
               <ScaleTransform/> 
               <SkewTransform/> 
               <RotateTransform/> 
               <TranslateTransform/> 
            </TransformGroup> 
         </Ellipse.RenderTransform> 
      </Ellipse>
		
      <Button x:Name = "button" Content = "Play" HorizontalAlignment = "Left" Height = "24.238"
         Margin = "63.867,0,0,92.953" VerticalAlignment = "Bottom" Width = "74.654"> 
         <i:Interaction.Triggers> 
            <i:EventTrigger EventName = "Click"> 
               <ei:ControlStoryboardAction Storyboard = "{StaticResource Storyboard1}"/> 
            </i:EventTrigger> 
         </i:Interaction.Triggers> 
      </Button>
		
      <Button x:Name = "button1" Content = "Stop" HorizontalAlignment = "Left" Height = "24.239"
         Margin = "160.82,0,0,93.922" VerticalAlignment = "Bottom" Width = "75.138"> 
         <i:Interaction.Triggers> 
            <i:EventTrigger EventName = "Click"> 
               <ei:ControlStoryboardAction ControlStoryboardOption = "Stop"
                  Storyboard = "{StaticResource Storyboard1}"/> 
            </i:EventTrigger> 
         </i:Interaction.Triggers> 
      </Button> 
		
   </Grid> 
</Window>

위의 코드를 컴파일하고 실행하면 타원과 두 개의 버튼이 포함 된 다음과 같은 창이 생성됩니다.

재생 버튼을 누르면 왼쪽에서 오른쪽으로 움직이기 시작하고 원래 위치로 돌아갑니다. 중지 버튼은 타원의 이동을 중지합니다.

끌어서 놓기

사용자 인터페이스에서 드래그 앤 드롭하면 애플리케이션의 효율성과 생산성을 크게 향상시킬 수 있습니다. 사람들이 구현하기 어렵다고 생각하기 때문에 끌어서 놓기 기능을 사용하는 응용 프로그램은 거의 없습니다. 드래그 앤 드롭 기능은 어느 정도 다루기가 어렵지만 WPF에서는 매우 쉽게 처리 할 수 ​​있습니다.

작동 원리를 이해하기 위해 간단한 예를 들어 보겠습니다. 한 사각형에서 다른 사각형으로 색상을 끌어다 놓을 수있는 응용 프로그램을 만들 것입니다.

  • 이름이 WPFDragAndDrop 인 새 WPF 프로젝트를 만듭니다.

  • 5 개의 사각형을 디자인 창으로 드래그하고 다음 XAML 파일과 같이 속성을 설정합니다.

<Window x:Class = "WPFDragAndDrop.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFDragAndDrop" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "402.551" Width = "604"> 
	
   <Grid> 
      <Rectangle Name = "Target" Fill = "AliceBlue" HorizontalAlignment = "Left"  
         Height = "345" Margin = "10,10,0,0" Stroke = "Black"  
         VerticalAlignment = "Top" Width = "387" AllowDrop = "True" Drop = "Target_Drop"/> 
			
      <Rectangle Fill = "Beige" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,10,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightBlue" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,80,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightCoral" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,150,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "LightGray" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,220,0,0" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/> 
			
      <Rectangle Fill = "OliveDrab" HorizontalAlignment = "Left" Height = "65"  
         Margin = "402,290,0,-7" Stroke = "Black" VerticalAlignment = "Top"  
         Width = "184" MouseLeftButtonDown = "Rect_MLButtonDown"/>  
   </Grid> 
	
</Window>
  • 첫 번째 사각형은 대상 사각형이므로 사용자는 다른 사각형의 색상을 대상 사각형으로 드래그 할 수 있습니다.

  • 다음은 드래그 앤 드롭을위한 C #의 이벤트 구현입니다.

using System.Windows; 
using System.Windows.Input; 
using System.Windows.Media; 
using System.Windows.Shapes; 
 
namespace WPFDragAndDrop { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      }  
		
      private void Rect_MLButtonDown(object sender, MouseButtonEventArgs e) { 
         Rectangle rc = sender as Rectangle; 
         DataObject data = new DataObject(rc.Fill); 
         DragDrop.DoDragDrop(rc, data,DragDropEffects.Move); 
      }  
		
      private void Target_Drop(object sender, DragEventArgs e) { 
         SolidColorBrush scb = (SolidColorBrush)e.Data.GetData(typeof(SolidColorBrush)); 
         Target.Fill = scb; 
      } 
   } 
}

응용 프로그램을 실행하면 다음 창이 생성됩니다.

오른쪽에있는 사각형에서 색상을 드래그하여 왼쪽에있는 큰 사각형에 놓으면 즉시 효과를 볼 수 있습니다.

오른쪽에서 네 번째 항목을 드래그하겠습니다 .

대상 사각형의 색상이 변경되었음을 알 수 있습니다. 위의 코드를 실행하고 그 기능을 실험 해 보는 것이 좋습니다.

WPF는 응용 프로그램 요구 사항에 따라 향상 될 수있는 광범위한 2D 그래픽을 제공합니다. WPF는 그래픽 콘텐츠를 그리는 데 사용되는 Drawing 및 Shape 개체를 모두 지원합니다.

모양과 그림

  • Shape 클래스는 FrameworkElement 클래스에서 파생되며 Shape 개체는 패널 및 대부분의 컨트롤 내에서 사용할 수 있습니다.

  • WPF는 Ellipse, Line, Path, Polygon, Polyline 및 Rectangle과 같은 Shape 클래스에서 파생 된 몇 가지 기본 모양 개체를 제공합니다.

  • 반면 그리기 개체는 FrameworkElement 클래스에서 파생되지 않으며보다 가벼운 구현을 제공합니다.

  • 그리기 개체는 Shape 개체에 비해 더 간단합니다. 그들은 또한 더 나은 성능 특성을 가지고 있습니다.

다양한 모양 객체를 사용하는 방법을 이해하기 위해 간단한 예를 들어 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPF2DGraphics.

  • 다음 코드는 다양한 유형의 모양을 만듭니다.

<Window x:Class = "WPF2DGraphics.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF2DGraphics" 
   xmlns:PresentationOptions = "http://schemas.microsoft.com/winfx/2006/xaml/present ation/options"  
   mc:Ignorable = "PresentationOptions" Title = "MainWindow" Height = "400" Width = "604"> 
	
   <StackPanel> 
      <Ellipse Width = "100" Height = "60" Name = "sample" Margin = "10"> 
         <Ellipse.Fill> 
            <RadialGradientBrush> 
               <GradientStop Offset = "0" Color = "AliceBlue"/> 
               <GradientStop Offset = "1" Color = "Gray"/> 
               <GradientStop Offset = "2" Color = "Red"/> 
            </RadialGradientBrush> 
         </Ellipse.Fill> 
      </Ellipse>
		
      <Path Stroke = "Red" StrokeThickness = "5" Data = "M 10,70 L 200,70"
         Height = "42.085" Stretch = "Fill" Margin = "140.598,0,146.581,0" /> 
      <Path Stroke = "BlueViolet" StrokeThickness = "5" Data = "M 20,100 A 100,56 42 1 0 200,10"
         Height = "81.316" Stretch = "Fill" Margin = "236.325,0,211.396,0" /> 
			
      <Path Fill = "LightCoral" Margin = "201.424,0,236.325,0"
         Stretch = "Fill" Height = "124.929"> 
         <Path.Data> 
            <PathGeometry> 
               <PathFigure StartPoint = "50,0" IsClosed = "True"> 
                  <LineSegment Point = "100,50"/> 
                  <LineSegment Point = "50,100"/> 
                  <LineSegment Point = "0,50"/> 
               </PathFigure> 
            </PathGeometry> 
         </Path.Data> 
      </Path> 
		
   </StackPanel> 
	
</Window>

위 코드를 컴파일하고 실행하면 타원, 직선, 호 및 다각형이 생성됩니다.

그림으로 영역을 그리는 방법을 보여주는 또 다른 예를 살펴 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPF2DGraphics1.

  • 다음 XAML 코드는 이미지 그리기로 다르게 그리는 방법을 보여줍니다.

<Window x:Class = "WPF2DGraphics1.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:PresentationOptions = "http://schemas.microsoft.com/winfx/2006/xaml/present ation/options"  
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   mc:Ignorable = "PresentationOptions" 
   xmlns:local = "clr-namespace:WPF2DGraphics1" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Border BorderBrush = "Gray" BorderThickness = "1"  
         HorizontalAlignment = "Left" VerticalAlignment = "Top" 
         Margin = "20"> 
			
         <Image Stretch = "None">
            <Image.Source> 
               <DrawingImage PresentationOptions:Freeze = "True"> 
					
                  <DrawingImage.Drawing> 
                     <DrawingGroup> 
                        <ImageDrawing Rect = "300,100,300,180" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "0,100,250,100" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "150,0,25,25" ImageSource = "Images\DSC_0104.JPG"/> 
                        <ImageDrawing Rect = "0,0,75,75" ImageSource = "Images\DSC_0104.JPG"/> 
                     </DrawingGroup> 
                  </DrawingImage.Drawing> 
						
               </DrawingImage> 
            </Image.Source> 
         </Image> 
			
      </Border> 
   </Grid> 
	
</Window>

응용 프로그램을 실행하면 다음과 같은 출력이 생성됩니다.

위의 코드를 실행하고 더 많은 2D 모양과 그림을 시도하는 것이 좋습니다.

WPF (Windows Presentation Foundation)는 응용 프로그램 요구 사항에 따라 3D 그래픽을 그리고, 변형하고, 애니메이션하는 기능을 제공합니다. 본격적인 3D 게임 개발을 지원하지는 않지만 어느 정도 3D 그래픽을 만들 수 있습니다.

2D 및 3D 그래픽을 결합하여 풍부한 컨트롤을 만들거나 복잡한 데이터 그림을 제공하거나 응용 프로그램 인터페이스의 사용자 경험을 향상시킬 수도 있습니다. Viewport3D 요소는 3D 모델을 WPF 애플리케이션에 호스팅합니다.

3D 그래픽을 사용하는 방법을 이해하기 위해 간단한 예를 들어 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPF3DGraphics.

  • 다음 XAML 코드는 3D 지오메트리를 사용하여 2D 개체를 만드는 방법을 보여줍니다.

<Window x:Class = "WPF3DGraphics.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF3DGraphics" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "500" Width = "604"> 
	
   <Grid> 
      <Viewport3D> 
		
         <Viewport3D.Camera> 
            <PerspectiveCamera Position = "2,0,10" LookDirection = "0.2,0.4,-1"
               FieldOfView = "65" UpDirection = "0,1,0" /> 
         </Viewport3D.Camera> 
			
         <ModelVisual3D> 
            <ModelVisual3D.Content> 
               <Model3DGroup> 
                  <AmbientLight Color = "Bisque" /> 
						
                  <GeometryModel3D> 
                     <GeometryModel3D.Geometry> 
                        <MeshGeometry3D Positions = "0,0,0 0,8,0 10,0,0 8,8,0"
                           Normals = "0,0,1 0,0,1 0,0,1 0,0,1" TriangleIndices = "0,2,1 1,2,3"/> 
                     </GeometryModel3D.Geometry>
							
                     <GeometryModel3D.Material> 
                        <DiffuseMaterial Brush = "Bisque" /> 
                     </GeometryModel3D.Material> 
                  </GeometryModel3D>
						
               </Model3DGroup> 
            </ModelVisual3D.Content> 
         </ModelVisual3D> 
			
      </Viewport3D> 
   </Grid> 
	
</Window>

위 코드를 컴파일하고 실행하면 3D로 2D 객체가 생성됩니다.

3D 객체를 보여주는 또 다른 예를 살펴 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPF3DGraphics1

  • 다음 XAML 코드는 3D 개체와 슬라이더를 만듭니다. 슬라이더를 사용하여이 3D 개체를 회전 할 수 있습니다.

<Window x:Class = "WPF3DGraphics1.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPF3DGraphics1" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "525"> 
	
   <Grid>
      <Viewport3D Name="viewport3D1">
		
         <Viewport3D.Camera> 
            <PerspectiveCamera x:Name = "camMain" Position = "6 5 4" LookDirection = "-6 -5 -4"> 
            </PerspectiveCamera> 
         </Viewport3D.Camera>
		
         <ModelVisual3D> 
            <ModelVisual3D.Content> 
               <DirectionalLight x:Name = "dirLightMain" Direction = "-1,-1,-1"> 
               </DirectionalLight> 
            </ModelVisual3D.Content> 
         </ModelVisual3D>
		
         <ModelVisual3D x:Name = "MyModel"> 
            <ModelVisual3D.Content> 
               <GeometryModel3D>
				
                  <GeometryModel3D.Geometry>
                     <MeshGeometry3D x:Name = "meshMain"
                        Positions = "0 0 0  1 0 0  0 1 0  1 1 0  0 0 1  1 0 1  0 1 1  0 1 1"  
                        TriangleIndices = "2 3 1  3 1 0  7 1 3  7 5 1  6 5 7  6 4 5  6 2 0
                        2 0 4  2 7 3  2 6 7  0 1 5  0 5 4"> 
                     </MeshGeometry3D> 
                  </GeometryModel3D.Geometry> 
					
                  <GeometryModel3D.Material> 
                     <DiffuseMaterial x:Name = "matDiffuseMain">
                        <DiffuseMaterial.Brush> 
                           <SolidColorBrush Color = "Bisque"/> 
                        </DiffuseMaterial.Brush> 
                     </DiffuseMaterial> 
                  </GeometryModel3D.Material> 
					
               </GeometryModel3D>
            </ModelVisual3D.Content>
			 
            <ModelVisual3D.Transform>                         
               <RotateTransform3D> 
                  <RotateTransform3D.Rotation> 
                     <AxisAngleRotation3D x:Name = "rotate" Axis = "1 2 1"/> 
                  </RotateTransform3D.Rotation> 
               </RotateTransform3D>
            </ModelVisual3D.Transform> 
			 
         </ModelVisual3D> 
      </Viewport3D>
		
      <Slider Height = "23" HorizontalAlignment = "Left"  
         Margin = "145,271,0,0" Name = "slider1" 
         VerticalAlignment = "Top" Width = "269"  
         Maximum = "360" 
         Value = "{Binding ElementName = rotate, Path=Angle}" /> 
			
   </Grid> 
	
</Window>

응용 프로그램을 실행하면 창에 3D 개체와 슬라이더가 생성됩니다.

슬라이더를 밀면 창의 개체도 회전합니다.

위의 코드를 실행하고 더 많은 3D 지오메트리를 시도하는 것이 좋습니다.

WPF 응용 프로그램은 다음을 사용하여 비디오 및 오디오를 지원합니다. MediaElement. 오디오와 비디오를 응용 프로그램에 통합 할 수 있습니다. MediaElement 클래스는 Image 클래스와 비슷한 방식으로 작동합니다. 미디어를 가리 키기 만하면 렌더링됩니다. 가장 큰 차이점은 움직이는 이미지라는 것입니다.하지만 MP3와 같이 오디오 만 포함하고 비디오가없는 파일을 가리키면 화면에 아무것도 표시되지 않고 재생됩니다.

WPF는 컴퓨터 구성에 따라 모든 유형의 비디오 / 오디오 형식을 지원합니다. 미디어 파일이 미디어 플레이어를 재생하는 경우 동일한 컴퓨터의 WPF에서도 작동합니다.

애플리케이션에 멀티미디어를 통합하는 방법을 이해하는 예를 들어 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPFMultimedia.

  • 다음 XAML 코드는 미디어 요소와 세 개의 단추를 만들고 몇 가지 속성을 사용하여 초기화합니다.

<Window x:Class = "WPFMultimedia.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFMultimedia" 
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <StackPanel HorizontalAlignment = "Center" VerticalAlignment = "Center"> 
         <MediaElement Name = "myMedia" Source = "D:\MicrosoftMVA.mp4" 
            LoadedBehavior = "Manual" Width = "591" Height = "274" /> 
         <StackPanel Orientation = "Horizontal" Margin = "0,10,0,0">
            <Button Content = "Play" Margin = "0,0,10,0" Padding = "5" Click = "mediaPlay" /> 
            <Button Content = "Pause" Margin = "0,0,10,0" Padding = "5" Click = "mediaPause" />
            <Button x:Name = "muteButt" Content = "Mute" Padding = "5" Click = "mediaMute" /> 
         </StackPanel> 
      </StackPanel>
   </Grid> 
	
</Window>

다음은 다양한 버튼에 대한 C #의 Click 이벤트 구현입니다.

using System; 
using System.Windows; 
 
namespace WPFMultimedia { 

   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
         myMedia.Volume = 100; 
         myMedia.Play(); 
      } 
		
      void mediaPlay(Object sender, EventArgs e) { 
         myMedia.Play(); 
      }  
		
      void mediaPause(Object sender, EventArgs e) { 
         myMedia.Pause();
      } 
		
      void mediaMute(Object sender, EventArgs e) { 
		
         if (myMedia.Volume == 100) { 
            myMedia.Volume = 0; 
            muteButt.Content = "Listen"; 
         } 
         else { 
            myMedia.Volume = 100; 
            muteButt.Content = "Mute"; 
         } 
      } 
   } 
}

위 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다. 세 개의 버튼으로 동영상을 재생하고 재생을 제어 할 수 있습니다.

버튼을 사용하여 비디오를 일시 중지, 음소거 및 재생할 수 있습니다.

음성 합성기

WPF에는 텍스트를 음성으로 변환하는 기능이 있습니다. 이 API는 System.Speech 네임 스페이스에 포함되어 있습니다.SpeechSynthesizer 클래스는 텍스트를 말로 변환합니다.

간단한 예를 살펴 보겠습니다.

  • 이름으로 새 WPF 프로젝트 만들기 WPFTextToSpeech.

  • 참조로 추가하려면 System.Speech 어셈블리가 필요합니다. SpeechSynthesizer 일하는 수업.

  • 참조를 마우스 오른쪽 버튼으로 클릭하고 참조 추가를 선택합니다.

  • 참조 관리자 대화 상자가 열립니다. 이제 System.Speech 확인란을 선택하십시오.

  • 확인 버튼을 클릭합니다. 참조에서 System.Speech 어셈블리를 볼 수 있습니다.

  • 이제 버튼과 텍스트 상자를 도구 상자에서 디자인 창으로 드래그하십시오.

  • 다음 XAML 코드는 단추와 텍스트 상자를 만들고 일부 속성을 사용하여 초기화합니다.

<Window x:Class = "WPFTextToSpeech.MainWindow" 
   xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
   xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml" 
   xmlns:d = "http://schemas.microsoft.com/expression/blend/2008" 
   xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006" 
   xmlns:local = "clr-namespace:WPFTextToSpeech"
   mc:Ignorable = "d" Title = "MainWindow" Height = "350" Width = "604"> 
	
   <Grid> 
      <Button x:Name = "button" Content = "Speak"  
         HorizontalAlignment = "Left" Margin = "218,176,0,0"  
         VerticalAlignment = "Top" Width = "75"/> 
			
      <TextBox x:Name = "textBox" HorizontalAlignment = "Left"  
         Height = "23" Margin = "60,104,0,0" TextWrapping = "Wrap"  
         VerticalAlignment = "Top" Width = "418"/> 
   </Grid> 
	
</Window>
  • 다음은 텍스트 상자 내의 텍스트를 음성으로 변환하는 C #의 간단한 구현입니다.

using System.Speech.Synthesis; 
using System.Windows; 
 
namespace WPFTextToSpeech { 
   /// <summary> 
      /// Interaction logic for MainWindow.xaml 
   /// </summary> 
	
   public partial class MainWindow : Window { 
	
      public MainWindow() { 
         InitializeComponent(); 
      } 
		
      private void button_Click(object sender, RoutedEventArgs e) { 
		
         if (textBox.Text != "") {
            SpeechSynthesizer speechSynthesizer = new SpeechSynthesizer(); 
            speechSynthesizer.Speak(textBox.Text);  
         } 
         else { 
            MessageBox.Show("Write some thing in the textbox!");
         } 
      } 
   }
}

위 코드를 컴파일하고 실행하면 다음과 같은 창이 생성됩니다. 이제 텍스트 상자에 Hello World를 입력하고 말하기 버튼을 클릭합니다.

"Hello World"사운드가 생성됩니다. 텍스트 상자에 아무것도 입력하지 않으면 다음 메시지가 깜박입니다.

위의 예를 실행하는 것이 좋습니다.