昨天一个同事问我DataGridView单元格合并的问题,一开始按照我的设想是算出两个单元格的Rectangle,
然后直接使用e.Graphics.FillRectangle(backColorBrush, rectangle)从新填充下背景色,然后在绘制显示的字符,当然这种想法是行不通的。

下面是我写的一个单元格合并的方法,其实这种方法并不好,只是看上去是把单元格合并了,其实实际DataGridView的列结构是没有发生变化,而且这种重绘的方法并不能编辑,所以我还是建议如果遇到合并单元格的问题,最好还是不要用DataGridView
如有不懂得可以加(源码分享群 81582487)一起交流
代码部分:

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;

namespace WindowsDemo
{
    public partial class DataGridViewMergeCell : Form
    {
        public DataGridViewMergeCell()
        {
            InitializeComponent();
            Init();
        }

private void DataGridViewMergeCell_Load(object sender, EventArgs e)
        {
            int rowIndex = view.Rows.Add();
            DataGridViewRow row = view.Rows[rowIndex];
            row.Cells["value"].Value = "bbb";
            row.Cells["name"].Value = "aaa";
            row.Cells["price"].Value = "bbb";
            row.Cells["num"].Value = "aaa";
            rowIndex = view.Rows.Add();
            row = view.Rows[rowIndex];
            row.Cells["value"].Value = "bbb";
            row.Cells["name"].Value = "aaa";
            row.Cells["price"].Value = "bbb";
            row.Cells["num"].Value = "aaa";
            rowIndex = view.Rows.Add();
            row = view.Rows[rowIndex];
            row.Cells["value"].Value = "bbb";
            row.Cells["name"].Value = "aaa";
            row.Cells["price"].Value = "bbb";
            row.Cells["num"].Value = "aaa";
            rowIndex = view.Rows.Add();
            row = view.Rows[rowIndex];
            row.Cells["value"].Value = "bbb";
            row.Cells["name"].Value = "aaa";
            row.Cells["price"].Value = "bbb";
            row.Cells["num"].Value = "aaa";
            rowIndex = view.Rows.Add();
            row = view.Rows[rowIndex];
            row.Cells["value"].Value = "bbb";
            row.Cells["name"].Value = "aaa";
            row.Cells["price"].Value = "bbb";
            row.Cells["num"].Value = "aaa";
           
           
        }

private void Init() {

view.AllowUserToAddRows = false;
            // view.ColumnHeadersVisible = false;
            view.AutoGenerateColumns = false;
            view.AutoSize = false;
            //view.RowHeadersVisible = false;
            //view.GridColor = System.Drawing.ColorTranslator.FromHtml("#F8F8FF");
            view.ColumnHeadersHeight = 60;
           
            view.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.None;
            view.ColumnHeadersBorderStyle = DataGridViewHeaderBorderStyle.Single;
            view.Columns.Add(new DataGridViewTextBoxColumn() { Name = "name", HeaderText = "主体", Width = 90, ReadOnly = true,Visible = true, AutoSizeMode = DataGridViewAutoSizeColumnMode.None });
            view.Columns.Add(new DataGridViewTextBoxColumn() { Name = "value", HeaderText = "", Width = 80, ReadOnly = true, Visible = true, AutoSizeMode = DataGridViewAutoSizeColumnMode.None });
            view.Columns.Add(new DataGridViewTextBoxColumn() { Name = "price", HeaderText = "主体", Width = 90, ReadOnly = true, Visible = true, AutoSizeMode = DataGridViewAutoSizeColumnMode.None });
            view.Columns.Add(new DataGridViewTextBoxColumn() { Name = "num", HeaderText = "", Width = 80, ReadOnly = true, Visible = true, AutoSizeMode = DataGridViewAutoSizeColumnMode.Fill });
            view.CellPainting -= new DataGridViewCellPaintingEventHandler(view_CellPainting);
            view.CellPainting += new DataGridViewCellPaintingEventHandler(view_CellPainting);
        }

void view_CellPainting(object sender, DataGridViewCellPaintingEventArgs e)
        {
            if (e.RowIndex == 1 && e.ColumnIndex == 1)
            {
                Brush backColorBrush = new SolidBrush(e.CellStyle.BackColor);
                e.Paint(e.CellBounds, DataGridViewPaintParts.All);
                e.Graphics.FillRectangle(backColorBrush, e.CellBounds);
                //绘制背景色(被选中状态下)  
                if (e.State == (DataGridViewElementStates.Displayed | DataGridViewElementStates.Selected | DataGridViewElementStates.Visible))
                    e.PaintBackground(e.CellBounds, false);
                //分别绘制原文本和现在改变颜色的文本  
                Brush fontColor = new SolidBrush(e.CellStyle.ForeColor);
                // e.Graphics.DrawString("", this.Font, fontColor, e.CellBounds, StringFormat.GenericDefault);
                //绘制下边框线
                Brush gridBrush = new SolidBrush(this.view.GridColor);
                Pen pen = new Pen(gridBrush);
                e.Graphics.DrawLine(pen, e.CellBounds.Left, e.CellBounds.Bottom - 1,
                          e.CellBounds.Right - 1, e.CellBounds.Bottom - 1);
              DataGridViewCell cell =  this.view.Rows[e.RowIndex].Cells[e.ColumnIndex];
                
              cell.Value = "";
              e.Handled = true;
            }
            if (e.RowIndex == 1 && e.ColumnIndex == 2)
            {
                Brush backColorBrush = new SolidBrush(e.CellStyle.BackColor);
                e.Paint(e.CellBounds, DataGridViewPaintParts.All);
                e.Graphics.FillRectangle(backColorBrush, e.CellBounds);
                //绘制背景色(被选中状态下)  
                if (e.State == (DataGridViewElementStates.Displayed | DataGridViewElementStates.Selected | DataGridViewElementStates.Visible))
                    e.PaintBackground(e.CellBounds, true);
                //分别绘制原文本和现在改变颜色的文本  
                Brush fontColor = new SolidBrush(e.CellStyle.ForeColor);
               // e.Graphics.DrawString("", this.Font, fontColor, e.CellBounds, StringFormat.GenericDefault);
                //绘制下边框线
                Brush gridBrush = new SolidBrush(this.view.GridColor);
                Pen pen = new Pen(gridBrush);

e.Graphics.DrawLine(pen, e.CellBounds.Left, e.CellBounds.Bottom - 1,
                          e.CellBounds.Right - 1, e.CellBounds.Bottom - 1);

//绘制右边框线
                e.Graphics.DrawLine(pen, e.CellBounds.Right - 1,
                           e.CellBounds.Top, e.CellBounds.Right - 1,
                           e.CellBounds.Bottom - 1);
                DataGridViewCell cell = this.view.Rows[e.RowIndex].Cells[e.ColumnIndex];
                cell.Value = "";
                cell.Tag = "ccccc";
                Rectangle rectanle = e.CellBounds;
                rectanle.X = rectanle.X -15;
                rectanle.Y = rectanle.Y + 5;
                //这里需要注意的是我没有用 e.Graphics 原因是这个只能在当前单元格绘制
                //而我是在DataGridView上面建一个绘图Graphics对象这样就可以看上去在两个单元格里面绘制了
                //这里你们也可以自己试一试
                Graphics graphics = this.view.CreateGraphics();
                //分别绘制原文本和现在改变颜色的文本  
                graphics.DrawString("cccc", this.Font, new SolidBrush(e.CellStyle.ForeColor), rectanle, StringFormat.GenericDefault);
                e.Handled = true;
            }
        }
    }
}

