1
votes

I am trying to find a good way to print a flow document in WPF. What I want is to have a possibility to see how the document turns out as I design it, so therefore creating a pure FlowDocument as a XAML is out of the questions (as Visual Studio wont show the design view for it).

So what I have done now is to create a window that contains a FlowDocument like this (some excessive parts have been removed to make the code more consise):

<Window x:Class="MyNamespace.ProjectPrintout...>
  <Grid>
    <FlowDocumentReader>
      <FlowDocument ColumnWidth="500" Name="Document">
        <!-- Header -->
        <Paragraph Name="HeaderText">
          The header will go here
        </Paragraph>
      </FlowDocument>
    </FlowDocumentReader>
  </Grid>
</Window>

This is a bit strange since I will never show this Window to the user, and I only wrap the FlowDocument with a Window so that I can see how it looks like as I develop it. This Ican live with.

So somewhere else in my application, I want to print this FlowDocument to the default printer, but I also have to set the header dynamically (in addition to many other parts of the documents that needs dynamic data that are omitted here).

The code to print looks like this:

  var printout = new ProjectPrintout();
  printout.HeaderText= new Paragraph(new Run("Proper header text"));
  var document = printout.Document;

  var pd = new PrintDialog();
  IDocumentPaginatorSource dps = document;
  pd.PrintDocument(dps.DocumentPaginator, "Document");

The document is getting printed, and looks fine except that the header text still shows "The header will go here", even if I replaced it from my code with "Proper header text". I also tried changing it like this:

(printout.HeaderText.Inlines.FirstInline as Run).Text = "Proper header text";

But the result is the same.

So the question is: How can I change the contents in the FlowDocument from code before I print it, or are there a better way to do this instead of my approach?

1
Actually with (printout.HeaderText.Inlines.FirstInline as Run).Text = "Proper header text"; it works for meKlaus78
@Klaus78 - I tested now, and it seems I had both lines when I tested. So first ...HeaderText = new Paragraph... and then the line you mentions. Then it does not work. If I take only the line you mentions, then it works for me as well.Øyvind Bråthen
Any idea why the first one does not work? It seems like setting HeaderText to a new Paragraph changes the object HeaderText references to the new object, but does not affect the object in the FlowDocument sections. Quite ugly code, but then I might get it to work at least. Thanks :)Øyvind Bråthen
I agree. It looks like HeaderText is a new Paragraph that you need to include in the FlowDocument (yourFlowDoc.Blocks.Add(HeaderText)) if you want it to appear.Klaus78

1 Answers

4
votes

MVVM to the rescue:

Epiphany: UI is not Data. UI is not a data store. UI is meant to show Data, not to store it.

1 - Create a simple object to hold your data

public class MyDocumentViewModel: INotifyPropertyChanged //Or whatever viewmodel base class
{
    private string _header;
    public string Header 
    {
        get { return _header; }
        set
        {
            _header = value;
            NotifyPropertyChange(() => Header);
         }
     }

     //Whatever other data you need
}

2 - Define Bindings in your Document;

<Paragraph>
    <Run Text="{Binding Header}"/>
</Paragraph>

3 - Set your FlowDocument's DataContext to an instance of this class:

var flowdoc = new YourFlowDocument();
var data = new MyDocumentViewModel { Header = "this is the Header" };
//whatever other data

flowdoc.DataContext = data;

//do the printing stuff.