使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

这篇具有很好参考价值的文章主要介绍了使用Winform开发自定义用户控件,以及实现相关自定义事件的处理。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

在我们一些非标的用户界面中,我们往往需要自定义用户控件界面,从而实现不同的内容展示和处理规则,本篇内容介绍使用Winform开发自定义用户控件,以及实现相关自定义事件的处理。

1、用户控件的界面分析

对于比较规范的界面,需要进行一定的分析,以便从中找到对应的规则,逐步细化为自定义用户控件的方式。例如对于由下面多个集合组成的界面内容。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

 我们截取其中之一,也就是由这些内容多个组合而成,集合可以通过布局TableLayoutPanel(表格布局)或者FlowLayoutPanel(顺序流布局)来添加即可。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

 而其中之一的内容,不同的颜色方格又可以定义为一个用户控件,因此最终有多个小方格组成的用户控件的。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

而单个用户控件,可能承载不同的内容,我们可以定义更多的接口属性以及一些事件来处理相关的逻辑。

甚至,我能还可以在一个单元格里面放置更多的内容,如放置一些特殊的标签来展示信息。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

2、自定义用户控件的接口和实现

为了使用户控件更加规范化,我们可以定义一个接口,声明相关的属性和处理方法,如下代码所示。

    /// <summary>
    /// 自定义控件的接口
    /// </summary>
    public interface INumber
    {
        /// <summary>
        /// 数字
        /// </summary>
        string Number { get; set; }

        /// <summary>
        /// 数值颜色
        /// </summary>
        Color Color { get; set; }

        /// <summary>
        /// 显示文本
        /// </summary>
        string Animal { get; set; }

        /// <summary>
        /// 显示文本
        /// </summary>
        string WuHan { get; set; }

        /// <summary>
        /// 设置选中的内容的处理
        /// </summary>
        /// <param name="data">事件数据</param>
        void SetSelected(ClickEventData data);
    }

