Components, components, components. Using components is the difference between writing an application in one week vs. an application in a few hours. Printing is all carried out through components included in the toolbox. The main component that talks to the printer is the PrintDocument component. To use this component, just call the Print function and intercept the PrintPage event. Below are the routines that do this:
- protected void ThePrintDocument_PrintPage(object sender, System.Drawing.Printing.PrintPageEventArgs ev)
- {
- float linesPerPage = 0;
- float yPosition = 0;
- int count = 0;
- float leftMargin = ev.MarginBounds.Left;
- float topMargin = ev.MarginBounds.Top;
- string line = null;
- Font printFont = this.richTextBox1.Font;
- SolidBrush myBrush = new SolidBrush(Color.Black);
-
- linesPerPage = ev.MarginBounds.Height / printFont.GetHeight(ev.Graphics);
-
- while (count < linesPerPage && ((line = myReader.ReadLine()) != null))
- {
-
- yPosition = topMargin + (count * printFont.GetHeight(ev.Graphics));
-
- ev.Graphics.DrawString(line, printFont, myBrush, leftMargin, yPosition, new StringFormat());
- count++;
- }
-
- if (line != null)
- ev.HasMorePages = true;
- else
- ev.HasMorePages = false;
- myBrush.Dispose();
- }
PrintFile_Click is called from the Print Menu. This routine first assigns the PrintDialog component to the document being printed. The dialog is shown using ShowDialog so the user can set their print settings and to decide whether to print or not. After ThePrintDocument.Print is called, the printing process begins. The PrintPage event is automatically called by the system to print the desired page information. The PrintPageEventArgs ev contains the Device Contexts ev.Graphics to use for drawing to the Printer. In this example we read each line of text from the rich text edit control using a StringReader. The StringReader class gives us the ability to treat a string like a stream and take advantage of the ReadLine function of the StringReader for reading each line out of the text edit control. We can calculate the position of the lines based on margins and the height of the font. It's interesting to note, that in order to predict the font height for the printer, we pass it the Device context of the printer in the GetHeight function (printFont.GetHeight(ev.Graphics())). This is because the font height on the screen pixels is different than the font height in printer pixels. The PrintPageEventArg attribute HasMorePages is continually set so that the PrintPage Event is triggered if their are more lines in the textbox than the current page can hold .
Print Preview
PrintPreview behaves a heck of a lot like Print as seen below:
- protected void PreviewFile_Click(object sender, System.EventArgs e)
- {
- try
- {
- string strText = this.richTextBox1.Text;
- myReader = new StringReader(strText);
- PrintPreviewDialog printPreviewDialog1 = new PrintPreviewDialog();
- printPreviewDialog1.Document = this.ThePrintDocument;
- printPreviewDialog1.BorderStyle = FormBorderStyle.Fixed3D;
- printPreviewDialog1.ShowDialog();
- }
- catch (Exception exp)
- {
- System.Console.WriteLine(exp.Message.ToString());
- }
- }
The PrintPreview component also triggers the PrintPage event, but instead of outputting to the printer it outputs to the Print Preview Screen.
Other Editor Functions
The Open and Save menu items use streams to read and write the files. The persistence of files is discussed in other articles on this site. See Using ListView in C#. In future articles we will add find/replace functionality to this text editor to make it a bit more powerful.