在使用WPF时,如何用C#无声打印HTML文件

5

编辑:完全重写问题并添加悬赏。

根据许多教程和stackoverflow的问题,我现在可以:

  • 将多个页面打印成一个文档。
  • 打印实际内容。
  • 正确对齐页面上的内容。
  • 打印正确大小的文档。

解决方案需要HTML文档底部有一些空白和样式标签html{ overflow:hidden; } - 隐藏滚动条并允许使用滚动进行分页 - 但我可以接受这一点。

唯一剩下的问题是WPF不会呈现WebBrowser屏幕外的部分。

这意味着如果我倾斜电脑屏幕,我可以正确打印,但如果我不这样做,文档会切掉下面的部分。

我尝试了将其渲染为位图,但当我将结果图像作为我的可视化内容打印时,页面为空。

如果您知道如何强制WPF进行完整呈现或如何正确渲染位图,请帮助我。

打印窗口XAML:(仅在UI线程上打印WPF才能正常工作,否则不会呈现任何内容...)

<Window x:Class="CardLoader2000.PrintWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="PrintWindow" Height="1139" Width="820">
    <Grid x:Name="grid">
        <!--The alignment and size of the webbrowser is reflected in the print. If larger than document
        it will be cut. The width here corresponds to A4 paper width with a little margin-->
        <WebBrowser x:Name="webBrowser" Height="1089" Width="770" VerticalAlignment="Top" Margin="0,10,0,0"/>
    </Grid>
</Window>

打印窗口代码后台:

public partial class PrintWindow : Window
    {
        public PrintWindow(string path)
        {
            InitializeComponent();

            FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read);

            webBrowser.NavigateToStream(fs);

            ContentRendered += OnContentRendered;
        }

        private void OnContentRendered(object sender, EventArgs eventArgs)
        {
            PrintDialog pd = new PrintDialog
            {
                PrintTicket = new PrintTicket
                {
                    Duplexing = Duplexing.TwoSidedLongEdge,
                    OutputColor = OutputColor.Monochrome,
                    PageOrientation = PageOrientation.Portrait,
                    PageMediaSize = new PageMediaSize(794, 1122),
                    InputBin = InputBin.AutoSelect
                }
            };

            //Ok, final TODO: Page only renders what is on the PC screen...
            WebPaginator paginator = new WebPaginator(webBrowser, 1089, 1122, 794);

            pd.PrintDocument(paginator, "CustomerLetter");

            Close();
        }
    }

自定义分页器:

public class WebPaginator : DocumentPaginator
    {
        private readonly WebBrowser webBrowser;
        private readonly int pageScroll;
        private Size pageSize;

        public WebPaginator(WebBrowser webBrowser, int pageScroll, double pageHeight, double pageWidth)
        {
            this.webBrowser = webBrowser;
            this.pageScroll = pageScroll;
            pageSize = new Size(pageWidth, pageHeight);
        }

        public override DocumentPage GetPage(int pageNumber)
        {
            HTMLDocument htmlDoc = webBrowser.Document as HTMLDocument;
            if (htmlDoc != null) htmlDoc.parentWindow.scrollTo(0, pageScroll * pageNumber);
            Rect area = new Rect(pageSize);

            return new DocumentPage(webBrowser, pageSize, area, area);
        }

        public override bool IsPageCountValid
        {
            get { return true; }
        }

        /// <summary>
        /// Returns one less than actual length.
        /// Last page should be whitespace, used for scrolling.
        /// </summary>
        public override int PageCount
        {
            get
            {
                var doc = (IHTMLDocument2)webBrowser.Document;
                var height = ((IHTMLElement2)doc.body).scrollHeight;
                int tempVal = height*10/pageScroll;
                tempVal = tempVal%10 == 0
                    ? Math.Max(height/pageScroll, 1)
                    : height/pageScroll + 1;
                return tempVal > 1 ? tempVal-1 : tempVal;
            }
        }

        public override Size PageSize
        {
            get
            {
                return pageSize;
            }
            set
            {
                pageSize = value;
            }
        }

        /// <summary>
        /// Can be null.
        /// </summary>
        public override IDocumentPaginatorSource Source
        {
            get
            {
                return null;
            }
        }
    }

