WPF中如何自定义控件

news2025/5/11 20:55:01

WPF自定义控件简化版:账户菜单按钮(AccountButton)

我们以**“账户菜单按钮”为例,用更清晰的架构实现一个支持标题显示、渐变背景、选中状态高亮**的自定义控件。以下是分步拆解:


一、控件核心功能

我们要做一个类似这样的控件:

  • 外观:一个圆形/圆角的小按钮,中间显示标题(如账户首字母)。
  • 背景:支持自定义渐变颜色(从颜色A到颜色B)。
  • 选中状态:左侧显示一条高亮竖线(选中时可见,未选中时隐藏)。

二、控件架构设计(分3步)

步骤1:创建UserControl(控件容器)

WPF中自定义控件最常用的方式是继承UserControl(用户控件),它本质是一个可复用的UI容器,包含自己的XAML布局和后台代码。

步骤2:定义依赖属性(暴露控件属性)

为了让控件能被外部配置(如修改标题、颜色、选中状态),需要将这些属性注册为依赖属性(DependencyProperty)。依赖属性的优势是支持数据绑定、样式设置和动态更新。

步骤3:设计XAML布局(可视化界面)

在XAML中定义控件的UI结构,将依赖属性绑定到具体的UI元素(如文本、背景颜色、可见性)。


三、完整代码实现(简化版)

1. 新建UserControl:AccountButton.xaml

XAML中定义控件的布局,核心是一个Button,内部包含两个Border(高亮条和背景块)。

<UserControl x:Class="MyApp.Controls.AccountButton"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             x:Name="Root"  <!-- 给控件起别名,方便绑定 -->
             Width="30" Height="30">  <!-- 固定控件大小 -->

    <!-- 根元素是一个按钮 -->
    <Button Style="{StaticResource BaseButtonStyle}">  <!-- 假设外部有基础样式 -->
        <Grid>
            <!-- 选中状态高亮条(左侧竖线) -->
            <Border x:Name="ActiveIndicator"
                    HorizontalAlignment="Left"
                    Width="3"
                    Background="#FF0078D7"  <!-- 蓝色高亮 -->
                    Visibility="Collapsed"  <!-- 默认隐藏 -->
                    CornerRadius="0 2 2 0"/>  <!-- 右侧圆角 -->

            <!-- 背景块(显示渐变和标题) -->
            <Border x:Name="ContentBorder"
                    Width="25" Height="25"
                    HorizontalAlignment="Center"
                    VerticalAlignment="Center"
                    CornerRadius="3">  <!-- 圆角 -->
                <!-- 渐变背景 -->
                <Border.Background>
                    <LinearGradientBrush StartPoint="0,0" EndPoint="1,1">
                        <!-- 起始色绑定控件的GradientStart属性 -->
                        <GradientStop Color="{Binding GradientStart, ElementName=Root}" Offset="0"/>
                        <!-- 结束色绑定控件的GradientEnd属性 -->
                        <GradientStop Color="{Binding GradientEnd, ElementName=Root}" Offset="1"/>
                    </LinearGradientBrush>
                </Border.Background>

                <!-- 标题文本(绑定控件的Title属性) -->
                <TextBlock Text="{Binding Title, ElementName=Root}"
                           VerticalAlignment="Center"
                           HorizontalAlignment="Center"
                           Foreground="White"
                           FontWeight="Bold"/>
            </Border>
        </Grid>
    </Button>
</UserControl>
2. 后台代码:AccountButton.xaml.cs

定义依赖属性,并处理选中状态的逻辑(例如:选中时显示高亮条)。

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

namespace MyApp.Controls
{
    public partial class AccountButton : UserControl
    {
        public AccountButton()
        {
            InitializeComponent();
        }

        // ----------------------
        //  依赖属性定义(共3个)
        // ----------------------

        // 1. 标题(如账户首字母)
        public string Title
        {
            get => (string)GetValue(TitleProperty);
            set => SetValue(TitleProperty, value);
        }
        public static readonly DependencyProperty TitleProperty = 
            DependencyProperty.Register(
                "Title",              // 属性名(必须与包装器一致)
                typeof(string),       // 属性类型
                typeof(AccountButton) // 控件类型(当前类)
            );

        // 2. 渐变起始色
        public Color GradientStart
        {
            get => (Color)GetValue(GradientStartProperty);
            set => SetValue(GradientStartProperty, value);
        }
        public static readonly DependencyProperty GradientStartProperty = 
            DependencyProperty.Register(
                "GradientStart", 
                typeof(Color), 
                typeof(AccountButton)
            );

        // 3. 渐变结束色
        public Color GradientEnd
        {
            get => (Color)GetValue(GradientEndProperty);
            set => SetValue(GradientEndProperty, value);
        }
        public static readonly DependencyProperty GradientEndProperty = 
            DependencyProperty.Register(
                "GradientEnd", 
                typeof(Color), 
                typeof(AccountButton)
            );

