WinForm程序中表单的自动保存列的宽度和位置,是一种常见的功能,对于用户体验来说是非常好的。现记录一下实现过程:

1、新建一个类,命为为:DataGridViewColumnStyle。

这个类实现的功能:当DataGridView的列宽或列的位置发生改变时,系统将自动记录DataGridView的设置。当用户下次打开此窗体的时候,表单的样式是他上次设置的模样。代码如下:

    class DataGridViewColumnStyle
{
private DataGridView dgvTarget = null; //待处理的DataGridView对象
private string path; //文件路径
private DataTable dtColumnStyle = null; //列样式数据表
private bool isBindColumnStyle = false; //是否绑定列样式否 //DataGridView属性
public DataGridView DataGridView
{
get { return dgvTarget; }
set
{
//去除事件
if (dgvTarget != null)
{
dgvTarget.ColumnWidthChanged -= new DataGridViewColumnEventHandler(DataGridView_ColumnWidthChanged);
dgvTarget.ColumnDisplayIndexChanged -= new DataGridViewColumnEventHandler(DataGridView_ColumnDisplayIndexChanged);
}
dgvTarget = value;
//注册事件
if (dgvTarget != null)
{
dgvTarget.ColumnWidthChanged += new DataGridViewColumnEventHandler(DataGridView_ColumnWidthChanged);
dgvTarget.ColumnDisplayIndexChanged += new DataGridViewColumnEventHandler(DataGridView_ColumnDisplayIndexChanged);
}
}
} //无参构造函数
public DataGridViewColumnStyle()
{
} //有参构造函数
public DataGridViewColumnStyle(DataGridView dataGridView) : this()
{
DataGridView = dataGridView;
//文件名
string formName = dgvTarget.FindForm().Name;
string userId = "Test";
path = Application.StartupPath + @"\Accounts\" + userId + "\\" + formName + "_" + dgvTarget.Name + ".xml";
//列样式数据表
dtColumnStyle = new DataTable();
dtColumnStyle.TableName = dgvTarget.Name; //表名
dtColumnStyle.Columns.Add("Name"); //列名
dtColumnStyle.Columns.Add("Width"); //列宽度
dtColumnStyle.Columns.Add("DisplayIndex"); //显示顺序
//绑定列样式
BindColumnStyle();
} /// <summary>
/// 绑定列样式
/// </summary>
private void BindColumnStyle()
{
try
{
//赋初始值
isBindColumnStyle = true; //如果不存在则保存列样式
if (!File.Exists(path))
{
SaveColumnStyle();
} //加载列样式
dtColumnStyle.ReadXml(path); foreach (DataRow row in dtColumnStyle.Rows)
{
if (dgvTarget.Columns.Contains(row["Name"].ToString().Trim()) && dgvTarget.Columns[row["Name"].ToString().Trim()].Visible == true)
{
dgvTarget.Columns[row["Name"].ToString().Trim()].Width = int.Parse(row["Width"].ToString().Trim());
dgvTarget.Columns[row["Name"].ToString().Trim()].DisplayIndex = int.Parse(row["DisplayIndex"].ToString().Trim());
}
}
}
catch (Exception ex)
{
DeleteColumnStyle();
MessageBox.Show(ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
finally
{
isBindColumnStyle = false;
}
} /// <summary>
/// 列显示位置改变时
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void DataGridView_ColumnDisplayIndexChanged(object sender, DataGridViewColumnEventArgs e)
{
if (isBindColumnStyle == false)
{
SaveColumnStyle();
}
} /// <summary>
/// 列宽度改变时
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void DataGridView_ColumnWidthChanged(object sender, DataGridViewColumnEventArgs e)
{
if (isBindColumnStyle == false)
{
SaveColumnStyle();
}
} /// <summary>
/// 保存列样式
/// </summary>
private void SaveColumnStyle()
{
try
{
//如果目录不存在则创建
string dir = path.Substring(, path.LastIndexOf('\\'));
if (!Directory.Exists(dir))
{
Directory.CreateDirectory(dir);
}
//读取列样式
string[] columnStyle = new string[dgvTarget.Columns.Count];
foreach (DataGridViewColumn col in dgvTarget.Columns)
{
if (col.Visible == true)
{
columnStyle[col.DisplayIndex] = col.Name + '|' + col.Width + '|' + col.DisplayIndex;
}
}
int colsCount = columnStyle.Length;
//保存列样式
dtColumnStyle.Rows.Clear();
for (int i = ; i < colsCount; i++)
{
string[] str = new string[];
try
{
DataRow newRow = dtColumnStyle.NewRow();
str = columnStyle.GetValue(i).ToString().Split('|');
newRow["Name"] = str[];
newRow["Width"] = str[];
newRow["DisplayIndex"] = str[];
dtColumnStyle.Rows.Add(newRow);
}
catch
{
continue;
}
}
dtColumnStyle.WriteXml(path);
}
catch (Exception ex)
{
MessageBox.Show(ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
} /// <summary>
/// 删除列样式
/// </summary>
private void DeleteColumnStyle()
{
try
{
if (File.Exists(path))
{
File.Delete(path);
}
}
catch (Exception ex)
{
MessageBox.Show(ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
}
}

2、以上这些,已经实现了全部的功能。下面开始建一个WinForm程序来测试结果,为方便测试将DataGridView的数据源由xml文件读取。

从SQL Server数据库随便找张数据表生成XML,文件保存为Test.xml。(请将Test.xml文件拷贝到Debug文件夹下面)

SELECT TOP 10 MO_NO,MRP_NO,QTY,BIL_NO
FROM MF_MO
WHERE MO_DD='2019-11-07'
ORDER BY MO_NO
FOR XML PATH ('Category'),TYPE,ROOT('DocumentElement')

3、新建一个WinForm程序,命名为Main,并拖入一个DataGridView控件,请保留【启用列重新排序】的勾选。

Main_Load方法如下:

        private void Main_Load(object sender, EventArgs e)
{
try
{
//xml文件路径
string path = @"Test.xml";
//读取文件
DataSet ds = new DataSet();
if (File.Exists(path))
{
ds.ReadXml(path);
}
dataGridView1.DataSource = ds.Tables.Count > ? ds.Tables[] : null;
//加工dataGridView1
#region 加列标题测试
dataGridView1.Columns[].HeaderText = "制令单号";
dataGridView1.Columns[].HeaderText = "成品编号";
dataGridView1.Columns[].HeaderText = "生产数量";
dataGridView1.Columns[].HeaderText = "来源单号";
#endregion
DataGridViewColumnStyle style = new DataGridViewColumnStyle(dataGridView1);
}
catch (Exception ex)
{
MessageBox.Show(ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
}

4、执行程序,随意拖动或拉宽DataGridView列,然后重新进一次程序即可看到效果:

好了,分享就到此结束了,希望对有此需要的人有一些帮助。

DataGridView自动保存列的宽度和位置的更多相关文章

  1. 如何不让DataGridView自动生成列

    如果不想让DataGridView自动生成与数据源对应的列, 只需要把属性AutoGenerateColumns设为false即可. 需要注意: 在界面设计的属性窗口中是看不到AutoGenerate ...

  2. datagridview自动填充列头

    //填充datagridview dgv.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.Fill;

  3. C# 禁止datagridview 自动产生列

    dataGridView1.AutoGenerateColumns = false;

  4. 【转】如何不让DataGridView自动生成列

    源地址:https://www.cnblogs.com/hailexuexi/p/3983856.html

  5. DatagridView自动充满屏幕,并能指定某列宽度

    1.要使datagridview正好充满屏幕,设置其AutoSizeColumnsMode属性为fill 2. 同时,我们想要某列宽点,某列窄点,在AutoSizeColumnsMode属性为fill ...

  6. [Winform]DataGridView列自适应宽度

    引言 在做winform项目中,数据控件DataGridView的使用多多少少是会用到的,如果不设置它的属性,默认情况下是不会自适应宽度的,你想查看某项的数据,就不得不将标题栏拖来拖去,挺烦的. 方法 ...

  7. ## GridView 布局:item设置的高度和宽度不起作用、自动适配列数、添加Header和Footer ##

    一.item设置的高度和宽度不起作用 转自:http://www.cnblogs.com/0616--ataozhijia/p/6031875.html [Android Pro] listView和 ...

  8. DataGridView列自适应宽度

    来源:http://www.cnblogs.com/wolf-sun/p/3480104.html 在做winform项目中,数据控件DataGridView的使用多多少少是会用到的,如果不设置它的属 ...

  9. DataGridView列的宽度、行的高度自动调整

    注意:DataGridView控件是从.NET Framework 2.0版本开始追加的. 介绍一下DataGridView列的宽度和行的高度,根据单元格或Header的内容(一般是内容全部被表示)自 ...

随机推荐

  1. Oracle SCN 详解

    一.简介 scn,system change number 在某个时间点定义数据库已提交版本的时间戳标记,Oracle为每个已提交事务分配一个唯一的scn,scn值是对数据库进行更改的逻辑时间点.sc ...

  2. ASP.NET Core SignalR :学习消息通讯,实现一个消息通知

    什么是 SignalR 目前我用业余时间正在做一个博客系统,其中有个功能就是评论通知,就是假如A用户评论B用户的时候,如果B用户首页处于打开状态,那么就会提示B用户有未读消息.暂时用SignalR来实 ...

  3. MVC 入门 自动生成 增删改查所有功能

    MVC现在版本已经是5了   EF现在最新的应该是6.0.2了 开发工具是 Visual Studio2013 数据库是 SQL Server 2012 这些需要.NET Framework4.5 的 ...

  4. mybatis错题

    第一题 解析: MyBatis的动态SQL中没有else元素,when元素的test属性中直接书写表达式即可,即test=”表达式”. 第二题 解析: resource属性和url属性是必须的属性,但 ...

  5. Mac 配置 PlantUML

    PlantUML简介 UML: Unified Modeling Language 统一建模语言,是非专利的第三代建模和规约语言.UML是一种开放的方法,用于说明.可视化.构建和编写一个正在开发的.面 ...

  6. Android Studio 3.0下创建menu布局文件的图例

    当开始一个android app的时候,android studio项目中没有看到menu文件夹:如下所示: 当要添加一个按钮时,很多文档上都会说,通过在项目的 res/menu 目录中新增一个 XM ...

  7. Bayer图像处理 raw 数据解析

    Bayer是相机内部的原始图片, 一般后缀名为.raw. 很多软件都可以查看, 比如PS. 我们相机拍照下来存储在存储卡上的.jpeg或其它格式的图片, 都是从.raw格式转化 过来的. .raw格式 ...

  8. 【MobX】390- MobX 入门教程(上)

    点击上方"前端自习课"关注,学习起来~ 本文考虑到篇幅问题,将<MobX 入门教程>分成上.下两篇文章,方便阅读.分配安排: 一.MobX 介绍 首先看下官网介绍: ★ ...

  9. UWP 中的全局异常处理

    问题 在开发一款应用的过程中,我们开发者很难考虑到所有问题,往往会忘记处理一些可能发生的异常.随之而来的结果就是用户使用过程中接连不断的崩溃.所以,我们有必要处理所有未被我们处理的异常. 本文介绍了 ...

  10. Linux sudo用户提权与日志审计

    一.格式说明及常用配置选项 格式: 用户或组 主机=授权可以使用哪个用户的权限 可以执行的命令 User_Alias 用户定义别名(别名可以是用户,用户组(用户组前面要加%))例:User_Alias ...