请尝试此链接。https://social.msdn.microsoft.com/Forums/vstudio/en-US/63ae5345-b2ca-45a9-9113-0ddc43e9925b/print-functionality-dialog-less-using-wpf-webbrowser-control?forum=wpf - Ayyappan Subramanian
使用IHTMLDocument2打印也会得到一个空文档,并且始终提示用户,即使在execCommand调用中将ShowUI设置为false。 - Martin Clemens Bloch
从UI直接打印是否绝对必要?我的意思是从WebBrowser打印吗?您考虑过将相同的URL独立传递给更适合此目的的工具吗?比如http://www.princexml.com/doc/dotnet/?最终从PDF打印将在打印时产生更一致的结果... - Daniel C
好吧,我开始意识到使用WebBrowser控件的方法根本行不通。它不是一个真正的WPF控件,而是一个包装的COM控件。这意味着它不会按照你的要求执行操作。 我设法让它认为屏幕比实际更大,但由于它是一个COM控件,它没有渲染 - 只有黑色。 我还尝试了RenderRotateTransform,但控件中的内容仍然未旋转。我会考虑您提供的链接/解决方案,但我有点厌倦了尝试随机的东西。我可能会完全改变我的方法。 - Martin Clemens Bloch
我无法解决这个问题,它涉及到 IHTMLDocument2。我正在使用核心 5.0。 - Marinpietri
1个回答

9
您可以使用标准的IE打印功能(通过ExecWB方法),如下所示:
public partial class PrintWindow : Window
{
    public PrintWindow()
    {
        InitializeComponent();
        webBrowser.Navigate("http://www.google.com");
    }

    // I have added a button to demonstrate
    private void Button_Click(object sender, RoutedEventArgs e)
    {
        // NOTE: this works only when the document as been loaded
        IOleServiceProvider sp = webBrowser.Document as IOleServiceProvider;
        if (sp != null)
        {
            Guid IID_IWebBrowserApp = new Guid("0002DF05-0000-0000-C000-000000000046");
            Guid IID_IWebBrowser2 = new Guid("D30C1661-CDAF-11d0-8A3E-00C04FC9E26E");
            const int OLECMDID_PRINT = 6;
            const int OLECMDEXECOPT_DONTPROMPTUSER = 2;

            dynamic wb; // will be of IWebBrowser2 type, but dynamic is cool
            sp.QueryService(IID_IWebBrowserApp, IID_IWebBrowser2, out wb);
            if (wb != null)
            {
                // note: this will send to the default printer, if any
                wb.ExecWB(OLECMDID_PRINT, OLECMDEXECOPT_DONTPROMPTUSER, null, null);
            }
        }
    }

    [ComImport, Guid("6D5140C1-7436-11CE-8034-00AA006009FA"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
    private interface IOleServiceProvider
    {
        [PreserveSig]
        int QueryService([MarshalAs(UnmanagedType.LPStruct)] Guid guidService, [MarshalAs(UnmanagedType.LPStruct)]  Guid riid, [MarshalAs(UnmanagedType.IDispatch)] out object ppvObject);
    }
}

我的点赞,我认为这是一个很好的答案(根据您的说明,可以直接从webBrowser_LoadCompleted调用它)。在这种情况下,我也建议使用IE的打印功能,即使我个人更喜欢使用工具(如SharpDevelopReporting)并从PDF中打印(如上面的评论中已经写过)。 - user4569980
这个可以运行,但是它会打印双面吗?(我现在没有打印机的访问权限,只有XPS文件打印机) - Martin Clemens Bloch
它将使用默认打印机和默认设置进行打印,无论它们是什么。 - Simon Mourier
没问题,直接使用即可。我复制粘贴了这段代码并检查了打印机默认设置,没有问题 :) - Martin Clemens Bloch
有没有这样的答案,其中默认打印机无法更改,所需打印机不是默认打印机? - Slate
显示剩余4条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接