        // 4. 选中状态(新增:控制高亮条可见性)
        public bool IsActive
        {
            get => (bool)GetValue(IsActiveProperty);
            set => SetValue(IsActiveProperty, value);
        }
        public static readonly DependencyProperty IsActiveProperty = 
            DependencyProperty.Register(
                "IsActive", 
                typeof(bool), 
                typeof(AccountButton),
                new PropertyMetadata(false, OnIsActiveChanged) // 添加回调:状态变化时触发
            );

        // 当IsActive属性变化时,更新高亮条可见性
        private static void OnIsActiveChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            var button = d as AccountButton;
            if (button != null)
            {
                // 根据IsActive的值显示/隐藏高亮条
                button.ActiveIndicator.Visibility = (bool)e.NewValue 
                    ? Visibility.Visible 
                    : Visibility.Collapsed;
            }
        }
    }
}

四、代码关键点解释(超简单版)

1. 依赖属性的作用
  • 外部可以通过TitleGradientStart等属性直接配置控件(类似原生控件的WidthHeight)。
  • 示例:在主窗口中使用控件时,可以这样写:
    <controls:AccountButton 
        Title="A" 
        GradientStart="#FF0078D7" 
        GradientEnd="#FF00B4D8" 
        IsActive="True"/>
    
2. 选中状态的自动更新

通过IsActivePropertyPropertyMetadata添加了OnIsActiveChanged回调函数。当IsActive的值变化时(如从False变为True),会自动触发这个函数,更新高亮条的可见性。

3. XAML绑定的逻辑
  • ElementName=RootRoot是UserControl的x:Name,通过它可以直接访问控件本身的属性(如GradientStart)。
  • 渐变颜色直接绑定到GradientStartGradientEnd,当这两个属性的值变化时,背景会自动更新。

五、最终效果

  • 未选中时:显示渐变背景和标题,左侧高亮条隐藏。
  • 选中时:左侧高亮条显示(颜色固定为蓝色),其他部分不变。

通过这种架构,你可以快速扩展控件功能(如添加点击事件、修改高亮条颜色等),核心逻辑清晰易理解!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2373418.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL事务和JDBC中的事务操作

一、什么是事务 事务是数据库操作的最小逻辑单元&#xff0c;具有"全有或全无"的特性。以银行转账为例&#xff1a; 典型场景&#xff1a; 从A账户扣除1000元 向B账户增加1000元 这两个操作必须作为一个整体执行&#xff0c;要么全部成功&#xff0c;要么全部失败…

每日脚本学习5.10 - XOR脚本

xor运算的简介 异或就是对于二进制的数据可以 进行同0异1 简单的演示 &#xff1a; 结果是 这个就是异或 异或的作用 1、比较两数是否相等 2、可以进行加密 加密就是需要key 明文 :0b010110 key : 0b1010001 这个时候就能进行加密 明文 ^ key密文 还有这个加密比…

【编译原理】总结

核心 闭包&#xff0c;正则闭包 产生式&#xff08;规则&#xff09; 文法 G[S](&#xff0c;&#xff0c;P&#xff0c;S) 一组规则的集合 &#xff1a;非终结符 &#xff1a;终结符 P&#xff1a;产生式 S&#xff1a;开始符号 推导 归约 规范&#xff08;最右&#xff…

docker创建一个centOS容器安装软件(以宝塔为例)的详细步骤

备忘&#xff1a;后续偶尔忘记了docker虚拟机与宿主机的端口映射关系&#xff0c;来这里查看即可&#xff1a; docker run -d \ --name baota \ --privilegedtrue \ -p 8888:8888 \ -p 8880:80 \ -p 8443:443 \ -p 8820:20 \ -p 8821:21 \ -v /home/www:/www/wwwroot \ centos…

OpenVLA:开源的视觉-语言-动作模型

1. 简介 让我们先来介绍一下什么是OpenVLA&#xff0c;在这里&#xff1a; https://openvla.github.io/ 可以看到他们的论文、数据、模型。 OpenVLA 是一个拥有 70亿参数的开源 **视觉-语言-动作&#xff08;VLA&#xff09;**模型。它是在 Open X-Embodiment 数据集 中的 97万…

Matlab/Simulink的一些功能用法笔记(4)

水一篇帖子 01--MATLAB工作区的保护眼睛颜色设置 默认的工作区颜色为白色 在网上可以搜索一些保护眼睛的RGB颜色参数设置 在MATLAB中按如下设置&#xff1a; ①点击预设 ②点击颜色&#xff0c;点击背景色的三角标符号 ③点击更多颜色&#xff0c;找到RGB选项 ④填写颜色参数…

Elasticsearch:我们如何在全球范围内实现支付基础设施的现代化?

作者&#xff1a;来自 Elastic Kelly Manrique SWIFT 和 Elastic 如何应对基础设施复杂性、误报问题以及日益增长的合规要求。 金融服务公司在全球范围内管理实时支付方面面临前所未有的挑战。SWIFT&#xff08;Society for Worldwide Interbank Financial Telecommunication -…

matlab介绍while函数

MATLAB 中的 while 语句介绍 在 MATLAB 中&#xff0c;while 语句是一种循环结构&#xff0c;用于在满足特定条件时反复执行一段代码块。与 for 循环不同&#xff0c;while 循环的执行次数是动态的&#xff0c;取决于循环条件是否为真。 语法 while condition% 循环体代码 e…

