C#快速自适应布局
一种自适应布局的方法,它可以根据界面比例自动缩放内容控件,在较短时间内完成软件布局。此方法的优点是:简单易学,布局速度快,适用于绝大多数软件界面。缺点是:不太适合需要高度定制的复杂软件界面。实现的原理是:会将from装进Panel里面对控件进行自动计算大小,每次界面变化时,修改每个控件的尺寸。
创建AutoWindowsSize.cs类(复制以下代码即可)
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
//using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Drawing.Drawing2D;
//using System.Threading.Tasks;
namespace AutoWindowsSize
{
class AutoAdaptWindowsSize //窗口缩放对象
{
double formOriginalWidth;//窗体高度原始宽度
double formOriginalHeight;//窗体原始
double scaleX;//水平缩放比例
double scaleY;//垂直缩放比例
Dictionary<string, string> ControlsInfo = new Dictionary<string, string>();//控件中心Left,Top,控件Width,控件Height,控件字体Size
private Form _form;
Panel Win_Panel1 = new Panel();
public AutoAdaptWindowsSize(Form form)
{
_form = form;
//代码生成一个容器panel1,添加至窗体
_form.Controls.Add(Win_Panel1);
Win_Panel1.BorderStyle = BorderStyle.None; //容器border样式
Win_Panel1.Dock = DockStyle.Fill; //设置填充,下面添加控件至容器完成后,容器会填充窗口
Win_Panel1.BackColor = Color.Transparent; // 这里默认的背景颜色是form的背景颜色,如果form页面时图片,需要将这里的颜色设置成透明,否则会被覆盖。
//将窗体所有控件添加至panel1
while (_form.Controls[0].Name.Trim() != "")
{
foreach (Control item in _form.Controls)
{
if (item.Name.Trim() != "" && item.Name.Trim() != Win_Panel1.Name.Trim())
{
Win_Panel1.Controls.Add(item);
}
}
}
//保存窗体和控件初始大小
InitControlsInfo(Win_Panel1);
}
public void InitControlsInfo(Control ctrlContainer)
{
if (ctrlContainer.Parent == _form)//获取窗体的高度和宽度
{
formOriginalWidth = Convert.ToDouble(ctrlContainer.Width);
formOriginalHeight = Convert.ToDouble(ctrlContainer.Height);
}
foreach (Control item in ctrlContainer.Controls)
{
if (item.Name.Trim() != "")
{
//添加信息:键值:控件名,内容:据左边距离,距顶部距离,控件宽度,控件高度,控件字体。
ControlsInfo.Add(item.Name, (item.Left + item.Width / 2) + "," + (item.Top + item.Height / 2) + "," + item.Width + "," + item.Height + "," + item.Font.Size);
}
if ((item as UserControl) == null && item.Controls.Count > 0)
{
InitControlsInfo(item);
}
}
}
public void FormSizeChanged()
{
try
{
if (ControlsInfo.Count > 0)//如果字典中有数据,即窗体改变
{
ControlsZoomScale(Win_Panel1);//表示pannel控件
ControlsChange(Win_Panel1);
}
}
catch { }
}
private void ControlsZoomScale(Control ctrlContainer)
{
scaleX = (Convert.ToDouble(ctrlContainer.Width) / formOriginalWidth);
scaleY = (Convert.ToDouble(ctrlContainer.Height) / formOriginalHeight);
}
// 改变控件大小
private void ControlsChange(Control ctrlContainer)
{
double[] pos = new double[5];//pos数组保存当前控件中心Left,Top,控件Width,控件Height,控件字体Size
foreach (Control item in ctrlContainer.Controls)//遍历控件
{
if (item.Name.Trim() != "")//如果控件名不是空,则执行
{
if ((item as UserControl) == null && item.Controls.Count > 0)//如果不是自定义控件
{
ControlsChange(item);//循环执行
}
string[] strs = ControlsInfo[item.Name].Split(',');//从字典中查出的数据,以‘,’分割成字符串组
for (int i = 0; i < 5; i++)
{
pos[i] = Convert.ToDouble(strs[i]);//添加到临时数组
}
double itemWidth = pos[2] * scaleX; //计算控件宽度,double类型
double itemHeight = pos[3] * scaleY; //计算控件高度
item.Left = Convert.ToInt32(pos[0] * scaleX - itemWidth / 2);//计算控件距离左边距离
item.Top = Convert.ToInt32(pos[1] * scaleY - itemHeight / 2);//计算控件距离顶部距离
item.Width = Convert.ToInt32(itemWidth);//控件宽度,int类型
item.Height = Convert.ToInt32(itemHeight);//控件高度
if (float.Parse((pos[4] * Math.Min(scaleX, scaleY)).ToString()) != 0) //缩放字体大小不能为0
{ item.Font = new Font(item.Font.Name, float.Parse((pos[4] * Math.Min(scaleX, scaleY)).ToString())); } //字体
}
}
}
}
}
2.创建结束以后,在Form1引入using AutoWindowsSize;
using AutoWindowsSize;
3.在Form1中设置全局变量AutoSize。
AutoAdaptWindowsSize AutoSize;
4.在Form1_Load添加如下代码,在界面初始化的过程中创建AutoSize。
private void Form1_Load(object sender, EventArgs e)
{
AutoSize = new AutoAdaptWindowsSize(this);
}
5.创建Form1_SizeChanged事件函数,一定加这个AutoSize != null判断,电脑缩放布局不是100%的时候,会报错。代码如下:
private void Form1_SizeChanged(object sender, EventArgs e)
{
if (AutoSize != null) // 一定加这个判断,电脑缩放布局不是100%的时候,会报错
{
AutoSize.FormSizeChanged();
}
}
以上部分就实现了,自适应布局的代码步骤。但在使用过程中,可能会有以下问题,可更具需要进行修改:
1.Form页面如果是背景图片,页面更新特别快的时候(比如有图片更新),页面会有闪烁情况,如果纯色不会有此类情况,暂时没有解决。
2.如果Form中的背景是图片,会直接显示成纯图的背景,原因是在AutoAdaptWindowsSize类中会创建一个panel,panel背景颜色会自动设置,而pannel的纯色背景会覆盖掉Form1的背景,本质上是panel的背景。所以可以给Panel设置背景图片或将panel背景色修改为Transparent,如下:
Win_Panel1.BackColor = Color.Transparent;
我已经将类里面的内容做了修改,可以直接忽略。
3.其次自适应布局,每次修改窗口需要对界面重新绘制,在拖动窗口时会出现比较严重的闪屏现象,需要在Form1的Disiger中添加任意位置添加如下代码:
// 引入Forms。
using System.Windows.Forms;
protected override CreateParams CreateParams
{
get
{
CreateParams cp = base.CreateParams;
cp.ExStyle |= 0x02000000;
return cp;
}
}
以上就是自适应布局的内容,经过测试可以满足我们绝大部分的界面自适应需求。
引用出处:作者:666号特派员 https://www.bilibili.com/read/cv23844020 出处:bilibili
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
2021-08-18 计算机网络-4-10-IPv6