首页 > 编程 > C# > 正文

基于Silverlight打印的使用详解,是否为微软的Bug问题

2020-01-24 03:19:58
字体:
来源:转载
供稿:网友

1:新建Silverlight4 应用程序,名称为SLStudy。建好后如下:

image

2:在SLStudy下新建Silverlight用户控件,Print1.xaml作为要打印的控件。

在Print1.xaml里面添加代码为:

复制代码 代码如下:

<Grid x:Name="LayoutRoot" Background="White">
      <Button>这是第一个例子,简单的按钮</Button>
</Grid>

3:已经建立好了要打印的内容了,这里打印的是一个按钮。

4:修改MainPage.xaml代码如下:

复制代码 代码如下:

<Grid x:Name="LayoutRoot" Background="White">
        <StackPanel>
            <Button x:Name="btnPrint1" Click="btnPrint1_Click">Print1</Button>
        </StackPanel>
</Grid>

5:后台代码为:
复制代码 代码如下:

private void btnPrint1_Click(object sender, RoutedEventArgs e)
        {
            PrintDocument printDocument = new PrintDocument();
            printDocument.PrintPage += new EventHandler<PrintPageEventArgs>(printDocument_PrintPage);
            PrintDocument.Print("要打印的文档的名称,这个可以随便设置");
        }

        void printDocument_PrintPage(object sender, PrintPageEventArgs e)
        {
            e.PageVisual = new Print1();
        }




在SL4 中提供打印功能的是PrintDocument类,所以先实例化一个该类对象。

接着注册一个PrintPage事件,PrintPage事件在打印的时候会触发。

然后调用printDocument的Print方法来打印。

在PrintDocument的PrintPage事件中,PrintPageEventArgs,是打印的参数。

里面可以获取当前打印机的一些信息。

image

在这里设置PageVisual,也就是要打印的对象就可以了。

复制代码 代码如下:

void printDocument_PrintPage(object sender, PrintPageEventArgs e)
{
  e.PageVisual = new Print1();
}

全部写好后,可以运行应用程序,点击Print1,弹出打印窗口。打印效果如下图:

image

当然我们的打印需求不可能这么简单,也许需要设置Print1的内容。假设我们要修改按钮显示的字,那么我们可以这样:

复制代码 代码如下:

void printDocument_PrintPage(object sender, PrintPageEventArgs e)
{
    Print1 printVisual = new Print1();
    printVisual.btnSample.Content = "修改后的值,当然也可以从数据库中获取";
    e.PageVisual = printVisual;
}

通过设置PrintPageEventArgs 参数的PageVisual对象,我们就可以实现打印那个页面的功能了。

在这里我总结下:

1:确定要打印的内容,然后新建一个UserControl来显示打印的内容。

2:新建PrintDocument对象,注册PrintPage事件,调用Print方法。

3:在PrintPage事件中,构造要打印的对象,然后去数据库中获取数据,然后把数据绑定到控件上,接着把绑定好数据的控件赋值给PrintPageEventArgs的PageVisual 对象。
 

多页打印问题:

如果要打印的只有一张,那么这种方法应该就够了,但是有时候需要将一份文档打印多张,

比如将上面的按钮打印5张,那么又该如何实现了。

还记得我们上面PrintPageEventArgs的HasMorePages参数吗?


在PrintPage 事件触发后,默认的HasMorePages 为false。将HasMorePages设置为true,可以让PrintPage事件不断被触发。当 HasMorePages 属性为 true,PrintPage 事件将多次发生,直到 HasMorePages 为 false。


假设我们要将上面的按钮打印5张,那么可以设置4次HasMorePages为true,最后设置HasMorePages为false就可以了。

修改后的printDocument_PrintPage 方法如下:

复制代码 代码如下:

int count = 5;
int printCount = 0;
void printDocument_PrintPage(object sender, PrintPageEventArgs e)
{
    Print1 printVisual = new Print1();
    printVisual.btnSample.Content = "修改后的值,当然也可以从数据库中获取";
    e.PageVisual = printVisual;
    printCount++;

    if (printCount < count) //如果已经打印的页数小于要打印的页数,说明还需要打印。
    {
         e.HasMorePages = true;
    }
    else
    {
        e.HasMorePages = false;
    }
}


有时候需要知道当前打印的是第几页,这可以通过查询printDocument.PrintedPageCount 属性来获得,

在PrintDocument_PrintPage 方法中,sender对象其实就是PrintDocument对象,所以我们可以将它强制类型转换。

假设我们要将上面的5个 Button的内容都修改为1,2,3,4,5.那么我们可以修改代码为:

复制代码 代码如下:

       int count = 5;
        int printCount = 0;

        void printDocument_PrintPage(object sender, PrintPageEventArgs e)
        {
            PrintDocument printDocument = sender as PrintDocument;
            Print1 printVisual = new Print1();
            printVisual.btnSample.Content = string.Format("按钮{0}", printDocument.PrintedPageCount);
            e.PageVisual = printVisual;
            printCount++;

            if (printCount < count)
            {
                e.HasMorePages = true;
            }
            else
            {
                e.HasMorePages = false;
            }
        }


实际上,我们的printCount变量都不需要了,直接使用printDocument.PrintedPageCount 就可以了,具体代码实现由读者自己实现吧。

微软的Bug??

如果你的打印机设置为

image

那么打印的结果就是*.xps的文件,但是在打印的过程中会弹出提示框,询问保存地址。

如果你在PrintPage事件中打上断点的话,可以看到在询问保存地址的时候,PrintPage方法已经执行了,也就是说PringPage方法会被执行两遍,第一遍并没有真正的打印。

例如:

image

image 

如果在上图的界面上点击取消,则有可能会导致系统失去响应而卡死,

假设用户点击保存,那么PrintPage事件会再次的触发。

但是由于已经打印了一次了,所以有可能导致在多页打印的时候出现问题。

使用两个标志变量可以解决这个问题。

例如修改代码为:

复制代码 代码如下:

int count = 5;
        int printCount = 0;

        /// <summary>
        /// 是否是第一次打印,因为只有第二次打印的时候才开始真正的打印。
        /// </summary>
        private bool isInitialized = false;
        private bool realPrint = false;

        void printDocument_PrintPage(object sender, PrintPageEventArgs e)
        {
            PrintDocument printDocument = sender as PrintDocument;
            int currentPage = printDocument.PrintedPageCount;

            #region 因为要经过两次,第一次是初始化,而第二次才是真正的打印,而两次PrintedPageCount都是0
            if (currentPage == 0)
            {
                if (isInitialized) //如果已经初始化,则设置realPrint为true
                {
                    realPrint = true;
                }

                isInitialized = true; //运行到这里,说明已经初始化了。
            }
            #endregion

            if (realPrint)
            {
                //PrintDocument printDocument = sender as PrintDocument;

                Print1 printVisual = new Print1();
                printVisual.btnSample.Content = string.Format("按钮{0}", printDocument.PrintedPageCount);

                e.PageVisual = printVisual;

                printCount++;
                if (printCount < count)
                {
                    e.HasMorePages = true;
                }
                else
                {
                    e.HasMorePages = false;
                }
            }
        }

因为两次打印,第一次可以被认为是初始化,第二次可以被认为是打印机开始真正的打印,

所以可以使用两个变量isInitialized 和realPrint 来分别表示是初始化还是真实的打印。

在执行第一遍的时候printDocument.PrintedPageCount ==0,在这时候将isInitialized 设置为true。

在执行第二遍的时候,因为isInitialized ==true,所以可以将realPrint设置为true。

在后面的代码中只需要判断realPrint为true就可以了。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表