以下是我的建议,这种虽然看上去是把问题解决了,其实这样有很多缺点,一个是不能编辑,而且这个还有个bug就是点击左边的边框的时候会把左边的文字隐藏起来,我试了很多方法都没解决,这个应该是编辑的时候背景颜色给遮住了。

一下推荐一个可以合并单元格的第三方控件(DevExpress.XtraGrid.v7.3.dll),这个现在我也没有研究,不过后续我会研究的,也会在写篇文章的

DataGridView合并单元格的更多相关文章

  1. DataGridView合并单元格(多行多列合并)

    一.点击在拖入的显示控件(TreeList)右上方的箭头,在Treelist任务中选择数据源,添加项目数据源,依次选择数据库.数据集,新建连接,浏览选择数据库(*.mdb),依次点击 下一步,选择“表 ...

  2. DataGridView合并单元格(一列或一行)

    #region"合并单元格的测试(一列或一行)" // int?是搜索一种类型(可空类型),普通的int不能为null,而用int?,其值可以为null //private int ...

  3. Windows Forms DataGridView中合并单元格

    Windows Forms DataGridView 没有提供合并单元格的功能,要实现合并单元格的功能就要在CellPainting事件中使用Graphics.DrawLine和 Graphics.D ...

  4. datagridview 纵向 横向 合并单元格

    datagridview 单元格合并:纵向以及横向合并参考了csdn上不知哪位的代码,具体哪位找不到连接了. 纵向合并: /// <summary> /// 纵向合并,即合并数据项的值 / ...

  5. NPOI_winfrom导出Excel表格(一)(合并单元格、规定范围加外边框、存储路径弹框选择)

    1.导出 private void btn_print_Click(object sender, EventArgs e) { DataTable dtNew = new DataTable(); d ...

  6. C# 获取Excel中的合并单元格

    C# 获取Excel中的合并单元格 我们在制作表格时,有时经常需要合并及取消合并一些单元格.在取消合并单元格时需要逐个查找及取消,比较麻烦.这里分享一个简单的方法来识别Excel中的合并单元格,识别这 ...

  7. jquery操作表格 合并单元格

    jquery操作table,合并单元格,合并相同的行 合并的方法 $("#tableid").mergeCell({ cols:[X,X] ///参数为要合并的列}) /** * ...

  8. NPOI操作EXCEL(五)——含合并单元格复杂表头的EXCEL解析

    我们在第三篇文章中谈到了那些非常反人类的excel模板,博主为了养家糊口,也玩命做出了相应的解析方法... 我们先来看看第一类复杂表头: ...... 博主称这类excel模板为略复杂表头模板(蓝色部 ...

  9. poi获取合并单元格内的第一行第一列的值

    当读取如图所示的excel时,显示为第1行 第1列 的内容是:合并单元格 其它在合并单元格区域内的单元格不显示 示例代码如下: import java.io.FileInputStream; impo ...

随机推荐

  1. 浏览器对象模型(BOM,Browser Object Model)

    本文内容     1.概述     2.windows与document     3.对话框     4.定时调用     5.URL解析与访问历史     6.浏览器和屏幕信息 ★概述     &q ...

  2. Python基础:数值(布尔型、整型、长整型、浮点型、复数)

    一.概述 Python中的 数值类型(Numeric Types)共有5种:布尔型(bool).整型(int).长整型(long).浮点型(float)和复数(complex). 数值类型支持的主要操 ...

  3. WCF小结

    WCF总结 1.WCF(原代号为Indigo)是一个用于创建和运行分布式系统的技术集合,使用它能创建安全的.可靠的.跨平台的的分布式解决方案.它的面向服务的编程模型,整合了.Net平台下以往全部的分布 ...

  4. U-boot的环境变量值得注意的有两个: bootcmd 和bootargs

    本文转载至:http://www.cnblogs.com/cornflower/archive/2010/03/27/1698279.html U-boot的环境变量值得注意的有两个: bootcmd ...

  5. NYOJ:题目860 又见01背包

    题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=860 方法一:不用滚动数组(方法二为用滚动数组,为方法一的简化) 动态规划分析:最少要拿总 ...

  6. Javaweb上下文监听者ServletContextListener

    一个监听类,不是一个servlet或JSP,它能监听ServletContext一生中的两个关键事件:初始化(创建)和撤销.这个类实现了javax.servlet.ServletContextList ...

  7. Docker on CentOS for beginners

    Introduction The article will introduce Docker on CentOS. Key concepts Docker Docker is the world's ...

  8. mysql ALL_O_DIRECT引发的unaligned AIO/DIO导致hang

    公司内部有一套mysql环境,使用的是percona server分支(和其他几十套环境的版本.参数完全相同),就这套环境每隔两三天就会hang一次,关键hang的时候服务器cpu也就是百分之三四十, ...

  9. 误报的java.sql.SQLException: Parameter number 21 is not an OUT parameter

    今天为了模拟一个mysql内存不释放问题,要测试一个存储过程,同时具有出参和入参,启动时报了上述错误. <select id="funcl_trd_secu_execution_que ...

  10. (四)play之yabe项目【页面】

    (四)play之yabe项目[页面] 博客分类: 框架@play framework   主页面 显示当前发表博客的完整内容,以及历史博客列表 Bootstrap Job 一个play job任务就是 ...