最简单有效的思路实现各类表格打印(包括票据)

程序界面本单位所需水电管理,学习水晶报表太累,不大懂,又难部署

通过控件在窗体的布局,实现模拟打印

如图:

打印效果   

 '打印datagridview内容单元格
    Public Sub PrintDataGridDetail(ByVal loc As Point, ByRef e As System.Drawing.Printing.PrintPageEventArgs, ByVal dg As DataGridView, Optional ByVal bnFormat As Boolean = False)
        Dim irow As Integer : Dim icol As Integer '+ New Point(0, datagrid.ColumnHeadersHeight)
        Dim strValue As String = ""
        Dim rect As New Rectangle

        Dim dgDetailX As Integer : Dim dgDetailY As Integer
        dgDetailX = Loc.X : dgDetailY = Loc.Y

        Dim StrFormat As New StringFormat
        With StrFormat
            .LineAlignment = StringAlignment.Center
            .Alignment = StringAlignment.Center
            .Trimming = StringTrimming.None
        End With

    

        '全部打印,打印datagrid指定行数的表格
        If bnFormat = False Then
            Dim tmpwidth As Integer
            For i As Integer = 0 To ModVia.rowsNumberPerPage - 1
                For j As Integer = 0 To dg.Columns.Count - 1
                    Dim rect1 As New Rectangle
                    rect1 = New Rectangle(dgDetailX, tmpwidth, dg.Columns(j).Width, dg.RowTemplate.Height)
                    rect1.Offset(0, dg.RowTemplate.Height * i)
                    tmpwidth += dg.Columns(j).Width
                Next
            Next
        End If
        '打印表格内容
        '循环分页
        Dim tmpStartRow As Integer : Dim tmpEndRow As Integer
        tmpStartRow = (Me.iCurrenPageNo - 1) * ModVia.rowsNumberPerPage
        tmpEndRow = tmpStartRow + ModVia.rowsNumberPerPage - 1
        If tmpEndRow >= dg.Rows.Count - 1 Then tmpEndRow = dg.Rows.Count - 1

        Dim offsetY As Integer
        If Me.iCurrenPageNo = 1 Then
            offsetY = 0
        Else
            offsetY = (Me.iCurrenPageNo - 1) * dg.RowTemplate.Height * ModVia.rowsNumberPerPage
        End If

        Dim ir As Integer = 0
        For irow = tmpStartRow To tmpEndRow  'datagrid.Rows.Count - 1
            Me.iCurrenRow = (Me.iCurrenPageNo - 1) * ModVia.rowsNumberPerPage + irow + 1
            Dim colWidth As Integer = 0
            For icol = 0 To dg.Columns.Count - 1
                If dg.Rows(irow).Cells(icol).Value Is DBNull.Value Then
                    strValue = ""
                Else
                    strValue = dg.Rows(irow).Cells(icol).FormattedValue
                End If
                '方法一
                e.Graphics.DrawString(strValue, dg.Font, Brushes.Black, dgDetailX + colWidth, dgDetailY + ir * dg.RowTemplate.Height + dg.ColumnHeadersHeight + 3)  '3为偏移量
                colWidth += dg.Columns(icol).Width
                '方法二
                rect = dg.GetCellDisplayRectangle(icol, irow, False)
                rect.Offset(dgDetailX + IIf(dg.RowHeadersVisible, 0, -2), dgDetailY - offsetY)
                'e.Graphics.DrawString(strValue, dg.Font, Brushes.Black, rect)

                'If Me.bnPrintFormat = False Then
                '    e.Graphics.DrawRectangle(Pens.Black, rect)
                'End If
            Next
            '当前打印页面的行标
            ir += 1
        Next
        '设置分页
        Me.iCurrenPageNo += 1

        If Me.iCurrenPageNo > Me.iTotalPageNo Then
            e.HasMorePages = False
            Me.iCurrenRow = 0
        Else
            e.HasMorePages = True
        End If

    End Sub

    '打印普通控件
    Public Sub PrintControl(ByVal loc As Point, ByVal e As System.Drawing.Printing.PrintPageEventArgs, ByVal cb As Control, Optional ByVal bnFormat As Boolean = False)
        Dim rect As New Rectangle
        Dim StrFormat As New StringFormat
        With StrFormat
            .LineAlignment = StringAlignment.Near
            .Alignment = StringAlignment.Near
        End With

        For Each ob As Control In cb.Controls
            rect = ob.Bounds
            rect.Offset(e.MarginBounds.Left, e.MarginBounds.Top)
            If TypeOf (ob) Is TextBox Then
                e.Graphics.DrawString(ob.Text, ob.Font, Brushes.Black, rect, StrFormat)
            ElseIf (TypeOf (ob) Is Label) Then
                'MsgBox(ob.Text + vbCrLf + rect.ToString)
                '非格式套打
                If bnFormat = False Then
                    e.Graphics.DrawString(ob.Text, ob.Font, Brushes.Black, rect, StrFormat)
                End If
            End If
        Next
        'e.Graphics.DrawRectangle(Pens.DarkCyan, cb.Bounds)
    End Sub

    '打印TabPanel控件 ,表格下面采用tabpanel控件
    Public Sub PrintTabPanel(ByVal loc As Point, ByVal e As System.Drawing.Printing.PrintPageEventArgs, ByVal tb As TableLayoutPanel, Optional ByVal bnFormat As Boolean = False)
        '===表格以下部分起始Y坐标 额定 几行后 开始
        Dim MsgX As Integer : Dim MsgY As Integer
        Dim strFormat As New StringFormat
        Dim rect As New Rectangle

        MsgX = loc.X : MsgY = loc.Y
        With strFormat
            .Alignment = StringAlignment.Center
            .LineAlignment = StringAlignment.Center
        End With
        '绘制容器内 控件字符串内容
        Try
            For Each ob1 As Control In Me.TableLayoutPanel1.Controls
                rect = ob1.Bounds
                rect.Offset(MsgX, MsgY)
                If TypeOf (ob1) Is TextBox Then
                    '根据原有对齐方式进行打印
                    With strFormat
                        .Alignment = StringAlignment.Center
                        '.LineAlignment= '0左,2右 ,1 中 textAlign  左0,右1,中2
                        .LineAlignment = StringAlignment.Near ' Me.ChangeAlignMent(ob1)
                    End With
                    e.Graphics.DrawString(ob1.Text, ob1.Font, Brushes.Black, rect, strFormat)
                ElseIf TypeOf (ob1) Is Label Then
                    If Me.bnPrintFormat = False Then
                        e.Graphics.DrawString(ob1.Text, ob1.Font, Brushes.Black, rect, strFormat)
                    End If
                End If
            Next
        Catch ex As Exception
        End Try

        If bnPrintFormat = True Then Exit Sub

        '绘制容器表格
        Dim rstyle As New RowStyle : Dim cStyle As New ColumnStyle
        Dim tempX As Integer : Dim tempY As Integer
        Dim ia As Integer : Dim ja As Integer
        Dim ob2 As New Control
        tempX = MsgX
        tempY = MsgY
        Try
            For ia = 0 To Me.TableLayoutPanel1.RowCount - 1
                rstyle = Me.TableLayoutPanel1.RowStyles(ia)
                '依次判断 列表
                For ja = 0 To Me.TableLayoutPanel1.ColumnCount - 1
                    cStyle = Me.TableLayoutPanel1.ColumnStyles(ja)
                    ob2 = Me.TableLayoutPanel1.GetControlFromPosition(ja, ia)
                    If ob2 Is Nothing Then
                        tempX += cStyle.Width
                        Continue For
                    End If
                    Dim jaSpan As Integer = 1
                    jaSpan = Me.TableLayoutPanel1.GetColumnSpan(ob2)
                    '如有跨列的情况
                    If jaSpan <> 1 Then
                        Dim tmpTotalWidth As Integer = 0
                        For k As Integer = ja To ja + jaSpan - 1
                            tmpTotalWidth += Me.TableLayoutPanel1.ColumnStyles(k).Width
                        Next
                        rect = New Rectangle(tempX, tempY, tmpTotalWidth, rstyle.Height)
                        e.Graphics.DrawRectangle(Pens.Black, rect)
                        ja = ja + jaSpan - 1
                        tempX += tmpTotalWidth
                        Continue For
                    ElseIf jaSpan = 1 Then
                        rect = New Rectangle(tempX, tempY, cStyle.Width, rstyle.Height)
                        'rect.Offset(New Point(0, Me.GroupBox1.Top - Yoffset))
                        e.Graphics.DrawRectangle(Pens.Black, rect)
                        tempX += cStyle.Width
                    End If
                Next
                'x坐标复位,y坐标增加
                tempX = MsgX
                tempY += rstyle.Height
            Next
        Catch ex As Printing.InvalidPrinterException
            MessageBox.Show(ex.Message, "测试", MessageBoxButtons.OK, MessageBoxIcon.Error)
            Exit Sub
        End Try
    End Sub

