winform 控件随页面大小进行自适应

这个功能网上很多人在问,也有不少人给出过答案,经过实际使用,觉得网上这段代码实现的效果比较好,记录一下

核心代码就是下面这个类

  1 using System;
  2 using System.Collections.Generic;
  3 using System.Linq;
  4 using System.Text;
  5 using System.Windows.Forms;
  6
  7 namespace UAVRadar
  8 {
  9     public class AutoSizeFormClass
 10     {
 11         //(1).声明结构,只记录窗体和其控件的初始位置和大小。
 12         public struct controlRect
 13         {
 14             public int Left;
 15             public int Top;
 16             public int Width;
 17             public int Height;
 18         }
 19         //(2).声明 1个对象
 20         //注意这里不能使用控件列表记录 List nCtrl;,因为控件的关联性,记录的始终是当前的大小。
 21         //      public List oldCtrl= new List();//这里将西文的大于小于号都过滤掉了,只能改为中文的,使用中要改回西文
 22         public List<controlRect> oldCtrl = new List<controlRect>();
 23         int ctrlNo = 0;//1;
 24         //(3). 创建两个函数
 25         //(3.1)记录窗体和其控件的初始位置和大小,
 26         public void controllInitializeSize(Control mForm)
 27         {
 28             controlRect cR;
 29             cR.Left = mForm.Left; cR.Top = mForm.Top; cR.Width = mForm.Width; cR.Height = mForm.Height;
 30             oldCtrl.Add(cR);//第一个为"窗体本身",只加入一次即可
 31             AddControl(mForm);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 32             //this.WindowState = (System.Windows.Forms.FormWindowState)(2);//记录完控件的初始位置和大小后,再最大化
 33             //0 - Normalize , 1 - Minimize,2- Maximize
 34         }
 35         private void AddControl(Control ctl)
 36         {
 37             foreach (Control c in ctl.Controls)
 38             {  //**放在这里,是先记录控件的子控件,后记录控件本身
 39                 //if (c.Controls.Count > 0)
 40                 //    AddControl(c);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 41                 controlRect objCtrl;
 42                 objCtrl.Left = c.Left; objCtrl.Top = c.Top; objCtrl.Width = c.Width; objCtrl.Height = c.Height;
 43                 oldCtrl.Add(objCtrl);
 44                 //**放在这里,是先记录控件本身,后记录控件的子控件
 45                 if (c.Controls.Count > 0)
 46                     AddControl(c);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 47             }
 48         }
 49         //(3.2)控件自适应大小,
 50         public void controlAutoSize(Control mForm)
 51         {
 52             if (ctrlNo == 0)
 53             { //*如果在窗体的Form1_Load中,记录控件原始的大小和位置,正常没有问题,但要加入皮肤就会出现问题,因为有些控件如dataGridView的的子控件还没有完成,个数少
 54                 //*要在窗体的Form1_SizeChanged中,第一次改变大小时,记录控件原始的大小和位置,这里所有控件的子控件都已经形成
 55                 controlRect cR;
 56                 //  cR.Left = mForm.Left; cR.Top = mForm.Top; cR.Width = mForm.Width; cR.Height = mForm.Height;
 57                 cR.Left = 0; cR.Top = 0; cR.Width = mForm.PreferredSize.Width; cR.Height = mForm.PreferredSize.Height;
 58
 59                 oldCtrl.Add(cR);//第一个为"窗体本身",只加入一次即可
 60                 AddControl(mForm);//窗体内其余控件可能嵌套其它控件(比如panel),故单独抽出以便递归调用
 61             }
 62             float wScale = (float)mForm.Width / (float)oldCtrl[0].Width;//新旧窗体之间的比例,与最早的旧窗体
 63             float hScale = (float)mForm.Height / (float)oldCtrl[0].Height;//.Height;
 64             ctrlNo = 1;//进入=1,第0个为窗体本身,窗体内的控件,从序号1开始
 65             AutoScaleControl(mForm, wScale, hScale);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 66         }
 67         private void AutoScaleControl(Control ctl, float wScale, float hScale)
 68         {
 69             int ctrLeft0, ctrTop0, ctrWidth0, ctrHeight0;
 70             //int ctrlNo = 1;//第1个是窗体自身的 Left,Top,Width,Height,所以窗体控件从ctrlNo=1开始
 71             foreach (Control c in ctl.Controls)
 72             { //**放在这里,是先缩放控件的子控件,后缩放控件本身
 73                 //if (c.Controls.Count > 0)
 74                 //   AutoScaleControl(c, wScale, hScale);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 75                 ctrLeft0 = oldCtrl[ctrlNo].Left;
 76                 ctrTop0 = oldCtrl[ctrlNo].Top;
 77                 ctrWidth0 = oldCtrl[ctrlNo].Width;
 78                 ctrHeight0 = oldCtrl[ctrlNo].Height;
 79                 //c.Left = (int)((ctrLeft0 - wLeft0) * wScale) + wLeft1;//新旧控件之间的线性比例
 80                 //c.Top = (int)((ctrTop0 - wTop0) * h) + wTop1;
 81                 c.Left = (int)((ctrLeft0) * wScale);//新旧控件之间的线性比例。控件位置只相对于窗体,所以不能加 + wLeft1
 82                 c.Top = (int)((ctrTop0) * hScale);//
 83                 c.Width = (int)(ctrWidth0 * wScale);//只与最初的大小相关,所以不能与现在的宽度相乘 (int)(c.Width * w);
 84                 c.Height = (int)(ctrHeight0 * hScale);//
 85                 ctrlNo++;//累加序号
 86                 //**放在这里,是先缩放控件本身,后缩放控件的子控件
 87                 if (c.Controls.Count > 0)
 88                     AutoScaleControl(c, wScale, hScale);//窗体内其余控件还可能嵌套控件(比如panel),要单独抽出,因为要递归调用
 89
 90                 if (ctl is DataGridView)
 91                 {
 92                     DataGridView dgv = ctl as DataGridView;
 93                     Cursor.Current = Cursors.WaitCursor;
 94
 95                     int widths = 0;
 96                     for (int i = 0; i < dgv.Columns.Count; i++)
 97                     {
 98                         dgv.AutoResizeColumn(i, DataGridViewAutoSizeColumnMode.AllCells);  // 自动调整列宽
 99                         widths += dgv.Columns[i].Width;   // 计算调整列后单元列的宽度和
100                     }
101                     if (widths >= ctl.Size.Width)  // 如果调整列的宽度大于设定列宽
102                         dgv.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.DisplayedCells;  // 调整列的模式 自动
103                     else
104                         dgv.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.Fill;  // 如果小于 则填充
105
106                     Cursor.Current = Cursors.Default;
107                 }
108             }
109
110
111         }
112     }
113 }

