WPF自定义控件开发实例 - ColorPicker
开发环境:Win10 + VS2017 + .Net4.5
这个 ColorPicker 是<<WPF编程宝典:使用C#2012和.NET4.5>>这本书中的例子.这里我记录一下,主要是为了加深印象,防止以后要开发自定义控件的时候忘记一些基本的步骤,可以随时来查一下.
调到博文的最后可以先查看一下效果图.
建立ColorPicker类继承自 Control类

为什么继承自 Control,Control类继承自UIElement -> FrameworkElement -> ColorPicker,同时 Control 类提供了 Template 功能.通知WPF,将为控件提供新的样式.
方法是在静态构造函数中调用 OverrideMetadata()方法.static ColorPicker()
{
DefaultStyleKeyProperty.OverrideMetadata(
typeof(ColorPicker), new FrameworkPropertyMetadata(typeof(ColorPicker)));
}
编写控件逻辑,添加一些必要的属性|事件|方法 ...
重写 OnApplyTemplate 方法,为模板中的元素添加数据绑定或者关联事件处理程序
ColorPicker类的完整代码:
using System;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Controls.Primitives;
using System.Windows.Data;
using System.Windows.Media; namespace Demo.DIYControl.Controls.DeepInDIY
{
[TemplatePart(Name = RedSliderName,Type = typeof(RangeBase))]
[TemplatePart(Name = GreenSliderName, Type = typeof(RangeBase))]
[TemplatePart(Name = BlueSliderName, Type = typeof(RangeBase))]
[TemplatePart(Name = PreviewBrushName, Type = typeof(SolidColorBrush))]
public class ColorPicker:Control
{
private const string RedSliderName = "PART_RedSlider";
private const string GreenSliderName = "PART_GreenSlider";
private const string BlueSliderName = "PART_BlueSlider";
private const string PreviewBrushName = "PART_PreviewBrush";private Brush _initializeBorderBrush; public const byte RGBMaxValue = 255; public byte Red
{
get { return (byte)GetValue(RedProperty); }
set { SetValue(RedProperty, value); }
}
// Using a DependencyProperty as the backing store for Red. This enables animation, styling, binding, etc...
public static readonly DependencyProperty RedProperty =
DependencyProperty.Register(
nameof(Red), typeof(byte), typeof(ColorPicker),
new FrameworkPropertyMetadata(OnColorRGBChanged)); public byte Green
{
get { return (byte)GetValue(GreenProperty); }
set { SetValue(GreenProperty, value); }
}
// Using a DependencyProperty as the backing store for Green. This enables animation, styling, binding, etc...
public static readonly DependencyProperty GreenProperty =
DependencyProperty.Register(
nameof(Green), typeof(byte), typeof(ColorPicker),
new FrameworkPropertyMetadata(OnColorRGBChanged)); public byte Blue
{
get { return (byte)GetValue(BlueProperty); }
set { SetValue(BlueProperty, value); }
}
// Using a DependencyProperty as the backing store for Blue. This enables animation, styling, binding, etc...
public static readonly DependencyProperty BlueProperty =
DependencyProperty.Register(
nameof(Blue), typeof(byte), typeof(ColorPicker),
new FrameworkPropertyMetadata(OnColorRGBChanged)); public Color Color
{
get { return (Color)GetValue(ColorProperty); }
set { SetValue(ColorProperty, value); }
}
// Using a DependencyProperty as the backing store for Color. This enables animation, styling, binding, etc...
public static readonly DependencyProperty ColorProperty =
DependencyProperty.Register(nameof(Color), typeof(Color),
typeof(ColorPicker), new FrameworkPropertyMetadata(Colors.Black,OnColorChanged)); public CornerRadius CornerRadius
{
get { return (CornerRadius)GetValue(CornerRadiusProperty); }
set { SetValue(CornerRadiusProperty, value); }
}
// Using a DependencyProperty as the backing store for CornerRadius. This enables animation, styling, binding, etc...
public static readonly DependencyProperty CornerRadiusProperty =
DependencyProperty.Register("CornerRadius", typeof(CornerRadius), typeof(ColorPicker),
new FrameworkPropertyMetadata(default(CornerRadius))); public bool UseDynamicBorder
{
get { return (bool)GetValue(UseDynamicBorderProperty); }
set { SetValue(UseDynamicBorderProperty, value); }
}
// Using a DependencyProperty as the backing store for UseDynamicBorder. This enables animation, styling, binding, etc...
public static readonly DependencyProperty UseDynamicBorderProperty =
DependencyProperty.Register(
"UseDynamicBorder", typeof(bool), typeof(ColorPicker),
new FrameworkPropertyMetadata(true, OnUseDynamicBorderChanged)); public static readonly RoutedEvent ColorChangedEvent = EventManager.RegisterRoutedEvent(
"ColorChangedEvent", RoutingStrategy.Bubble,
typeof(RoutedPropertyChangedEventHandler<Color>), typeof(ColorPicker));
public event RoutedPropertyChangedEventHandler<Color> ColorChanged
{
add { AddHandler(ColorChangedEvent, value); }
remove { RemoveHandler(ColorChangedEvent, value); }
} static ColorPicker()
{
DefaultStyleKeyProperty.OverrideMetadata(
typeof(ColorPicker), new FrameworkPropertyMetadata(typeof(ColorPicker)));
} public ColorPicker()
{
Loaded += (sender, args) =>
{
_initializeBorderBrush = BorderBrush; //save initial borderbrush
BorderBrush = UseDynamicBorder? new SolidColorBrush(Color) : BorderBrush;
};
} private static void OnColorRGBChanged(DependencyObject sender,
DependencyPropertyChangedEventArgs args)
{
ColorPicker colorPicker = sender as ColorPicker;
colorPicker = colorPicker ?? throw new ArgumentException(); Color color = colorPicker.Color;
if (args.Property == RedProperty)
color.R = (byte)args.NewValue;
else if (args.Property == GreenProperty)
color.G = (byte)args.NewValue;
else if (args.Property == BlueProperty)
color.B = (byte)args.NewValue; colorPicker.Color = color;
} private static void OnColorChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
ColorPicker colorPicker = d as ColorPicker;
d = d ?? throw new ArgumentException();
colorPicker.Red = colorPicker.Color.R;
colorPicker.Green = colorPicker.Color.G;
colorPicker.Blue = colorPicker.Color.B; //set border color
if (colorPicker.UseDynamicBorder)
colorPicker.BorderBrush = new SolidColorBrush(colorPicker.Color); colorPicker.RaiseEvent(
new RoutedEventArgs(ColorChangedEvent, e.NewValue));
} private static void OnUseDynamicBorderChanged(
DependencyObject d, DependencyPropertyChangedEventArgs args)
{
ColorPicker colorPicker = d as ColorPicker;
colorPicker = colorPicker ?? throw new ArgumentException(); colorPicker.BorderBrush = (bool)args.NewValue ? new SolidColorBrush(colorPicker.Color) : colorPicker._initializeBorderBrush;
} public override void OnApplyTemplate()
{
base.OnApplyTemplate(); RangeBase slider = GetTemplateChild("PART_RedSlider") as RangeBase;
if (slider != null)
{
Binding binding = new Binding()
{
Path = new PropertyPath("Red"),
Source = this,
Mode = BindingMode.TwoWay
};
slider.SetBinding(RangeBase.ValueProperty, binding);
slider.Maximum = RGBMaxValue;
} slider = GetTemplateChild("PART_GreenSlider") as RangeBase;
if (slider != null)
{
Binding binding = new Binding()
{
Path = new PropertyPath(nameof(Green)),
Source = this,
Mode = BindingMode.TwoWay
};
slider.SetBinding(RangeBase.ValueProperty, binding);
slider.Maximum = RGBMaxValue;
} slider = GetTemplateChild("PART_BlueSlider") as RangeBase;
if (slider != null)
{
Binding binding = new Binding()
{
Path = new PropertyPath(nameof(Blue)),
Source = this,
Mode = BindingMode.TwoWay
};
slider.SetBinding(RangeBase.ValueProperty, binding);
slider.Maximum = RGBMaxValue;
} SolidColorBrush brush = GetTemplateChild("PART_PreviewBrush") as SolidColorBrush;
if (brush != null)
{
Binding bd = new Binding
{
Path = new PropertyPath(nameof(brush.Color)),
Source = brush,
Mode = BindingMode.OneWayToSource
};
SetBinding(ColorPicker.ColorProperty, bd);
}
} }
}
差不多就是这样一个简单的ColorPicker就出来了.另外增加一个UseDynamicBorder,来控制控件边框的颜色是否动态改变.
看一下效果图

