요즘은 전문적인 앱을 만들고 싶다면 UI 디자인과 조작의 용이성이 매우 중요합니다. UI 디자인은 Photoshop이나 AI와 같은 디자인 도구를 사용할 수 있습니다. 이전에 WPF 디자인 도구인 Expression Blend가 PSD 파일이나 AI 디자인 파일을 직접 가져올 수 있다는 것을 배웠습니다(물론 모든 기능이 지원되는 것은 아닙니다). 다행히 드디어 기대했던 효과를 얻었습니다. 다음은 먼저 PS를 사용하여 벡터 그래픽 템플릿을 구축한 다음 Expression Blend를 사용하여 PSD 파일을 가져오고 PATH의 데이터 값을 얻어 멋진 개인화 진행률 표시줄을 만들기 위한 아름다운 UI를 구축하는 방법에 대한 단계별 예입니다. 제어.
1. Photoshop을 열고 새 빈 레이어를 만든 다음 PS 패턴 스탬프 도구를 클릭합니다.
2. 아래와 같이 원하는 브러시를 선택합니다(웹사이트에서 무료 브러시를 다운로드할 수 있음).
아래 그림과 같이 해당 위치를 클릭하세요.
3. CTRL을 누른 채 레이어를 선택하고 경로 패널로 전환한 다음 [다음에서 작업 경로 만들기]를 클릭합니다. 선택] 버튼, 아래와 같이
위 그림에 주목하세요. 빨간색 상자 버튼은 [선택에서 작업 경로 생성]입니다. 클릭하면 다음 그림이 나타납니다.
4. 이것은 가장 중요한 단계입니다. 벡터 마스크를 만들고 레이어 패널로 전환한 다음 [펜] 도구를 클릭하고 오른쪽 클릭 메뉴에서 [벡터 마스크 만들기] 항목을 선택합니다.
그러면 PS에서 아래 그림의 효과를 볼 수 있습니다. 창작이 성공했다는 것.
나중에 사용할 수 있도록 PS 파일을 진행률 표시줄.PSD 파일로 저장하세요.
5. Open Expression Blend 4를 사용하여 새 WPF 프로젝트를 만든 다음 아래와 같이 PSD 파일을 가져옵니다.
가져오기 성공 후 해당 영상의 클립 데이터를 복사할 수 있습니다. 그래픽, 이는 WPF의 PATH에 필요한 데이터 값입니다.
멋진 WPF 진행률 표시줄 컨트롤을 만들어 보겠습니다.
6. VS2010에서 프로젝트를 다시 열고 아래와 같이 WPF 사용자 정의 컨트롤 라이브러리를 추가합니다.
17 컨트롤 UI와 배경 코드를 다음과 같이 작성합니다.
<resourcedictionary> <style> <Setter Property="Template"> <Setter.Value> <ControlTemplate TargetType="{x:Type local:CustomProgressControl}"> <Grid x:Name="PART_container" Background="{TemplateBinding Background}" Width="{TemplateBinding Width}" Height="{TemplateBinding Height}"> <Path x:Name="PART_foreground_P" Visibility="Collapsed" Stretch="Fill" Stroke="Transparent" Fill="{TemplateBinding Foreground}" StrokeThickness="2" Data="F1M56,33C58,33 60,33 62,33 61.667,33.667 61.333,34.333 61,35 57.794,34.859 53.856,36.079 56,33z M49,24C54.364,24.735 53.554,24.821 56,28 53.807,30.696 55.287,29.902 51,31 50,30 49,29 48,28 48.333,26.667 48.667,25.333 49,24z M62,22C63.666,22.333 65.333,22.667 67,23 66.333,27.571 65.935,27.376 64,30 63,29.333 62,28.667 61,28 61,27.667 61,27.333 61,27 61.333,25.333 61.667,23.666 62,22z M46,15C46.667,15 47.333,15 48,15 47.333,17.333 46.667,19.667 46,22 45.333,22 44.667,22 44,22 44.667,19.667 45.333,17.333 46,15z M63,13C64.923,14.392 63.599,13.101 65,15 64.333,14.333 63.667,13.667 63,13z M55,12C56.666,12.667 58.333,13.333 60,14 60,15.333 60,16.667 60,18 59.333,18 58.667,18 58,18 54.722,21.928 52.838,16.561 52,14 53,13.333 54,12.667 55,12z M128,1C128.667,1 129.333,1 130,1 128.882,10.058 122.793,12.326 122,23 126.364,22.028 126.876,21.206 131,22 130,23.333 129,24.667 128,26 110.659,32.752 112.704,45.252 103,59 95.769,69.245 82.761,82.131 72,89 72.621,101.092 82.373,112.463 90,118 90.333,118 90.667,118 91,118 93.274,107.421 107.464,106.386 104,92 101.667,87.667 99.333,83.333 97,79 98,78 99,77 100,76 122.812,77.152 112.786,100.488 115,114 116.666,115.666 118.333,117.333 120,119 127.359,142.373 118.776,160.626 106,168 110.337,176.877 114.918,188.188 121,197 127.441,206.332 140.794,210.508 148,220 146.506,223.067 146.885,223.215 144,225 113.08,236.802 62.376,138.34 36,147 34.077,151.751 32.347,152.761 28,155 17.556,150.255 9.333,141.565 9,127 13.999,120.001 19,112.999 24,106 22.667,102.667 21.333,99.333 20,96 7.555,96.019 4.392,90.889 1,82 1.465,74.486 3.768,68.82 9,66 19.848,56.341 31.922,71.946 38,77 37.406,83.299 36.792,87.413 39,92 41.333,89.667 43.667,87.333 46,85 81.009,67.269 105.228,31.536 128,1z" /> <Rectangle x:Name="PART_mask" Fill="{TemplateBinding Background}" VerticalAlignment="Stretch" HorizontalAlignment="Stretch" /> <Path x:Name="PART_outline_P" Visibility="Collapsed" Stretch="Fill" Stroke="{TemplateBinding BorderBrush}" Fill="Transparent" StrokeThickness="2" Data="F1M56,33C58,33 60,33 62,33 61.667,33.667 61.333,34.333 61,35 57.794,34.859 53.856,36.079 56,33z M49,24C54.364,24.735 53.554,24.821 56,28 53.807,30.696 55.287,29.902 51,31 50,30 49,29 48,28 48.333,26.667 48.667,25.333 49,24z M62,22C63.666,22.333 65.333,22.667 67,23 66.333,27.571 65.935,27.376 64,30 63,29.333 62,28.667 61,28 61,27.667 61,27.333 61,27 61.333,25.333 61.667,23.666 62,22z M46,15C46.667,15 47.333,15 48,15 47.333,17.333 46.667,19.667 46,22 45.333,22 44.667,22 44,22 44.667,19.667 45.333,17.333 46,15z M63,13C64.923,14.392 63.599,13.101 65,15 64.333,14.333 63.667,13.667 63,13z M55,12C56.666,12.667 58.333,13.333 60,14 60,15.333 60,16.667 60,18 59.333,18 58.667,18 58,18 54.722,21.928 52.838,16.561 52,14 53,13.333 54,12.667 55,12z M128,1C128.667,1 129.333,1 130,1 128.882,10.058 122.793,12.326 122,23 126.364,22.028 126.876,21.206 131,22 130,23.333 129,24.667 128,26 110.659,32.752 112.704,45.252 103,59 95.769,69.245 82.761,82.131 72,89 72.621,101.092 82.373,112.463 90,118 90.333,118 90.667,118 91,118 93.274,107.421 107.464,106.386 104,92 101.667,87.667 99.333,83.333 97,79 98,78 99,77 100,76 122.812,77.152 112.786,100.488 115,114 116.666,115.666 118.333,117.333 120,119 127.359,142.373 118.776,160.626 106,168 110.337,176.877 114.918,188.188 121,197 127.441,206.332 140.794,210.508 148,220 146.506,223.067 146.885,223.215 144,225 113.08,236.802 62.376,138.34 36,147 34.077,151.751 32.347,152.761 28,155 17.556,150.255 9.333,141.565 9,127 13.999,120.001 19,112.999 24,106 22.667,102.667 21.333,99.333 20,96 7.555,96.019 4.392,90.889 1,82 1.465,74.486 3.768,68.82 9,66 19.848,56.341 31.922,71.946 38,77 37.406,83.299 36.792,87.413 39,92 41.333,89.667 43.667,87.333 46,85 81.009,67.269 105.228,31.536 128,1z" /> <TextBlock x:Name="PART_percentage_text" VerticalAlignment="Center" HorizontalAlignment="Center" FontSize="16" FontWeight="ExtraBlack" Foreground="{TemplateBinding TextForeground}"/> </style> </resourcedictionary>
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 WpfCustomProgressControl { [TemplatePart(Name = "PART_mask", Type = typeof(Rectangle))] [TemplatePart(Name = "PART_container", Type = typeof(Grid))] [TemplatePart(Name = "PART_percentage_text", Type = typeof(TextBlock))] [TemplatePart(Name = "PART_foreground_P", Type = typeof(Path))] [TemplatePart(Name = "PART_outline_P", Type = typeof(Path))] public class CustomProgressControl : ProgressBar { static CustomProgressControl() { DefaultStyleKeyProperty.OverrideMetadata(typeof(CustomProgressControl), new FrameworkPropertyMetadata(typeof(CustomProgressControl))); } Rectangle mask; Grid container; TextBlock percentageText; Path foreground_P; Path outline_P; #region TextForeground 文本 public SolidColorBrush TextForeground { get { return (SolidColorBrush)GetValue(TextForegroundProperty); } set { SetValue(TextForegroundProperty, value); } } public static readonly DependencyProperty TextForegroundProperty = DependencyProperty.Register("TextForeground", typeof(SolidColorBrush), typeof(CustomProgressControl), new FrameworkPropertyMetadata(new SolidColorBrush(Colors.DarkGray))); #endregion public override void OnApplyTemplate() { base.OnApplyTemplate(); foreground_P = this.Template.FindName("PART_foreground_P", this) as Path; outline_P = this.Template.FindName("PART_outline_P", this) as Path; mask = this.Template.FindName("PART_mask", this) as Rectangle; container = this.Template.FindName("PART_container", this) as Grid; percentageText = this.Template.FindName("PART_percentage_text", this) as TextBlock; if (foreground_P != null) { foreground_P.Visibility = Visibility.Visible; outline_P.Visibility = Visibility.Visible; } Width = double.IsNaN(Width) ? 50 : Width; Height = double.IsNaN(Height) ? 135 : Height; Minimum = double.IsNaN(Minimum) ? 0 : Minimum; Maximum = double.IsNaN(Maximum) ? 100 : Maximum; if (mask != null) { var percentageValue = Value / Maximum; var awayMargin = percentageValue * Height; var percentageString = string.Empty; if (percentageValue > 0) percentageString = (percentageValue * 100).ToString("##"); else if (percentageValue == 0) percentageString = "0"; percentageText.Text = string.Format("{0}%", string.IsNullOrEmpty(percentageString) ? "0" : percentageString); mask.Margin = new Thickness(0, 0, 0, awayMargin); } container.Clip = new RectangleGeometry { Rect = new Rect(0, 0, Width, Height) }; mask.Width = Width; mask.Height = Height; } protected override void OnValueChanged(double oldValue, double newValue) { base.OnValueChanged(oldValue, newValue); if (Value < Minimum) { Value = Minimum; } if (Value > Maximum) { Value = Maximum; } if (mask != null) { var percentageValue = Value / Maximum; var awayMargin = percentageValue * Height; var percentageString = string.Empty; if (percentageValue > 0) percentageString = (percentageValue * 100).ToString("##"); else if (percentageValue == 0) percentageString = "0"; percentageText.Text = string.Format("{0}%", string.IsNullOrEmpty(percentageString) ? "0" : percentageString); //蒙板来变更进度 mask.Margin = new Thickness(0, 0, 0, awayMargin); } } } }
18 컨트롤을 WpfPSDemo의 기본 인터페이스로 끌어서 속성을 사용자 정의합니다. 코드는 다음과 같습니다.
<Window xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:ed="http://schemas.microsoft.com/expression/2010/drawing" x:Class="WpfPSDemo.MainWindow" x:Name="Window" BorderThickness="0" Title="MainWindow" Width="430" Height="480" xmlns:my="clr-namespace:WpfCustomProgressControl;assembly=WpfCustomProgressControl"> <my:CustomProgressControl Name="customProgressControl1" Width="200" Height="200" Value="50" Background="Yellow" Foreground="Red" BorderBrush="red" /> </Window>
using System; using System.Collections.Generic; 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.Shapes; using System.Threading; namespace WpfPSDemo { /// <summary> /// MainWindow.xaml 的交互逻辑 /// </summary> public partial class MainWindow : Window { Thread timeThread; int i = 0; public MainWindow() { this.InitializeComponent(); this.customProgressControl1.Value = 0; this.Background = Brushes.Yellow; timeThread = new Thread(new ThreadStart(DispatcherThread)); timeThread.Start(); } public void DispatcherThread() { //可以通过循环条件来控制UI的更新 while (true) { ///线程方法委托(无参方法) this.customProgressControl1.Dispatcher.BeginInvoke(new Action(UpdateTime)); Thread.Sleep(200); } } private void UpdateTime() { if (i < 100) { i++; this.customProgressControl1.Value = i; } else { timeThread.Abort(); } } } }
코드를 실행하면 효과는 다음과 같습니다. 다음:
더 많은 Photoshop 및 WPF Twin Sword를 사용하여 멋지고 개인화된 진행률 표시줄 컨트롤을 만들려면, 관련 기사는 PHP 중국어에 주목하세요!