DataGridView合并单元格

昨天一个同事问我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),这个现在我也没有研究,不过后续我会研究的,也会在写篇文章的

时间: 2024-10-12 19:35:13

DataGridView合并单元格的相关文章

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

一.点击在拖入的显示控件(TreeList)右上方的箭头,在Treelist任务中选择数据源,添加项目数据源,依次选择数据库.数据集,新建连接,浏览选择数据库(*.mdb),依次点击 下一步,选择“表”,完成. 二.具体代码如下: #region"合并单元格(多行多列)" //需要(行.列)合并的所有列标题名 List<String> colsHeaderText_V = new List<String>(); List<String> colsHe

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

#region"合并单元格的测试(一列或一行)" // int?是搜索一种类型(可空类型),普通的int不能为null,而用int?,其值可以为null //private int? nextrow = null; //private int? nextcol = null; //在CellPainting方法后调用 private void dataGridView1_CellFormatting(object sender, System.Windows.Forms.DataGri

datagridview 纵向 横向 合并单元格

datagridview 单元格合并:纵向以及横向合并参考了csdn上不知哪位的代码,具体哪位找不到连接了. 纵向合并: /// <summary> /// 纵向合并,即合并数据项的值 /// </summary> /// <param name="e"></param> private void DrawCellVer(DataGridViewCellPaintingEventArgs e) { if (e.CellStyle.Alig

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

1.导出 1 private void btn_print_Click(object sender, EventArgs e) 2 { 3 DataTable dtNew = new DataTable(); 4 5 dtNew.Columns.Add(new DataColumn("commodity_name", typeof(object))); 6 dtNew.Columns.Add(new DataColumn("specifications", type

Datatables js 复杂表头 合并单元格

x →Datatables官网← x 项目中用到的Table都是用Datatables插件来搞得: 以前都是生成一般性的table: 近期要生成一些复杂表头,合并单元格之类的: 研究了一下. x 去官网上也查到了[复杂表头(rowspan 和 colspan)(Complex headers (rowspan and colspan))],[复杂表头], 但是只是html代码,蛋疼啊... 后来看到了[创建行回调(Row created callback)]哪里的时候 ,豁然开朗!!!   先上

DataGrid合并单元格(wpf)

在网上搜索wpf合并单元格,一直没搜索到,没办法,只能自己想办法搞定了.其实就是DataGrid套DataGrid,为了方便支持Column拖动,在合并的DataGridColumn那一列的Header也放一个DataGrid,但是合并的这一个连续列中只支持一列的拖动,效果如下,上图 代码如下 首先定义两个模板,一个用于合并的Header显示 <DataTemplate x:Key="MergeHeader">             <DataGrid Horizo

jQuery_easyUI 合并单元格 (DataGrid 数据表格)

<table id="dg" style="height:350px;z-index:-5555; " class="easyui-datagrid" rownumbers="true" data-options="fitColumns: true, iconCls: 'icon-edit', scrollbarSize:0, multiSort:true, remoteSort:true, paginatio

【记录】解析具有合并单元格的Excel

最近公司让做各种数据表格的导入导出,就涉及到电子表格的解析,做了这么多天总结一下心得. 工具:NOPI 语言:C# 目的:因为涉及到导入到数据库,具有合并单元格的多行必然要拆分,而NPOI自动解析的时候拆分单元格除第一个单元格外其余值都是空,对于列头有合并项目的,数据库设计一般才有合并单元格下面的最小列单元作为数据库字段.于是希望达到这样一个效果.于是有了一个思路就是把读入的Excel复制到新建的Excel,然后再去读新的Excel.总体思路就是把合并单元格所包含的所有最小单元格的值都设置成合并

devexpress实现单元格合并以及依据条件合并单元格

1.devexpress实现单元格合并非常的简单,只要设置属性[AllowCellMerge=True]就可以了,实现效果如下图: 2.但是在具体要求中并非需要所有的相同单元格都合并,可能需要其他的条件来控制合并.这个时候我们就需要在事件gridView1_CellMerge中来控制了.下图为根据最后一列判断是否合并单元格的效果图(其中第四列设置为不合并<非必需>,这里只是为了达到一个比较效果.). 3.重要代码: int row1 = e.RowHandle1; int row2 = e.R