What's the best approach to printing/reporting from WPF?
Solution 1
We had this same issue, and ended up using RDLC/ReportViewer for now. There's no native WPF reporting tool (that I know of) and RDLC is pretty simple to use, and is free. The runtime overhead for it is small (around 2Mb) but you must remember to distribute it as it isn't part of the .NET Framework.
Solution 2
Limitations of RDL
I originally went with RDLC/ReportViewer for printing with WPF but found it very limiting. Some of the limitations I found were:
- RDL could only create the most boring of reports
- It was much more work to create a report using RDL than in straight WPF: The design tools are very primitive compared to Expression Blend and RDL deals only in tables
- I didn't have the ability to use ControlTemplates, DataTemplates, Styles, etc
- My report fields and columns could not effectively resize and rearrange based on data size
- Graphics had to be imported as images - it could not be drawn or edited as vectors
- Positioning of items required code-behind rather than data binding
- Lack of transforms
- Very primitive data binding
Printing directly from WPF is very easy
Because of these limitations I looked into creating reports using pure WPF and discovered it was really quite trivial. WPF allows you to implement your own DocumentPaginator
subclass that can generate pages.
I developed a simple DocumentPaginator subclass that takes any Visual, analyzes the visual tree, and hides selected elements to create each page.
DocumentPaginator details
Here is what my DocumentPaginator subclass does during initialization (called when first PageCount is fetched, or during the first GetPage() call):
- Scans the visual tree and makes a map of all scrolled panels inside ItemsControls
- Starting with the outermost, makes items in the ItemsControls invisible last to first until the Visual fits on a single page without any need to scroll. If the outermost can't be reduced enough, reduces inner panels until it succeeds or has only one item at each level. Record the set of visible items as the first page.
- Hide the lowest-level items that have already been shown on the first page, then make subsequent items visible until they no longer fit on the page. Record all but the last-added item as the second page.
- Repeat the process for all pages, storing the results in a data structure.
My DocumentPaginator's GetPage method is as follows:
- Look up the given page number in the data structure generated during initialization
- Hide and show items in the visual tree as indicated in the data structure
- Set PageNumber and NumberOfPages attached properties so report can display page numbering
- Flush the Dispatcher (
Dispatcher.BeginInvoke(DispatcherPriority.ApplicationIdle, new Action(() => {} ));
) to get any background rendering tasks to complete - Create a Rectangle the size of the page whose VisualBrush is the visual being printed
- Measure, Arrange, and UpdateLayout the rectangle, then return it
This turned out to be quite simple code, and allowed me to turn practically anything I could create with WPF into pages and print it.
Additional reporting support
Now that my paginator is working, I no longer have to worry very much about whether I am creating my WPF content for screen or paper. In fact, often UI that I build for data entry and editing also works very well for printing.
From there I added a simple toolbar and some code behind, resulting in a full-fledged reporting system built around WPF that was far more capable than RDL. My reporting code can export to files, print to the printer, cut/paste page images, and cut/paste data for Excel. I can also switch any of my UI to "print view" with a click of a checkbox to see what it will look like if printed. All this in just a few hundred lines of C# and XAML!
At this point I think the only feature RDL has that my reporting code doesn't have is the ability to generate a formatted Excel spreadsheet. I can see how this could be done but so far there has been no need - cutting and pasting the data alone has been enough.
From my experience my recommendation would be to write a paginator, then start using WPF itself to create your reports.
Solution 3
Look at http://wpfreports.codeplex.com/
Solution 4
Take a look at PdfReports. It's a code first reporting engine, which is built on top of the iTextSharp and EPPlus libraries. It's compatible with both .NET 3.5+ Web and Windows applications.
Solution 5
How about Scryber? It allows PDF report templates to be defined using xml and bound to data within your application at run time. http://scryber.codeplex.com/
Matt Hamilton
Follow me on Twitter! Check out my hobby projects: Comicster A free, open-source WPF application to manage your comic-book collection. Halfwit A minimalist, open-source, WPF Twitter client. Budgie A simple, asynchronous Twitter library for .NET 4 and up. MadProps.AppArgs A library to parse command-line or ClickOnce-URI arguments into an instance of a class. MadProps.MvvmLight A contrib project to add co-routine support to the MVVM Light Toolkit. Feel free to contact me directly (mabster -at- madprops.org).@mabsterGoogle+
Updated on July 16, 2022Comments
-
Matt Hamilton almost 2 years
I have an upcoming project which will have to be able to print simple reports from its data. It'll be WPF-based, and I'm wondering which way to go.
I know that WPF introduces its own printing technology (based on XPS) which looks quite easy to use. However, part of me wonders whether it would just be easier to use the ReportViewer control and embed it in a Windows Forms host control, since that will give users the ability to export to a variety of formats as well as print.
Has anyone had any experience with printing/reporting from WPF? Which direction would you recommend?
-
Eduardo Molteni over 13 yearsVery interesting as I'm thinking of dropping RDLC. Have you published any of the code of your document paginator?
-
Alex Hope O'Connor over 11 yearsCould you expand on this at all? Perhaps with some short code/xaml examples?
-
Alex Hope O'Connor over 11 yearsFound this: codeproject.com/Articles/138233/…
-
heltonbiker over 11 yearsThis solution of yours is a perfectly fit candidate for OpenSourcing, which could be something that would bring back benefits to yourself and your company (at the cost of some extra initial work, we know...)
-
Donvini over 11 years"Printing directly from WPF is very easy". Is it really? I've already put a lot of time and effort into implementing this and paginating a databound datagrid is not easy by any stretch of the imagination. I'm afraid the devil's in the details and until I see some, this answer is not very useful.
-
WiiMaxx over 10 years@RayBurns would you like to share your code (just the part with the visibility would be enough)? I have a bounty on a Question where your answer would fit very well
-
Killnine over 10 yearsHere, here, @RayBurns. Let's see a github repo so we can help contribute to this awesome-sounding implementation.
-
WiiMaxx almost 10 yearsi found an artikel on nullskull which explains step by step how to create a WPF Report Engine using the same approach.
-
Gopichandar about 8 yearsInteresting, but it has been 6 years from the answer, still couldn't find any
class
that help me to convertxaml
view to page. Also, there is no other way :( -
Mike Marynowski over 6 yearsI have taken a similar approach to this but instead of messing around with the visual tree I use a view model that has
Items
andCurrentPageItems
where items are added until the page overflows then removes one item. The view model approach lets you do things like get page subtotals very easily and can have additional properties like IsLastPage which you can use to toggle a footer and such, making it very powerful. I will probably write some blog posts about this and post some code in the near future but that's the general approach I would suggest if you are thinking of doing this -
juFo over 5 yearsiTextSharp licenses :(
-
VahidN over 5 yearsThere's a .NET Core/Full .NET version of that library too, with an LGPL license: github.com/VahidN/PdfReport.Core
-
Lucy82 about 4 yearsIt's definitely not an easy task. I also tried to do this, but there are so many things to build that you ussually end up dropping everything. Styling DocumentViewer, code for exporting to pdf/excel/Word, building custom paginator... Just a few, not to mention dynamic values, grouping data, repeating elements like table header etc.... Hard and long work!
-
UяošKoт over 3 yearsLooks nice, but on 2020/11/11 project description still says "This is a very early alpha version not intented to be used in production environments."