百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

Winform下实现Grid布局

xsobi 2024-11-23 10:46 1 浏览

很久没写非常代码化的文章了,代码倒是每天都在写,但是前几天看到一个朋友在用一个“只有一个文本框”的窗体来写“让控件随窗体大小改变而改变”,他用的方法是在窗体大小变化后,重新计算根据窗体的新Size计算这一个控件的Size。所以我给他评论是:单个控件压根不需要这么做,而多个控件这么做又没有意义,所以最好的办法就是参考CSS的Grid布局方法自定义一套基于WIniform的Grid布局。

Winform的控件都有一个Anchor属性。只不过如果使用Anchor的话,对于一个控件确实是可以的,但是多个控件行列布局的时候用Anchor就不行了,放大缩小后完全乱套。比如我布局得好好的一个页面:

用Anchor布局的话,上下左右控件的位置倒是可以随窗口大小变化,但是大小却不会变。但是对于中间的空件,那么布局后很可能就是乱七八糟的了。

所以刚好周末,就花了点时间描述一下我所说的参考CSS的Grid布局方法来实现一个Winform下的Grid布局,思路如下

  1. 模拟CSS的Grid布局,加一个GridArea的类,用于存储grid-row-start / grid-column-start / grid-row-end / grid-column-end(想简单的话用一个Rectangle来替代也行);
  2. 在窗体的初始化方法中,完成控件初始化后(InitializeComponent),计算每个控件所属的GridArea;
  3. 在窗体Resize后,根据前面计算出来的每个控件GridArea,重新计算每个控件的位置,大小。

就是这么简单:

Grid布局后,窗体放大缩小(用了一个录屏软件,希望能审核通过吧)

总的代码就100行,如果有朋友需要,欢迎使用。

public partial class GridLayoutForm : Form
    {
        private static int GRIDSPACE = 2;
        private Dictionary<Control, GridArea> ControlGridArea = new Dictionary<Control, GridArea>();

        public GridLayoutForm()
        {
            InitializeComponent();

            this.InitGridAreas();
        }

        private int GridCols { get; set; }
        private int GridRows { get; set; }

        private GridArea CalcGridArea(Control ctrl)
        {
            int top = ctrl.Top;
            int left = ctrl.Left;
            int width = ctrl.Width;
            int height = ctrl.Height;

            int startRow = top / GRIDSPACE;
            int startCol = left / GRIDSPACE;
            if (startRow <= 0) startRow = 1;
            if (startCol <= 0) startCol = 1;

            int endRow = (top + height) / GRIDSPACE;
            int endCol = (left + width) / GRIDSPACE;
            if ((top + height) % GRIDSPACE > 0)
            {
                endRow = (top + height + GRIDSPACE) / GRIDSPACE;
            }
            if ((left + width) % GRIDSPACE > 0)
            {
                endCol = (left + width + GRIDSPACE) / GRIDSPACE;
            }

            // grid-row-start / grid-column-start / grid-row-end / grid-column-end
            return new GridArea(startRow, startCol, endRow, endCol + 1);
        }

        public void RelocationControls()
        {
            if (this.GridCols > 0 && this.GridRows > 0)
            {
                double spaceW = (double)this.Width / this.GridCols;
                double spaceH = (double)this.Height / this.GridRows;

                foreach (Control child in this.Controls)
                {
                    if (this.ControlGridArea.ContainsKey(child))
                    {
                        GridArea area = this.ControlGridArea[child];

                        int left = (int)Math.Round((area.GridColumnStart) * spaceW, 0);
                        int top = (int)Math.Round((area.GridRowStart) * spaceH, 0);

                        int width = (int)Math.Round((area.GridColumnEnd - 1) * spaceW - left, 0);
                        int height = (int)Math.Round((area.GridRowEnd - 1) * spaceH - top, 0);

                        child.Location = new Point(left, top);
                        child.Size = new Size(width, height);
                    }
                }
            }
        }

        public void InitGridAreas()
        {
            this.GridCols = this.Width / GRIDSPACE;
            this.GridRows = this.Height / GRIDSPACE;

            foreach (Control child in this.Controls)
            {
                this.ControlGridArea[child] = this.CalcGridArea(child);
            }

            this.RelocationControls();
        }

        protected override void OnResize(EventArgs e)
        {
            base.OnResize(e);
            this.RelocationControls();
        }

        private class GridArea
        {
            // grid-row-start / grid-column-start / grid-row-end / grid-column-end
            public GridArea(int rowStart, int columnStart, int rowEnd, int columnEnd)
            {
                this.GridRowStart = rowStart;
                this.GridColumnStart = columnStart;
                this.GridRowEnd = rowEnd;
                this.GridColumnEnd = columnEnd;
            }

            public int GridRowStart { get; set; }
            public int GridColumnStart { get; set; }
            public int GridRowEnd { get; set; }
            public int GridColumnEnd { get; set; }
        }
    }

