"Fast"在文档查看器中显示



我正在构建一个WPF应用程序,我需要在其中显示文档预览,例如使用DocumentViewer和DocumentPaginator可以实现的功能。然而,将报告转换为XPS并将其加载到DocumentViewer中被证明是非常慢的,当报告很大时(就像我需要显示的普通报告一样)。

这让我开始思考,可能有一些方法可以开始显示报告的前几页,而其余的页面正在被"加载"到DocumentViewer中——基本上加载/显示页面,因为它们被创建

有人知道这样的事情是可能的吗?如果是的话,你觉得我该如何开始尝试?我花了几个小时在网上寻找一个更快显示报告的解决方案,但还没有提出任何解决方案。

为了完全公开,在本例中,我需要显示的报告是在HTML中创建的。我知道我需要将其转换为XPS才能使用DocumentViewer,但我提出这个问题是因为如果有人有快速显示HTML的方法,请随时提出。我不能使用WebBrowser控件,因为我必须在"打印预览"类型的模式下显示。一个好的算法来决定如何"分页"一个HTML网站可能会导致我解决这个问题,以及然后我可以创建一个自定义控件来显示它。我会使用DocumentPaginator,但随后输出的文件是XPS,然后我回到DocumentViewer问题。

再一次,任何帮助都是非常感谢的。谢谢你!

好的,我想我有办法了。

我又找到了一个更好的URL来参考。这个不能直接为我加载,所以我从谷歌缓存中抓取:http://webcache.googleusercontent.com/search?q=cache:LgceMCkJBrsJ:joshclose.net/%3Fp%3D247

定义每篇文章中描述的IViewObject接口:

    [ComVisible(true), ComImport()]
    [GuidAttribute("0000010d-0000-0000-C000-000000000046")]
    [InterfaceTypeAttribute(ComInterfaceType.InterfaceIsIUnknown)]
    public interface IViewObject
    {
        [return: MarshalAs(UnmanagedType.I4)]
        [PreserveSig]
        int Draw(
            [MarshalAs(UnmanagedType.U4)] UInt32 dwDrawAspect,
            int lindex,
            IntPtr pvAspect,
            [In] IntPtr ptd,
            IntPtr hdcTargetDev,
            IntPtr hdcDraw,
            [MarshalAs(UnmanagedType.Struct)] ref Rectangle lprcBounds,
            [MarshalAs(UnmanagedType.Struct)] ref Rectangle lprcWBounds,
            IntPtr pfnContinue,
            [MarshalAs(UnmanagedType.U4)] UInt32 dwContinue);
        [PreserveSig]
        int GetColorSet([In, MarshalAs(UnmanagedType.U4)] int dwDrawAspect,
           int lindex, IntPtr pvAspect, [In] IntPtr ptd,
            IntPtr hicTargetDev, [Out] IntPtr ppColorSet);
        [PreserveSig]
        int Freeze([In, MarshalAs(UnmanagedType.U4)] int dwDrawAspect,
                        int lindex, IntPtr pvAspect, [Out] IntPtr pdwFreeze);
        [PreserveSig]
        int Unfreeze([In, MarshalAs(UnmanagedType.U4)] int dwFreeze);
        void SetAdvise([In, MarshalAs(UnmanagedType.U4)] int aspects,
          [In, MarshalAs(UnmanagedType.U4)] int advf,
          [In, MarshalAs(UnmanagedType.Interface)] IAdviseSink pAdvSink);
        void GetAdvise([In, Out, MarshalAs(UnmanagedType.LPArray)] int[] paspects,
          [In, Out, MarshalAs(UnmanagedType.LPArray)] int[] advf,
          [In, Out, MarshalAs(UnmanagedType.LPArray)] IAdviseSink[] pAdvSink);
    }

