开发环境:Win10 + VS2017 + .Net4.5

这个 ColorPicker 是<<WPF编程宝典:使用C#2012和.NET4.5>>这本书中的例子.这里我记录一下,主要是为了加深印象,防止以后要开发自定义控件的时候忘记一些基本的步骤,可以随时来查一下.

调到博文的最后可以先查看一下效果图.

  1. 建立ColorPicker类继承自 Control



    为什么继承自 Control,Control类继承自UIElement -> FrameworkElement -> ColorPicker,同时 Control 类提供了 Template 功能.

  2. 通知WPF,将为控件提供新的样式.

    方法是在静态构造函数中调用 OverrideMetadata()方法.

    static ColorPicker()
    {
    DefaultStyleKeyProperty.OverrideMetadata(
    typeof(ColorPicker), new FrameworkPropertyMetadata(typeof(ColorPicker)));
    }
  3. 编写控件逻辑,添加一些必要的属性|事件|方法 ...

  4. 重写 OnApplyTemplate 方法,为模板中的元素添加数据绑定或者关联事件处理程序

  5. 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(&quot;CornerRadius&quot;, 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(
    &quot;UseDynamicBorder&quot;, typeof(bool), typeof(ColorPicker),
    new FrameworkPropertyMetadata(true, OnUseDynamicBorderChanged)); public static readonly RoutedEvent ColorChangedEvent = EventManager.RegisterRoutedEvent(
    &quot;ColorChangedEvent&quot;, RoutingStrategy.Bubble,
    typeof(RoutedPropertyChangedEventHandler&lt;Color&gt;), typeof(ColorPicker));
    public event RoutedPropertyChangedEventHandler&lt;Color&gt; ColorChanged
    {
    add { AddHandler(ColorChangedEvent, value); }
    remove { RemoveHandler(ColorChangedEvent, value); }
    } static ColorPicker()
    {
    DefaultStyleKeyProperty.OverrideMetadata(
    typeof(ColorPicker), new FrameworkPropertyMetadata(typeof(ColorPicker)));
    } public ColorPicker()
    {
    Loaded += (sender, args) =&gt;
    {
    _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(&quot;PART_RedSlider&quot;) as RangeBase;
    if (slider != null)
    {
    Binding binding = new Binding()
    {
    Path = new PropertyPath(&quot;Red&quot;),
    Source = this,
    Mode = BindingMode.TwoWay
    };
    slider.SetBinding(RangeBase.ValueProperty, binding);
    slider.Maximum = RGBMaxValue;
    } slider = GetTemplateChild(&quot;PART_GreenSlider&quot;) 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(&quot;PART_BlueSlider&quot;) 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(&quot;PART_PreviewBrush&quot;) as SolidColorBrush;
    if (brush != null)
    {
    Binding bd = new Binding
    {
    Path = new PropertyPath(nameof(brush.Color)),
    Source = brush,
    Mode = BindingMode.OneWayToSource
    };
    SetBinding(ColorPicker.ColorProperty, bd);
    }
    } }

    }

  6. 差不多就是这样一个简单的ColorPicker就出来了.另外增加一个UseDynamicBorder,来控制控件边框的颜色是否动态改变.

    看一下效果图

  </div>