</div>
WPF自定义控件开发实例 - ColorPicker的更多相关文章
- TemplateBinding与Binding区别,以及WPF自定义控件开发的遭遇
在上一次的文章WPF OnApplyTemplate 不执行 或者执行滞后的疑惑谈到怎么正确的开发自定义控件,我们控件的样式中,属性的绑定一般都是用TemplateBinding来完成,如下一个基本的 ...
- WPF和Expression Blend开发实例:一个样式实现的数字输入框
原文:WPF和Expression Blend开发实例:一个样式实现的数字输入框 今天来一个比较奇淫技巧的手法,很少人用,同时也不推荐太过频繁的使用. 先上样式: <Style x:Key=&q ...
- WPF开发实例——仿QQ登录界面
原文:WPF开发实例--仿QQ登录界面 版权声明:本文为博主原创文章,如需转载请标明转载地址 http://blog.csdn.net/u013981858 https://blog.csdn.net ...
- WPF自定义控件与样式(3)-TextBox & RichTextBox & PasswordBox样式、水印、Label标签、功能扩展
一.前言.预览 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要是对文本 ...
- WPF自定义控件与样式(15)-终结篇 & 系列文章索引 & 源码共享
系列文章目录 WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与样式(3)-TextBox & Ric ...
- WPF自定义控件与样式(12)-缩略图ThumbnailImage /gif动画图/图片列表
一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要针对WPF项目 ...
- WPF自定义控件与样式(15)-终结篇
原文:WPF自定义控件与样式(15)-终结篇 系列文章目录 WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与 ...
- WPF自定义控件创建
WPF自定义控件创建 本文简单的介绍一下WPF自定义控件的开发. 首先,我们打开VisualStudio创建一个WPF自定义控件库,如下图: 然后,我们可以看到创建的解决方案如下: 在解决方案中,我们 ...
- 【转】WPF自定义控件与样式(12)-缩略图ThumbnailImage /gif动画图/图片列表
一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要针对WPF项目 ...
随机推荐
- @Html.EditorFor() 用法
@Html.EditorFor()返回一个由表达式表示的对象中的每个属性所对应的input元素,主要是针对强类型,一般这种方式用得多些a.@Html.EditorFor(mode=>mode.N ...
- 【Git】 GitLab服务器社区版安装与配置
GitLab简介 GitLab 是一个用于仓库管理系统的开源项目,使用Git作为代码管理工具,并在此基础上搭建起来的web服务 GitLab系统架构 当~git在图片中引用时,它表示git用户的主目录 ...
- ios蓝牙自定义快捷键
http://www.paopaoche.net/app/12072.html Beekeyboard
- lambda表达式和表达式树(深入理解c#)
1.Lambda形式 1). Lambda表达式最冗长的形式: (显式类型的参数列表)=>{语句} 2). 大多数时候,都可以用一个表达式来表示主体,该表达式的值是Lambda的结果,在这些情况 ...
- HTML-入门篇day01
HTML-入门篇day01 1.web C/S:Client Server 客户端 服务器 QQ,... B/S:Browser Server 浏览器 服务器 PC机: ...
- 2019.01.08 bzoj4543: [POI2014]Hotel加强版(长链剖分+dp)
传送门 代码: 长链剖分好题. 题意:给你一棵树,问树上选三个互不相同的节点,使得这个三个点两两之间距离相等的方案数. 思路: 先考虑dpdpdp. fi,jf_{i,j}fi,j表示iii子树中离 ...
- 2019.01.02 bzoj5300: [Cqoi2018]九连环(fft优化高精+快速幂)
传送门 题意不好描述(自己看样例解释) 首先可以推出一个递推式:fn=fn−1+2fn−2+1f_n=f_{n-1}+2f_{n-2}+1fn=fn−1+2fn−2+1 然后可以构造两个等式: ...
- 2018.06.27"Shortest" pair of paths(费用流)
"Shortest" pair of paths Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 1589 A ...
- vi三种模式的切换
基础上vi/vim共分为三种模式,分别是命令模式,输入模式和底线命令模式. 一.命令模式 用户刚刚启动vi/vim,便进入了命令模式. 在此状态下敲击键盘动作会被vim识别为命令,而非输入字符.比如我 ...
- CAS 界面根据不同的域名显示不同的界面
概要 在实际需求中,客户想通过不同的域名显示不同的登录界面,比如输入 manage.aps.cn 显示运维管理登录,business.aps.cn 显示业务管理登录. 实现方法 1.准备两套登录UI ...