创建一个HtmlPaginator类,用来截屏浏览器的文档(如上所述),然后将其裁剪为页面/框架:

    class HtmlPaginator
    {
        public event EventHandler<PageImageEventArgs> PageReady;
        protected virtual void OnPageReady(PageImageEventArgs e)
        {
            EventHandler<PageImageEventArgs> handler = this.PageReady;
            if (handler != null)
                handler(this, e);
        }
        public class PageImageEventArgs : EventArgs
        {
            public Image PageImage { get; set; }
            public int PageNumber { get; set; }
        }
        public void GeneratePages(string doc)
        {
            Bitmap htmlImage = RenderHtmlToBitmap(doc);
            int pageWidth = 800;
            int pageHeight = 600;
            int xLoc = 0;
            int yLoc = 0;
            int pages = 0;
            do
            {
                int remainingHeightOrPageHeight = Math.Min(htmlImage.Height - yLoc, pageHeight);
                int remainingWidthOrPageWidth = Math.Min(htmlImage.Width - xLoc, pageWidth);
                Rectangle cropFrame = new Rectangle(xLoc, yLoc, remainingWidthOrPageWidth, remainingHeightOrPageHeight);
                Bitmap page = htmlImage.Clone(cropFrame, htmlImage.PixelFormat);
                pages++;
                PageImageEventArgs args = new PageImageEventArgs { PageImage = page, PageNumber = pages };
                OnPageReady(args);
                yLoc += pageHeight;
                if (yLoc > htmlImage.Height)
                {
                    xLoc += pageWidth;
                    if (xLoc < htmlImage.Width)
                    {
                        yLoc = 0;
                    }
                }
            } 
            while (yLoc < htmlImage.Height && xLoc < htmlImage.Width);
        }
        private static Bitmap RenderHtmlToBitmap(string doc)
        {
            Bitmap htmlImage = null;
            using (var webBrowser = new WebBrowser())
            {
                webBrowser.ScrollBarsEnabled = false;
                webBrowser.ScriptErrorsSuppressed = true;
                webBrowser.DocumentText = doc;
                while (webBrowser.ReadyState != WebBrowserReadyState.Complete)
                {
                    Application.DoEvents();
                }
                webBrowser.Width = webBrowser.Document.Body.ScrollRectangle.Width;
                webBrowser.Height = webBrowser.Document.Body.ScrollRectangle.Height;
                htmlImage = new Bitmap(webBrowser.Width, webBrowser.Height);
                using (Graphics graphics = Graphics.FromImage(htmlImage))
                {
                    var hdc = graphics.GetHdc();
                    var rect1 = new Rectangle(0, 0, webBrowser.Width, webBrowser.Height);
                    var rect2 = new Rectangle(0, 0, webBrowser.Width, webBrowser.Height);
                    var viewObject = (IViewObject)webBrowser.Document.DomDocument;
                    viewObject.Draw(1, -1, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, hdc, ref rect1, ref rect2, IntPtr.Zero, 0);
                    graphics.ReleaseHdc(hdc);
                }
            }
            return htmlImage;
        }
    }

这样写:

        WebBrowser browser = new WebBrowser();
        browser.Navigate("http://www.stackoverflow.com");
        while (browser.ReadyState != WebBrowserReadyState.Complete)
        {
            Application.DoEvents();
        }
        HtmlPaginator pagr = new HtmlPaginator();
        pagr.PageReady += new EventHandler<HtmlPaginator.PageImageEventArgs>(pagr_PageReady);
        pagr.GeneratePages(browser.DocumentText);
为了测试它,我实现了一个带有按钮、图片框和List集合的基本表单。我将页面添加到集合中,因为它们已经从HtmlPaginator中准备好了,并使用按钮将下一个图像添加到图片框中。

神奇的数字是你想要的宽度和高度。我用的是800x600,但你可能想要不同的尺寸。

这里的缺点是你仍然在等待WebBrowser渲染HTML,但我真的看不出有什么替代解决方案可以减少这个时间——首先必须有一些东西来解释和绘制HTML。写你自己的浏览器吧。:)

我试着玩IViewObject。画一下,看看我是否可以让它直接渲染页面框架,而不是有裁剪循环,但它不适合我。

相关内容

  • 没有找到相关文章

最新更新