相关推荐

【互联网那些事】高效开发Android App的10个建议

  假如要GooglePlay上做一个最失败的案例,那最好的秘诀就是界面奇慢无比、耗电、耗内存。接下来就会得到用户的消极评论,最后名声也就臭了。即使你的应用设计精良、创意无限也没用。  耗电或者内存...

手机APP开发方式有哪些? 手机app的开发模式有哪三种?

微信小程序开发定制_软件开发_APP开发_网站制作-优软软件开发...

Android开发入门(一):Android系统简介

Android系统是Google公司在2008年推出的一款智能移动设备操作系统,通过不断地版本迭代,目前已经推出到Android11版本了。Android系统广泛应用在手机、平板、电视等各种电子设...

物联网app开发流程 物联网app开发工具

现在随着科技的发展,很多产品都想用一个手机app去显示他的参数数据或者通过手机app去控制它。但是很多人不知道他的流程。今天我就来说下物联网app开发流程。首先需要把物联网app开发流程分2个步骤,一...

Android开发进阶 | 如何学习 Android Framework?

大部分有“如何学习Framework源码”这个疑问的,应该大都是应用层开发。应用层是被Framework层调用执行的,知道自己的代码是怎么被调用的,才能理解程序的本质,理解本质有助于解决遇到的...

快速实现APP混合开发(Hybrid App开发)攻略

前言:...

三个阶段带你了解一款app开发的完整流程

第一个阶段需求阶段:1.需求讨论--开发类型、开发平台、具体的产品功能需求、项目预计完成时间、预算2.需求评估--确认合作后评估具体的预算3.界面设计--设计部门进行产品界面设计,形成效果图...

Android 开发中文引导-应用小部件

应用小部件是可以嵌入其它应用(例如主屏幕)并收到定期更新的微型应用视图。这些视图在用户界面中被叫做小部件,并可以用应用小部件提供者发布。可以容纳其他应用部件的应用组件叫做应用部件的宿主(1)。下面的截...

手机软件开发从零开始【Android第2篇Hello】

Hello,朋友们我们又见面了。上一篇我们讲到了《Android开发环境搭建【Android基础第1篇】》,错过的朋友可以点击文章末尾的“阅读原文”查看。另外需要下载JDK和ADT-bundle工具的...

「全栈工程师之梦的开始--安卓开发(二)」开发安卓app

在配置好jdk开发环境、安装好开发工具Androidstudio后,我们就可以开始开发安卓app了。首先,我们需要先了解下android的术语。...

二、Android界面开发 android 开发

学习目标了解Android常用布局了解Android常用控件...

如何开发一款APP既快捷也简便 开发一款app的步骤

具体较为简单的步骤可以选择用androidstudio开发app1、打开软件,在菜单中选择file-》newproject打开创建向导。2、配置项目,确定各个名称和存放项目的存放路径;Applic...

安卓开发中的“Android高手”,需要具备哪些技术?

前言成为一名安卓开发者很容易,但是要成为一名“Android高手”却不那么容易;...

移动开发(一):使用.NET MAUI开发第一个安卓APP

对于工作多年的C#程序员来说,近来想尝试开发一款安卓APP,考虑了很久最终选择使用.NETMAUI这个微软官方的框架来尝试体验开发安卓APP,毕竟是使用VisualStudio开发工具,使用起来也...

微软推出PowerApps:零基础开发Win10/iOS/安卓企业应用

IT之家讯微软今天面向企业宣布了全新的应用开发解决方案PowerApps,让Windows(包括Win10)、iOS以及安卓应用的开发和分发变得更加简单。PowerApps的用户界面与Office办...