July 12th, 2007

Using the PrintForm component in Visual Basic (Matt Gertz)

(This is the fourth and final part of the Paint-by-Numbers series)

Late last year, some clever guys on our Visual Basic team released the PrintForm component on the web.  The idea behind the PrintForm component was to make printing and previewing your form very easy in .NET.  I’m going to leverage this component to enable printing of the Paint-by-Number puzzles created in the application I’ve been building in this series.

First of all, I need to download the component, which is available here on the Microsoft site – it’s about 500KB.  Once downloaded, I’ll install it (you can even do this while Visual Studio is open), and now I can add the PrintForm control in the component to my toolbox.  With my form selected, I right-click on the “Printing” category in the toolbox and select “Choose Items…”  In the resulting dialog, I’ll check the box next to “Print Form” on the .NET tab, and that’ll do it once I press “OK.” 

Now, I’ll drag an instance of the PrintForm control to the component tray and rename it to be “VBPBNPrintForm.” We don’t need to change any properties on it otherwise, so on the form, I’ll drop down the File menu item and double-click on “Print Preview” to generate its event handler for that command – now I can start coding!

The PrintForm command really just has one important verb – Print.  It can take several options, but overall it’s just a print job to either a printer, a file, or a preview window.  Which of those it does depends on the value you assign to its PrintAction property.  In this case, I want to just preview the document, and so I’ll need to set it to “PrintToPreview.”  The code looks like this:

    Private Sub PrintPreviewToolStripMenuItem_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) _

Handles PrintPreviewToolStripMenuItem.Click

            Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToPreview

            Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

    End Sub

 

Note that I’ve used an overload of Print which tells the control which form to print (which is my main form in this case), and also indicates that I just want to print the client area and not the window borders and so on.  However, if you run this, you’ll notice that the menu strip is printing in the preview!  This is because it’s in the client area.  Print doesn’t use document technology – it will in fact print anything visible in the client area, which works to our benefit when printing a puzzle solution (View/Show Solution is checked) or just the puzzle (View/Show Solution is unchecked), but not in this case.  So, for the duration of the printing job, I’ll simply hide the Menu, and I’ll wrap that in a try/catch/finally so that the menu will always get turned back on even if an exception is thrown from printing.  (This is hackery, plain and simple, but the benefits are enough to outweigh any sense of guilt I might feel about that.)  Now the code looks like this:

    Private Sub PrintPreviewToolStripMenuItem_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) _

Handles PrintPreviewToolStripMenuItem.Click

        Try

            Me.VBPBNMenuStrip.Visible = False

            Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToPreview

            Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

        Catch ex As Exception

        Finally

            Me.VBPBNMenuStrip.Visible = True

        End Try

    End Sub

 

and that will work nicely.  You can zoom in and out, resize the display, and even print to the default printer from the dialog that pops up. 

Now, for the actual Print command from the file menu, I could write very similar code (the one change is underlined below):

    Private Sub PrintToolStripMenuItem_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles PrintPreviewToolStripMenuItem.Click

        Try

            Me.VBPBNPrintForm.PrinterSettings = Me.VBPBNPrintDialog.PrinterSettings

            Me.VBPBNMenuStrip.Visible = False

            Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToPrinter

            Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

        Catch ex As Exception

        Finally

            Me.VBPBNMenuStrip.Visible = True

        End Try

    End Sub

 

But that would only allow us to print to the default printer, and I want to be a little more sophisticated.  I’ll drag out a PrintDialog control (which I’ll rename VBPBNPrintDialog) and use it to capture settings before doing the actual printing.  The PrintForm control uses the same PrinterSettings object that the PrintDialog does, which makes things very easy – I just assign one to the other before doing the actual printing.  Then, I check to see if we’re printing to a file or to a printer.  If printing to a printer, then I have everything I need; if a file, I’ll need to pop up a “save file” dialog to get the filename from the user first.  The code now looks like this:

    Private Sub PrintToolStripMenuItem_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) _