我相信以上思路 会对大家有个帮助

 

 

DataGridView打印控件和.NET打印控件5.6版(含报表模板设计组件)2014年6月22日修改完成,完全免费,在.NET2.0及以上环境下都可以使用(VB打印、C#打印都是可以的),有帮助文档与使用实例。 与上一版本的5.5版相比,新控件5.6版的主要更改如下: 1、增加了一个新打印组件SimpleReport组件,该组件与DGVPrint组件一样在运行时可进行打印参数设置,但比DGVPrint组件功能更强大,可以自动管理多个打印方案,在打印预览时可以自由在各个打印方案之间切换;可以在打印参数设置窗口中动态定义多表头,还可以像EasyReport组件一样设置和使用参数变量(具体使用效果参见实例程序); 2、解决了EasyReport组件在插入变量参数时,变量参数未排序导致查找不方便的问题; 3、在打印预览界面添加了简单的双面打印功能。(打印预览界面工具栏中的“打印\双面打印”菜单); 4、其他一些完善,比如DGVPrint组件中设置的行高无效等问题。 本控件特色: 1、强大的DataGridView打印功能,不仅可以以多种形式(普通打印、分栏打印、跨页打印、工资条打印打印DGV表格,基本上能完全按DGV控件本身设置的格式如字体、字号、背景颜色、前景颜色、单元格对齐方式等打印出来,文字图像都可以打印,而且是完全根据表格当前的显示顺序进行打印的,基本上做到了所见即所得的打印。 2、报表设计功能。报表模板设计组件EasyReport与WebEasyReport组件可以设计普通报表、分组报表、套打模板等,分别以DataGridView为数据源。控件的位置以毫米为计量单位,定位准确,很适合套打单据设计。 3、强大的图表打印功能。5.2版控件新增了一个Chartlet的组件,使用非常方便,可以生成柱形图、饼图、折线图等多种图形,而且可以设置2D或3D效果,既可以在打印控件中打印出来,也可以在Graphics对象中显示。 4、分组汇总打印DataGridVeiw功能,每组还可以自动换新页打印,还可以自动增加行号。 5、强大的文本打印输出功能,控件提供多个文本打印重载函数,打印文本时,如果需要,控件会自动换行和换页打印输出。还增加了以指定行间距及字符间距打印文本的功能,可以用固定行距,也可以用单倍或多倍行距打印文本。 6、强大的绘图功能,基本上.NET的GDI+的绘图函数(如直线、矩形、路径、多边形、曲线等)都有,只有个别函数的名称有点区别。 7、支持同一文档多种版面格式打印(类似于Word中的节的功能):对同一份文档,不同的页面可以设置不同的格式(纸张大小、纸张方向、页边距),只需要在新增一页时在NewPage方法中指定要使用的页面格式即可,使用非常简单。 8、报表文件保存功能。本控件允许将当前打印预览的内容保存为报表文件,以后使用本控件重新打开该报表文件即可重现原来保存报表时的打印内容。 9、Excel导出功能,可以将DataGridView和GridView导出为Excel文件,5.2版控件还增加了不依赖Office的导出Excel功能,而且速度非常快,5.4版还增加了合并单元格的导出功能。 10、打印DataGridView时的打印方案保存与读取功能。可以将当前打印参数保存为打印方案文件,或者从保存的打印方案文件中读取打印参数。 11、水印打印功能。根据需要,可以在页面中打印或不打印以半透明空心文字打印水印。 12、强大的容器控件打印功能(DrawPanel函数)。借助该函数,您只需要在您的容器控件中设计好要打印的内容及打印内容的相对位置,控件轻松帮你打印出来(如果超过一页,控件会自动换页续打)。 13、特殊文字效果打印功能。控件具有打印浮雕文字、阴影文字、空心文字、块文字的功能。 14、页眉页脚中既可打印文字,也可打印图像,或者即打印图像又打印输出文字。 15、图像与图标打印输出功能。 16、多表头(跨行跨列的复杂表头)打印功能,多表头组件支持多表头显示与打印、单元格内容的合并显示、打印与导出。 17、自定义纸张支持功能。 18、纸张背景图片设置打印功能。 19、.NET4.0支持功能(是单独的一个文件)。 20、直接打印窗口中的TreeView控件功能。 21、打印窗口中的ListView功能。 22、RichTextBox控件的RTF文本打印功能。 23、斜线表头打印功能(5.4版新增)。 24、二维码打印功能(5.5版本增加)。 25、5.6版新增的SimpleReport组件允许您在一个方案文件中管理多个打印方案,在打印预览时能自由在各个打印方案之间切换。 我将持续改进该控件,并将不断推出控件的新版本,要查看或下载控件的升级版本,请登陆网站:http://myyouping.download.csdn.n
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值