如何解决 PowerShell 显示 “此系统上禁用了脚本运行” 的问题

在 Windows 11 或 10 的 PowerShell 中运行脚本时,你可能会遇到一个错误,提示系统上禁用了脚本运行。这是一种安全功能,而不是系统问题,旨在防止可能有害的脚本自动运行。然而,如果你需要运行脚本来完成某些任务,或者你在系统上做了软件开发或测试的环境,那么你需要在 P…

深入浅出之STL源码分析4_类模版

1.引言 我在上面的文章中讲解了vector的基本操作&#xff0c;然后提出了几个问题。 STL之vector基本操作-CSDN博客 1.刚才我提到了我的编译器版本是g 11.4.0&#xff0c;而我们要讲解的是STL&#xff08;标准模板库&#xff09;&#xff0c;那么二者之间的关系是什么&#x…

初学者入门指南:什么是网络拓扑结构?

初学者入门指南&#xff1a;什么是网络拓扑结构&#xff1f; 在构建或学习计算机网络时&#xff0c;一个绕不开的核心概念便是“网络拓扑结构”&#xff08;Network Topology&#xff09;。它决定了网络中各个设备如何连接、通信以及如何扩展。理解网络拓扑不仅有助于我们更清…

Satori:元动作 + 内建搜索机制,让大模型实现超级推理能力

Satori&#xff1a;元动作 内建搜索机制&#xff0c;让大模型实现超级推理能力 论文大纲一、背景&#xff1a;LLM 推理增强的三类方法1. 基于大规模监督微调&#xff08;SFT&#xff09;的推理增强2. 借助外部机制在推理时进行搜索 (RLHF / 多模型 / 工具)3. 现有局限性总结 二…

SDC命令详解:使用all_outputs命令进行查询

相关阅读 SDC命令详解https://blog.csdn.net/weixin_45791458/category_12931432.html all_outputs命令用于创建一个输出端口对象集合&#xff0c;关于设计对象和集合的更详细介绍&#xff0c;可以参考下面的博客。 Synopsys&#xff1a;设计对象https://chenzhang.blog.csdn…

printf调试时候正常,运行时打印不出来

问题是在添加了 printf 功能后&#xff0c;程序独立运行时无法正常打印输出&#xff0c;而调试模式下正常。这表明问题可能与 printf 的重定向实现、标准库配置、或编译器相关设置有关。 解决&#xff1a; 原来是使用 Keil/IAR&#xff0c;printf可能需要启用 MicroLIB 或正确…

解决 TimeoutError: [WinError 10060] 在 FramePack项目中连接 Hugging Face 超时的问题

#工作记录 以下是针对 TimeoutError: [WinError 10060] 的完整排查方案&#xff0c;适用于 FramePack项目中。 &#xff08;一般该错误的发生原因请重点排查Hugging Face模型仓库受限需要登录的情形&#xff09; FramePack项目参考资料 FramePack部署&#xff08;从PyCharm解…

分布式-Redis分布式锁

Redis实现分布式锁优点 &#xff08;1&#xff09;Redis有很高的性能&#xff1b; &#xff08;2&#xff09;Redis命令对此支持较好&#xff0c;实现起来比较方便 实现思路 &#xff08;1&#xff09;获取锁的时候&#xff0c;使用setnx加锁&#xff0c;并使用expire命令为锁…

UniRepLknet助力YOLOv8:高效特征提取与目标检测性能优化

文章目录 一、引言二、UniRepLknet 的框架原理&#xff08;一&#xff09;架构概述&#xff08;二&#xff09;架构优势 三、UniRepLknet 在 YOLOv8 中的集成&#xff08;一&#xff09;集成方法&#xff08;二&#xff09;代码实例 四、实验与对比&#xff08;一&#xff09;对…

自研时序大模型讲解(4月29日)直播回顾

4 月 29 日&#xff0c;清华团队揭秘&#xff1a;时序大模型如何让数据“活”起来线上直播圆满结束。清华大学软件学院博士生&#xff0c;IoTDB 原生机器学习引擎 AINode 研发同学刘雍在线上面向数千人次的时序数据分析人员与 AI 大模型行业关注者&#xff0c;就时序大模型的发…

k8s之ingress解释以及k8s创建业务的流程定义

matchLabels ingress Ingress 是反向代理规则&#xff0c;用来规定 HTTP/S 请求应该被转发到哪个 Service 上&#xff0c;比如根据请求中不同的 Host 和 url 路径让请求落到不同的 Service 上。 Ingress Controller 就是一个反向代理程序&#xff0c;它负责解析 Ingress 的反向…

从0开始学习大模型--Day05--理解prompt工程

提示词工程原理 N-gram&#xff1a;通过统计&#xff0c;计算N个词共同出现的概率&#xff0c;从而预测下一个词是什么。 深度学习模型&#xff1a;有多层神经网络组成&#xff0c;可以自动从数据中学习特征&#xff0c;让模型通过不断地自我学习不断成长&#xff0c;直到模型…