使用FastReport打印二维码

        简单介绍一下该功能所在的项目背景:C#语言编写的WPF客户端应用程序,在“结账”模块中,打印出的收款小票上需要显示一个二维码,服务生拿着小票去找顾客,顾客可以选择现金、银行卡等普通支付方式,也可以直接扫小票上的二维码进行微信支付。用于打印小票的模板使用FastReport制作。结账功能和收款单打印模板已经在使用,现在只需要在原有模板的基础上进行修改。

        首先,我们要明确一点:二维码其实就是一张图片。因此和打印条形码不同,打印二维码,只需要使用FastReport中的图片控件就可以了。双击图片控件后,有四种方式可以用于设置图片的数据源,如下图所示:

                        使用FastReport打印二维码_第1张图片 使用FastReport打印二维码_第2张图片 使用FastReport打印二维码_第3张图片

使用FastReport打印二维码_第4张图片

        这些方法大家一看就知道怎么用,本项目中我们使用最下面这张大图中的“数据列”方法,用FastReport数据源中的内容对图片进行设置。

        以本项目的结账收款单打印模板为例,实现打印二维码的步骤如下所示:

1、在程序中获取需要打印的二维码数据,并将数据源传给打印模板

        在该项目中,使用DataSet向打印模板传递数据源,使用Dictionary<string, string>向打印模板传递参数。传递的方法这里不再详述。

        为了实现打印二维码这个功能,在传递给打印模板的数据源DataSet中添加了一张用于显示二维码的“微信数据”表,还添加了一个参数“微信二维码”。“微信二维码”参数中保存了这个二维码的地址,若地址为空字符串,说明没有微信二维码,不需要打印。在程序端,调用下面的方法AddQRCodeForPrint为打印模板添加二维码所需的数据。微信二维码图片的url(例:weixin://wxpay/payurl?pr=9BF4WY0)是线上的代码生成的,这里不做介绍。

        /// <summary>
        /// 向打印模板添加微信二维码
        /// </summary>
        /// <param name="dsPrint">数据源</param>
        /// <param name="objdict">参数</param>
        /// <param name="codeURL">微信二维码图片URL</param>
        public static void AddQRCodeForPrint(DataSet dsPrint, Dictionary<string, string> objdict, String codeURL)
        {
            try
            {
                //添加参数:微信二维码图片URL
                objdict.Add("微信二维码", codeURL);

                //将微信二维码写入数据源
                Bitmap bp = GetQrImage(codeURL);
                MemoryStream ms = new MemoryStream();
                bp.Save(ms, System.Drawing.Imaging.ImageFormat.Bmp);
                Byte[] code = ms.ToArray();

                DataTable dt = new DataTable();
                dt.Columns.Add("二维码", typeof(Byte[]));
                DataRow dr = dt.NewRow();
                dr["二维码"] = code;
                dt.Rows.Add(dr);
                dt.TableName = "微信数据";
                dsPrint.Tables.Add(dt);
            }
            catch
            {
                if (objdict != null && !objdict.ContainsKey("微信二维码"))
                    objdict.Add("微信二维码", "");
            }
        }

        这个方法中,用于绘制二维码图形的方法GetQrImage如下所示:

        /// <summary>
        /// 绘制二维码
        /// </summary>
        /// <param name="qrstr"></param>
        /// <param name="ImageWidth"></param>
        /// <param name="ImageHeight"></param>
        /// <param name="savelocal"></param>
        /// <returns>返回:二维码</returns>
        private static Bitmap GetQrImage(String qrstr, int ImageWidth = 256, int ImageHeight = 256, bool savelocal = false)
        {
            try
            {
                Dictionary<EncodeHintType, object> ht = new Dictionary<EncodeHintType, object>();
                ht.Add(EncodeHintType.MARGIN, 1);
                BitMatrix matrix = new MultiFormatWriter().encode(qrstr, BarcodeFormat.QR_CODE, ImageWidth, ImageHeight, ht);
                Bitmap bitmap = new BarcodeWriter().Write(matrix); //toBitmap(matrix);
                int[] rec = matrix.getEnclosingRectangle(); //二维码所有的位置及大小  前两位是位置  后两位是大小
                Bitmap argb32bp = new Bitmap(ImageWidth, ImageHeight);
                Graphics g = Graphics.FromImage(argb32bp);
                System.Drawing.Point[] point = new System.Drawing.Point[3];
                point[0] = new System.Drawing.Point(5, 5);
                point[1] = new System.Drawing.Point(ImageWidth - 5, 5);
                point[2] = new System.Drawing.Point(5, ImageHeight - 5);
                g.FillRectangle(System.Drawing.Brushes.White, new System.Drawing.Rectangle(0, 0, ImageWidth, ImageHeight));
                g.DrawImage(bitmap, point, new System.Drawing.Rectangle(rec[0], rec[1], rec[2], rec[3]), GraphicsUnit.Pixel);
                argb32bp.SetResolution(120, 120);
                if (savelocal)
                {
                    String filename = DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".png";
                    bitmap.Save(String.Format(@"{0}\{1}", System.Windows.Forms.Application.StartupPath, filename), System.Drawing.Imaging.ImageFormat.Png);
                }
                bitmap.Dispose();
                return argb32bp;
            }
            catch (Exception ex)
            {
                return null;
            }
        }

       上述方法需要添加的引用:System.Drawing、System.IO,以及一个名为zxing的动态库。(zxing的下载链接:http://pan.baidu.com/s/1c2G39IS)

2、在上面的方法中,程序端已经实现了向打印模板提供所需的数据,接下来需要修改打印模板,接收这些新添加的数据

        这里有一个简单的方法,能够为FastReport的数据源中添加一张表。那就是不直接双击打开打印模板,而是在打印模板的文件上右击,选择打开方式为“记事本”。在记事本中找到<Dictionary></Dictionary>这部分,在里面添加数据源和参数。以本项目的结账收款单打印模板为例,添加了一张用于显示二维码的表“微信数据”,还添加了一个用于确定是否需要打印二维码的参数“微信二维码”。

        因此,需要在<Dictionary></Dictionary>中添加如下内容

    <TableDataSource Name="微信数据" ReferenceName="Data.微信数据" DataType="System.Int32" Enabled="true">
      <Column Name="二维码" DataType="System.Byte[]"/>
    </TableDataSource>
    <Parameter Name="微信二维码" DataType="System.String"/>

        保存并关闭打印模板的记事本文件。再双击打开打印模板,此时在FastReport的右侧边栏中,能看到数据源里已经多了一个“微信数据”表,如下图所示:

使用FastReport打印二维码_第5张图片

       这里的“二维码”就是步骤1中代码里绘制出的二维码图形,检查这个字段的属性,需要确保它的DataType是Byte[]型的。并且在下面的“参数”列表中,也能看到多了一个“微信二维码”参数。

       为打印模板新建一个数据区,用于打印二维码。点击“报表”——“设置报表栏”菜单,在打开的窗体中点击“添加“按钮,添加”数据区”。将新添加的数据区重命名为Data_PictureCode。主要步骤如下图所示:

                                                              

       为什么要新建一个数据区,而不是在原来的数据区里添加二维码呢?因为当不需要打印二维码,也就是“微信二维码”参数为空时,我们可以直接把这个数据区隐藏,从而不用在打印出的小票上显示出空白的一大片来,也为客户节省纸张。

       在新添加的数据区中插入图片,如下图所示:

                                        使用FastReport打印二维码_第6张图片      使用FastReport打印二维码_第7张图片

       双击图片,在图片编辑器的“数据列”中,选择“微信数据”表中的“二维码”。如下图所示:

使用FastReport打印二维码_第8张图片

        最后,设置是否需要显示二维码所在的这个数据区。选择一个一定会打印的数据区,在这个数据区的BeforePrint事件中进行控制。首先为所选的数据区添加事件,在FastReport的右侧边栏中,进入到数据区的事件列表(点击那个闪电形状的按钮),在BeforePrint事件后面的编辑框内双击,即可为该数据区添加一个BeforePrint事件。该事件中的代码在打印数据区前执行,如下图所示:

使用FastReport打印二维码_第9张图片

        在所选数据区的BeforePrint事件中添加如下代码,控制是否显示二维码所在的那个数据区

    private void Data10_BeforePrint(object sender, EventArgs e)
    {
      //获取微信二维码url
      string codeURL=(String)Report.GetParameterValue("微信二维码");
      
      //若微信二维码url为空,则不显示二维码数据区
      if(codeURL=="")
      {
        Data_PictureCode.Visible=false;
      }
    }

        代码和打印模板设计界面的切换按钮,在FastReport的左下角,见下图所示:

        在进行了上述全部修改后,本项目的结账收款单上就能够打印出用于微信支付的二维码了。

      

      

       

你可能感兴趣的:(使用FastReport打印二维码)