然后我们创建一个用户控件,并命名为NumberItem,并使它继承前面定义的接口 INumber ,实现相关的属性和事件,如下代码所示。

    /// <summary>
    /// 自定义用户控件
    /// </summary>
    public partial class NumberItem : UserControl, INumber
    {
        /// <summary>
        /// 数字
        /// </summary>
        public string Number { get; set; }

        /// <summary>
        /// 颜色
        /// </summary>
        public Color Color { get; set; }

        /// <summary>
        /// 显示文本
        /// </summary>
        public string Animal { get; set; }

        /// <summary>
        /// 显示文本
        /// </summary>
        public string WuHan { get; set; }

其中处理方法SetSelected先保留为空,后面继续完善。

        /// <summary>
        /// 设置选中的数值
        /// </summary>
        /// <param name="data">传递的数据</param>
        public void SetSelected(ClickEventData data)
        {

        }

由于自定义控件,我们需要跟踪用户的单击处理,并且需要把这个逻辑逐步推动到顶级界面上去进行处理,因此需要定义一个事件信息,如下所示。

        /// <summary>
        /// 事件处理
        /// </summary>
        public EventHandler<ClickEventData> ClickEventHandler { get; set; }

其中ClickEventData是我们定义的一个数据,用来承载用户单击的类型和值内容的信息结构,如下代码所示。

   /// <summary>
   /// 对自定义控件触发的事件信息
   /// </summary>
   public class ClickEventData
   {
       /// <summary>
       /// 事件触发类型
       /// </summary>
       public ClickEventType ClickEventType { get; set; } = ClickEventType.Number;

       /// <summary>
       /// 传递值
       /// </summary>
       public string Value { get; set; }

       public ClickEventData()
       {
       }

       /// <summary>
       /// 参数化构造
       /// </summary>
       /// <param name="clickEventType">事件触发类型</param>
       /// <param name="value">传递值</param>
       public ClickEventData(ClickEventType clickEventType, string value)
       {
           ClickEventType = clickEventType;
           Value = value;
       }
   }

再创建一个整合多个号码数值的一个自定义控件,它也是一个完整的单元之一,我们命名为 LotteryItemControl2。

我们相当于把前面的自定义控件,组合为一个新的用户控件,形成一个相对完整的部分,这里提供两种思路,一种是使用常规的用户控件,拖动已有的用户控件组合而成,如下所示。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

 另一种是利用TableLayoutPanel,动态添加控件进行组合,可以根据预设的TableLayout布局实现控件的顺序添加。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

表格的行列定义如下所示

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

两种方式都可以实现类似的效果,我们这里以第一种为例实现。

    public partial class LotteryItemControl2 : UserControl
    {
        /// <summary>
        /// 事件处理
        /// </summary>
        public EventHandler<ClickEventData> ClickEventHandler { get; set; }

        /// <summary>
        /// 第几期
        /// </summary>
        public string Qi { get; set; }

        /// <summary>
        /// 数据列表
        /// </summary>
        public List<string> NumberList { get; set; }

数据列表就是展示在自定义控件的数字。在控件中定义一个函数 统一处理数据内容的绑定显示。

        /// <summary>
        /// 绑定数据
        /// </summary>
        public void BindData()
        {
            //控件列表,方便统一处理
            var controlList = new List<NumberItem>
            {
                this.numberItem1, this.numberItem2, this.numberItem3, this.numberItem4, 
                this.numberItem5, this.numberItem6, this.numberItem7
            };

            this.labelQi.Text = Qi; //设置第几期
            for(int i =0; i < this.NumberList.Count; i++)
            {
                var control = controlList[i];
                var number = this.NumberList[i];

                var shenxiao = LotteryToolHelper.NumberToShenXiaoDict[number]; //"马";
                var wuhan = LotteryToolHelper.NumberToWuhanDict[number];//"土"

                control.Number = number;
                control.Animal = shenxiao;
                control.WuHan = wuhan;

                var colorStr = LotteryToolHelper.ColorBall[number];
                control.Color = LotteryToolHelper.GetColor(colorStr);  //item % 2 == 0 ? Color.Red : Color.Green;

                control.BindData();
                control.ClickEventHandler += (s, data) =>
                {
                    if (ClickEventHandler != null)
                    {
                        //传递父控件统一处理
                        ClickEventHandler(s, data);
                    }
                };
            }
        }

其中该控件也可以设置选中,有具体的子控件调用设置选中的处理规则即可。

        /// <summary>
        /// 遍历控件,设置选中的数值
        /// </summary>
        /// <param name="data">传递信息</param>
        public void SetSelected(ClickEventData data)
        {
            foreach (var control in this.Controls)
            {
                if (control is NumberItem item)
                {
                    item.SetSelected(data);
                }
            }
        }

为了提高性能,我们一般往往需要设置窗体或者Panel为双缓冲DoubleBuffered = true。

在主界面的面板中,我们可以添加一个FlowLayoutPanel 来按顺序堆叠用户控件,具体的实现逻辑就是根据从数据库获得的记录进行展示即可。

    var controlList = new List<LotteryItemControl2>();
    foreach (var info in list)
    {
        var control = new LotteryItemControl2();

        control.Qi = info.LineNo.ToString("D2");
        var numberList = new List<string>()
        {
            info.No1.ToString("D2"),
            info.No2.ToString("D2"),
            info.No3.ToString("D2"),
            info.No4.ToString("D2"),
            info.No5.ToString("D2"),
            info.No6.ToString("D2"),
            info.No7.ToString("D2"),
        };
        control.NumberList = numberList;
        control.BindData();

        control.ClickEventHandler += (s, data) =>
        {
            //遍历所有的控件统一处理样式
            foreach (var subCtrl in panel.Controls)
            {
                if (subCtrl is LotteryItemControl2 lottery)
                {
                    lottery.SetSelected(data);
                }
            }
        };
        controlList.Add(control);
    }
    this.panel.Controls.AddRange(controlList.ToArray());

以上就是相关的处理逻辑,用来组织自定义用户控件的统一展示处理。

如果需要用户进行不同条件的数据展示,那么展示前,就需要重新清空面板中的控件,如下所示。

            //清空界面
            while (panel.Controls.Count > 0)
            {
                var controltoremove = panel.Controls[0];
                panel.Controls.RemoveAt(0);
                controltoremove.Dispose();
            }
            panel.Controls.Clear();

上面代码记得调用Dispose方法来释放控件资源。

在最小的自定义控件中,我们可能需要根据一些条件进行一些自定义绘制处理,以突出显示不同的内容(重点强调选中项目)。

private void NumberItem_Paint(object sender, PaintEventArgs e)

如下是一些特殊的绘制处理内容。

        private void NumberItem_Paint(object sender, PaintEventArgs e)
        {
            this.BackColor = (this.BorderStyle == BorderStyle.FixedSingle) ? Color.Yellow : Color.Transparent;
            if (this.BorderStyle == BorderStyle.FixedSingle)
            {
                IntPtr hDC = GetWindowDC(this.Handle);
                Graphics g = Graphics.FromHdc(hDC);
                ControlPaint.DrawBorder(
                g,
                new Rectangle(0, 0, this.Width, this.Height),
                _borderColor,
                _borderWidth,
                ButtonBorderStyle.Solid,
                _borderColor,
                _borderWidth,
                ButtonBorderStyle.Solid,
                _borderColor,
                _borderWidth,
                ButtonBorderStyle.Solid,
                _borderColor,
                _borderWidth,
                ButtonBorderStyle.Solid);
                g.Dispose();
                ReleaseDC(Handle, hDC);
            }
        }

最终展示效果如下所示,黄色强调的处理,是选中相同号码的处理事件结果绘制。

使用Winform开发自定义用户控件,以及实现相关自定义事件的处理

 文章来源地址https://www.toymoban.com/news/detail-778671.html

注:本篇随笔借鉴一些特殊的界面来介绍自定义用户控件的处理经验,无其他不良引导,请关注技术本身的分析和处理。

 

到了这里,关于使用Winform开发自定义用户控件,以及实现相关自定义事件的处理的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Winform自定义控件 —— 指示灯

             在开始阅读本文之前,如果您有学习创建自定义控件库并在其他项目中引用的需求,请参考: 在Visual Studio中创建自定义Winform控件库并在其他解决方案中引用 https://blog.csdn.net/YMGogre/article/details/126508042 目录 1、应用场景:  1.1、本文的应用场景:  2、所需资源: 

    2024年02月04日
    浏览(38)
  • Winform自定义控件 —— 水印文本框

             在开始阅读本文之前,如果您有学习创建自定义控件库并在其他项目中引用的需求,请参考: 在Visual Studio中创建自定义Winform控件库并在其他解决方案中引用 https://blog.csdn.net/YMGogre/article/details/126508042 目录 1、应用场景:  2、所需资源:  3、源代码:  4、使用方

    2024年02月02日
    浏览(45)
  • winform 封装unity web player 用户控件

    VS2015 Unity 5.3.6f1 (64-bit) Unity官方提供的UnityWebPlayer控件在嵌入Winform时要求读取的.unity3d文件路径(Src)必须是绝对路径 ,如果移动代码到另一台电脑,需要重新修改src。于是考虑使用winform用户控件来进行封装,以实现读取存放在工程文件夹下的.unity3d文件的效果。 WinForm内嵌

    2024年02月12日
    浏览(30)
  • C#winform自定义Trackbar控件实例

    本篇讲解自定义Trackbar控件实例,我们项目开发中为了保持项目界面的风格的一致性,需要自定义Trackbar控件,系统自带的控件样式不能更改,无法满足需求。 先看效果     实现过程 创建winform项目,添加组件类控件     修改名称,点击添加控件 添加两个自定义类  

    2024年02月12日
    浏览(32)
  • 【Winform学习笔记(五)】引用自定义控件库(dll文件)

    在本文中主要介绍 如何引用自定义控件库(dll文件)。 通过生成解决方案 或 重新生成解决方案 生成 dll 文件 生成的 dll 文件在 bin 文件夹下的 Debug 文件夹中 点击工具-选择工具箱项 点击浏览 找到刚刚生成的 dll 文件 点击命名空间,确认要导入的控件 点击确定 这时在引用中是

    2024年02月14日
    浏览(32)
  • 在Winform分页控件中集成保存用户列表显示字段及宽度调整设置

    在Winform的分页控件里面,我们提供了很多丰富的功能,如常规分页,中文转义、导出Excel、导出PDF等,基于DevExpress的样式的分页控件,我们在其上面做了不少封装,以便更好的使用,其中就包括集成保存用户列表显示字段及宽度调整设置。本篇随笔介绍这个实现的过程,通过

    2024年02月03日
    浏览(31)
  • 循序渐进介绍基于CommunityToolkit.Mvvm 和HandyControl的WPF应用端开发(3)--自定义用户控件

    在我们创建界面元素的时候,不管在Vue3+ElementPlus的前端上,还是Winform桌面端上,都是会利用自定义用户控件来快速重用一些自定义的界面内容,对自定义用户控件的封装处理,也是我们开发WPF应用需要熟悉的一环。本篇随笔继续深入介绍介绍基于CommunityToolkit.Mvvm 和HandyCont

    2024年02月09日
    浏览(41)
  • 使用DevExpress22.X(Patch)控件库在VisualStudio2022使用C#进行Winform、WPF应用的开发,看这一篇就够了!

            写在开头,Dev Express是个十分强大的控件库(下文简称Dev),但碍于其高昂的使用费用,“出于学习目的”,我们一般使用的都是Patch版本(在版权意识日趋加强的当下,不要提那两个字,现在加上那些字,百度都搜不出内容)。         最重要的 Patch资源 (包括

    2024年02月09日
    浏览(51)
  • Android开发之自定义控件-组合控件的开发与实现

    最终实现的效果展示图:   类似支付宝微信,底部分隔线对齐标题效果:       完整渲染显示效果(包含三个条目右边不同颜色的文字): 立体效果:  隐藏资产总额条目右边更多箭头  隐藏中国历史条目右边的文字: 隐藏中国历史条目下边的分隔线: 隐藏条目2中国历史左

    2024年02月10日
    浏览(32)
  • Visual Studio C# WinForm开发入门(3):各种控件介绍

    窗口就是打开程序我们所面对的一个面板,里面可以添加各种控件,如下图所示,我们可以在属性栏设置其标题名称、图标、大小等。 双击标题框,会生成Load函数,也可以到事件里面去找 Load函数是窗口生成后需要执行的事件函数。 可以在里面加一些初始化函数。 (1)Na

    2024年02月09日
    浏览(40)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包