没有找到合适的产品?
联系客服协助选型:023-68661681
提供3000多款全球软件/控件产品
针对软件研发的各个阶段提供专业培训与技术咨询
根据客户需求提供定制化的软件开发服务
全球知名设计软件,显著提升设计质量
打造以经营为中心,实现生产过程透明化管理
帮助企业合理产能分配,提高资源利用率
快速打造数字化生产线,实现全流程追溯
生产过程精准追溯,满足企业合规要求
以六西格玛为理论基础,实现产品质量全数字化管理
通过大屏电子看板,实现车间透明化管理
对设备进行全生命周期管理,提高设备综合利用率
实现设备数据的实时采集与监控
利用数字化技术提升油气勘探的效率和成功率
钻井计划优化、实时监控和风险评估
提供业务洞察与决策支持实现数据驱动决策
转帖|使用教程|编辑:龚雪|2024-03-29 10:27:34.863|阅读 25 次
概述:本文主要介绍使用Winform开发自定义用户控件以及实现相关自定义事件的处理,欢迎下载最新版工具体验!
# 界面/图表报表/文档/IDE等千款热门软控件火热销售中 >>
相关链接:
在我们一些非标的用户界面中,往往需要自定义用户控件界面,从而实现不同的内容展示和处理规则,本文介绍使用Winform开发自定义用户控件,以及实现相关自定义事件的处理。
PS:给大家推荐一个C#开发可以用到的界面组件——DevExpress WinForms,它能完美构建流畅、美观且易于使用的应用程序,无论是Office风格的界面,还是分析处理大批量的业务数据,它都能轻松胜任!
DevExpress技术交流群9:909157416 欢迎一起进群讨论
对于比较规范的界面,需要进行一定的分析,以便从中找到对应的规则,逐步细化为自定义用户控件的方式,例如对于由下面多个集合组成的界面内容。
我们截取其中之一,也就是由这些内容多个组合而成,集合可以通过布局TableLayoutPanel(表格布局)或者FlowLayoutPanel(顺序流布局)来添加即可。
而其中之一的内容,不同的颜色方格又可以定义为一个用户控件,因此最终有多个小方格组成的用户控件的。
而单个用户控件,可能承载不同的内容,我们可以定义更多的接口属性以及一些事件来处理相关的逻辑。
甚至还可以在一个单元格里面放置更多的内容,如放置一些特殊的标签来展示信息。
为了使用户控件更加规范化,我们可以定义一个接口,声明相关的属性和处理方法,如下代码所示。
/// <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。
我们相当于把前面的自定义控件,组合为一个新的用户控件,形成一个相对完整的部分,这里提供两种思路,一种是使用常规的用户控件,拖动已有的用户控件组合而成,如下所示。
另一种是利用TableLayoutPanel,动态添加控件进行组合,可以根据预设的TableLayout布局实现控件的顺序添加。
表格的行列定义如下所示:
两种方式都可以实现类似的效果,我们这里以第一种为例实现。
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); } }
最终展示效果如下所示,黄色强调的处理,是选中相同号码的处理事件结果绘制。
本文转载自:
本站文章除注明转载外,均为本站原创或翻译。欢迎任何形式的转载,但请务必注明出处、不得修改原文相关链接,如果存在内容上的异议请邮件反馈至chenjj@wqylolg.cn
文章转载自:无论是零售、物流还是医疗保健,旋转条形码图像的功能都能增强不同应用的灵活性和适应性。使用Aspose.BarCode for Java,您可以轻松旋转条形码图像,确保它们无缝融入应用程序的设计和布局。
借助Aspose.Slides for Java,开发人员可以轻松编辑 PowerPoint 幻灯片(包括表格),以增强演示文稿的效果。
VMProtect 是保护程序代码免遭分析与破解的利器,但很多开发者在实现注册机制时犯了关键性错误,使得再强大的加壳工具也难以阻挡黑客破解。本文将从注册逻辑设计、密钥验证方式、注册状态存储等多个角度,系统拆解常见误区,并结合 VMProtect 的虚拟化和加密策略,提供构建高强度注册保护的实战方案。
在本文中,我们将探讨如何在FastReport .NET中配置与 Apache Ignite 的连接。您将学习通过代码和报表设计器连接插件的必要步骤。
优秀的界面控件开发包,帮助企业构建卓越应用!
DevExpress DXperience Subscription高性价比的企业级.NET用户界面套包,助力企业创建卓越应用!
DevExpress WinForms Subscription为Windows Forms平台创建具有影响力的业务解决方案,高性价比WinForms界面控件套包。
服务电话
重庆/ 023-68661681
华东/ 13452821722
华南/ 18100878085
华北/ 17347785263
客户支持
技术支持咨询服务
服务热线:400-700-1020
邮箱:sales@wqylolg.cn
关注我们
地址 : 重庆市九龙坡区火炬大道69号6幢