界面调用的代码如下:

 1 private AutoSizeFormClass asc = new AutoSizeFormClass();
 2 public MainWindow()
 3 {
 4     InitializeComponent();
 5     asc.controllInitializeSize(this);
 6 }
 7
 8 private void MainWindow_Resize(object sender, EventArgs e)
 9 {
10     //int length = this.Width;
11     //int high = this.Height;
12     asc.controlAutoSize(this);
13 }

原文地址:https://www.cnblogs.com/tlduck/p/9087250.html

时间: 2024-10-20 03:29:42

winform 控件随页面大小进行自适应的相关文章

C# Winform控件字体大小自适应

using System.Collections.Generic; using System.Drawing; using System.Windows.Forms; namespace WGClient { class AutoSizeFormClass { //(1).声明结构,只记录窗体和其控件的初始位置和大小. public struct controlRect { public int Left; public int Top; public int Width; public int

C# ——窗体和控件随着分辨率的变化自适应大小

一.说明 我们自己编写程序的界面,会遇到各种屏幕分辨 率,只有自适应才能显的美观.实际上,做到这点也很简单,就是首先记录窗体和它上面控件的初始位置和大小,当窗体改变比例时,其控件的位置和大小也按此比 例变化即可.因为窗体上控件的位置和大小是相对于自己所在的窗体的,也就是所谓的窗口坐标. 在这里我们只考虑相对于自己窗体的窗口坐标更简单,也就是成比例变化.为了多个窗体共用,我在这里创建一个类AutoSizeFormClass ,1.使用它去记录窗体和其控件的初始位置和大小,2.根据窗体变化了的大小,

Winform控件Tag使用规范

背景 Tag在WinForm控件中经常被用来存储临时数据,类型为object,但是当程序中多个地方使用到Tag时,容易造成Tag使用的混乱,Tag是如此重要的一个属性,应该要好好考虑下如何有效的使用Tag服务于程序开发. 借鉴Web开发时,Js可以通过自定义属性,将状态绑定到Dom节点的自定义Attribute上,这也是绝大多数js插件,附加状态数据的方式.如果把字典引入到Tag中,规范约定Tag就是一个键值对组成的字典,那么对状态的存取就不会局限于一个状态,而且通过有效的管理可以更大化的发挥T

.Net WinForm 控件键盘消息处理剖析

在WinForm控件上我们可以看到很多关于键盘消息处理的方法,比如OnKeyDown, OnKeyPress, ProcessCmdKey, ProcessDialogKey,IsInputKey等等,那么这些方法是如何被组织的,每一个方法的具体含义又是什么哪?Win32的键盘消息又是如何到达控件上的这些方法的,本文将着重阐述这些问题,对.Net WinForm控件的键盘消息处理过程进行剖析.  1.      WinForm消息循环 大家都知道WinForm也是依赖于底层的消息机制的,通常我们

用户控件与页面间相互给各自的控件赋值

用户控件 ->页面 ((Label)this.Parent.Page.FindControl("AAA")).Text = "ABC"; AAA:页面控件ID Label:页面控件类型 页面 -> 用户控件 ((HiddenField)POPUSER_1.FindControl("hidNO")).Value = "VNBB"; POPUSER_1:用户控件ID HiddenField:用户控件中需要处理的控件的类

通过WinForm控件创建的WPF控件无法输入的问题

今天把写的一个WPF程序发布到别的机器上执行,发现一个比较奇怪的问题:在那个机器上用英文输入法无法输入数字,非要切换到中文输入法才行:但在我的机器上却是好好的. 最开始以为是输入法的问题,弄了好一阵子后,终于找到了原因:虽然这个程序是个WPF程序,但为了复用之前的部分代码,使用着一个WinForm的菜单控件,后续的子窗口都是通过这个WinForm菜单创建的.而用WinForm控件创建的WPF控件可能出现无法正确响应键盘事件的情况. 找到了原因后,一个常规的解决方法是:将WinForm控件换成WP

优化laydate日期选取控件位于页面最右侧时会撑开页面的问题

laydate控件是我到目前以来用过最顺心的控件,虽然该控件已经非常优秀了,但在使用中还是会遇到点问题. 如果该控件位于页面的最右侧,点击后会出现如下问题. 可以看到点击后页面被撑大了. 解决方案: 修改laydate.js文件,先找到orien方法的定义,修改为: c.orien = function (a, b) { var d, e = c.elem.getBoundingClientRect(); var win_left = e.left + (b ? 0 : c.scroll(1))

C#中WinForm控件的跨线程更新Invoke

目的: 用WinForm(C#)搭建一个用户界面,一个进度条和一个按钮,按钮启动进度条,进度完成时停止更新 示例: 实现: 在按钮事件中设置循环,更新进度条         private void btnProgress_Click(object sender, EventArgs e)         {             for (int ii = 0; ii < 100; ii++)             {                 progressBar1.Value 

Wpf使用Winform控件后Wpf元素被Winform控件遮盖问题的解决

有人会说不建议Wpf中使用Winform控件,有人会说建议使用Winform控件在Wpf下的替代方案,然而在实际工作中由于项目的特殊需求,考虑到时间.成本等因素,往往难免会碰到在WPF中使用Winfrom控件的问题,我们知道Wpf可以通过使用WindowsFormsHost容器调用Winform控件,但是在一些场合需要将Wpf元素显示在Winform控件的上层,此时就会出现Wpf元素被Winform控件遮盖的问题. 一.场景再现 接到公司命令,在时间紧迫的情况下,需要将原来的Winform程序(