WPF自定义控件开发实例 - ColorPicker的更多相关文章

  1. TemplateBinding与Binding区别,以及WPF自定义控件开发的遭遇

    在上一次的文章WPF OnApplyTemplate 不执行 或者执行滞后的疑惑谈到怎么正确的开发自定义控件,我们控件的样式中,属性的绑定一般都是用TemplateBinding来完成,如下一个基本的 ...

  2. WPF和Expression Blend开发实例:一个样式实现的数字输入框

    原文:WPF和Expression Blend开发实例:一个样式实现的数字输入框 今天来一个比较奇淫技巧的手法,很少人用,同时也不推荐太过频繁的使用. 先上样式: <Style x:Key=&q ...

  3. WPF开发实例——仿QQ登录界面

    原文:WPF开发实例--仿QQ登录界面 版权声明:本文为博主原创文章,如需转载请标明转载地址 http://blog.csdn.net/u013981858 https://blog.csdn.net ...

  4. WPF自定义控件与样式(3)-TextBox & RichTextBox & PasswordBox样式、水印、Label标签、功能扩展

    一.前言.预览 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要是对文本 ...

  5. WPF自定义控件与样式(15)-终结篇 & 系列文章索引 & 源码共享

    系列文章目录  WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与样式(3)-TextBox & Ric ...

  6. WPF自定义控件与样式(12)-缩略图ThumbnailImage /gif动画图/图片列表

    一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要针对WPF项目 ...

  7. WPF自定义控件与样式(15)-终结篇

    原文:WPF自定义控件与样式(15)-终结篇 系列文章目录  WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与 ...

  8. WPF自定义控件创建

    WPF自定义控件创建 本文简单的介绍一下WPF自定义控件的开发. 首先,我们打开VisualStudio创建一个WPF自定义控件库,如下图: 然后,我们可以看到创建的解决方案如下: 在解决方案中,我们 ...

  9. 【转】WPF自定义控件与样式(12)-缩略图ThumbnailImage /gif动画图/图片列表

    一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要针对WPF项目 ...

随机推荐

  1. CentOS下利用mysqlbinlog恢复MySQL数据库

    如果不小心对数据库进行误操作,而又没有及时备份怎么办?这恐怕是广大的coder经常遇到的一类问题.我今天就因为不小心删除了某个数据库,但最后的备份是1个礼拜前的,唯一能解决的办法就是通过mysqlbi ...

  2. 20155312 2016-2017-2 《Java程序设计》第八周学习总结

    20155312 2016-2017-2 <Java程序设计>第八周学习总结 课堂内容总结 学习模式 游乐园模式-荒野求生模式 学习方法 以代码为中心->遇到不会的类和方法(参数等) ...

  3. lambda表达式和表达式树(深入理解c#)

    1.Lambda形式 1). Lambda表达式最冗长的形式: (显式类型的参数列表)=>{语句} 2). 大多数时候,都可以用一个表达式来表示主体,该表达式的值是Lambda的结果,在这些情况 ...

  4. Django的学习(一)————初入django

    一.基本指令 1.项目的建立: Django的项目建立,进入目录,打开cmd输入 django-admin startproject[项目名称],注意如果是在其他文件下把项目设计成资源文件. 2.Ap ...

  5. 纯css实现蒙层loading效果

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  6. 关于java项目中的XML文件

    一,xml的机制 1.xml文件会在服务器启动的时候进行加载 2.加载完成后根据xml文件里面配置的属性对集成的对象进行属性和行为赋予 3.xml会有很多不同的标签,每个标签都有它特定的含义 二.为什 ...

  7. UVaLive 3357 Pinary (Fib数列+递归)

    题意:求第 k 个不含前导 0 和连续 1 的二进制串. 析:1,10,100,101,1000,...很容易发现长度为 i 的二进制串的个数正好就是Fib数列的第 i 个数,因为第 i 个也有子问题 ...

  8. 利用JDK自带的keytool生成SSL证书然后导入到SpringBoot

    一:生成命令如下(这一步生成的暂不知道干嘛用的) E:\Desktop\Documents\证书>keytool -genkey -alias tomcat -keypass - -validi ...

  9. ACtiveMQ中间件-消息的接收和发送

    一.发送消息基本步骤 (1)启动ActiveMQ的的activemq.bat批处理文件或BrokerService节点 (2)创建连接使用的工厂类ActiveMQConnectionFactory通过 ...

  10. Oracle 数据库 dbConsole 配置笔记

    Oracle安装后,DBConsole有时安装会失败,这时可以通过以下命令恢复: set oracle_sid=orclemca -repos recreateemca -config dbcontr ...