Handles PrintToolStripMenuItem.Click

        ‘ Bring up the print dialog:

        Dim res As DialogResult = Me.VBPBNPrintDialog.ShowDialog

        If res = Windows.Forms.DialogResult.OK Then

            Try

                ‘ Copy the settings from the dialog

                Me.VBPBNPrintForm.PrinterSettings = Me.VBPBNPrintDialog.PrinterSettings

 

                ‘ Turn off the menu strip

                Me.VBPBNMenuStrip.Visible = False

 

                If Me.VBPBNPrintForm.PrinterSettings.PrintToFile = True Then

                    ‘ The user wants to print to a file, so bring up a “Save” dialog and get the file name

                    Me.VBPBNSaveFileDialog.Title = My.Resources.TITLE_SavePrintFile

                    Me.VBPBNSaveFileDialog.Filter = My.Resources.FILT_PrintFilter

                    If Me.VBPBNSaveFileDialog.ShowDialog() = Windows.Forms.DialogResult.OK Then

                        ‘ Go ahead & print to the file

                        Me.VBPBNPrintForm.PrintFileName = Me.VBPBNSaveFileDialog.FileName

                        Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToFile

                        Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

                    End If

                Else

                    ‘ Print to whatever printer was selected

                    Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToPrinter

                    Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

                End If

            Catch ex As Exception

            Finally

                ‘ Make sure that the menu turns back on!

                Me.VBPBNMenuStrip.Visible = True

            End Try

        End If

    End Sub

 

and we’re done with the Print code.  (The resource FILT_PrintFilter is similar to the other filter I used for saving files in the previous post, but aimed at .EPS files instead of .VBPBN files, and TITLE_SavePrintFile is just a title for the window.)

But, hold on – the preview can also use the PrinterSettings object.  For example, if the user chose “Print,” clicked “Apply” after changing some settings, but then chose Cancel, the settings would persist for the next call to Print, so we’d expect those to apply to the preview as well (otherwise, it wouldn’t be much of a preview, since there’s no other way to do settings in a preview). So, we’ll add one more line to the Print Preview handler (underlined):

    Private Sub PrintPreviewToolStripMenuItem_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles PrintPreviewToolStripMenuItem.Click

        Try

            Me.VBPBNPrintForm.PrinterSettings = Me.VBPBNPrintDialog.PrinterSettings

            Me.VBPBNMenuStrip.Visible = False

            Me.VBPBNPrintForm.PrintAction = Printing.PrintAction.PrintToPreview

            Me.VBPBNPrintForm.Print(Me, PowerPacks.Printing.PrintForm.PrintOption.ClientAreaOnly)

        Catch ex As Exception

        Finally

            Me.VBPBNMenuStrip.Visible = True

        End Try

    End Sub

 

And that’s it! 

PrintForm is a great alternative to fiddling about with print documents if you don’t need to do a lot of reformatting for the printout you generate.  (In this case, all I had to do was hide a menu strip temporarily, which is much easier that defining a print document , etc.) The final application is attached to this post.  I hope you’ve enjoyed this series – it was a lot of fun to write, and already I’m thinking about the improvements that could be made, such as arbitrarily-sized grids, select/copy/paste of arbitrary bitmaps into and within the puzzle, creating a “puzzle viewer” to solve the puzzle on the PC rather than on paper, adding authoring information to the saved puzzle, supporting the red/black Paint-by-Number puzzles that have been so popular recently, and so on – it’s hard to stop! 

I’ll be around for another week or so to answer any comments on these posts, but after that I’ll be on a month-long vacation (ahhhhhh…. J), after which I’ll be digging out of e-mail for days, so I’m going to be silent for a bit.  I hope everyone enjoys their summer (or winter, depending on where you are), and I’ll probably post again at the end of August.  I hope to come up with a lot of blog application ideas as my family and I drive around the east coast of the U.S.  ‘Til then…

–Matt–*

VB Paint-by-Numbers.zip

Author

0 comments

Leave a comment

Feedback