DataGridView Printing
Introduction
Well it is that time of the year again. Time to get in shape for the summer months, and what better way to do it then through a rigid exercise program. I was never very good at tracking my progress in the gym, but thanks to .NET (and my wife), I have a way to do just that. This program uses the DataGridView bound to a Microsoft Access Database to create a printed sheet with your latest work out plans. The workout chart includes the exercise, number of sets, number of reps, amount of weight, and most importantly, whether or not you completed the exercise. The best part of this program is that you can print out your exercise program and bring it with you to the weight room.
Figure 1 - Exercise Program
New Form Capture Feature
.NET comes with a new form capture feature: the ability to copy the form as you see it on the screen into a bitmap. Although not useful for controls that you need to scroll, New Form Capture Feature works if you can fit all the contents you need to see on the screen into a bitmap. In this application, we stretched the DataGridView to allow for the full contents of the exercises, and we enabled AutoScrolling on the Form. This way the DataGridView will be much larger than the form, and the screen capture will print the entire DataGridView contents to the printer. The exercise application gives you two options: turning on the form capture feature to print out the form bitmap, or printing using a customized DataGridView printing. The customized printing is the default mode of the exercise program and it's more powerful because you can print the data in the grid view beyond any scrolling limitations of the DataGridView. Below is the code for capturing the form in a bitmap and drawing it to a graphics surface:
Listing 1 - Printing the Form Capture to a Graphics Surface
private void DrawFormSurface(System.Drawing.Printing.PrintPageEventArgs e) // create a bitmap the size of the form // draw the form image to the bitmap // draw the bitmap image of the form onto the graphics surface |
Using a Filename to Map an Image in the DataGridView
We want to place an image of the exercise into the DataGridView without inserting the image in the database. We prefer to place all the images in an images directory. Then we will read the images file names from the database, retrieve the images from the images directory, and put the images into the DataGridView. In order to accomplish all this, we'll need to create a custom grid view column and grid view cell that inherits from DataGridViewImageColumn and DataGridViewImageCell. We then override the GetFormattedValue method in the custom DataGridViewImageCell class. In the GetFormatted value method, we create a mapping between the file names and the actual images. Below is the code for accomplishing the filename to image mapping in a DataGridView:
Listing 2 - Mapping Images into the DataGridView from file name Strings
using System; namespace ExerciseProgram /// <summary> public class StringImageColumn : DataGridViewImageColumn public StringImageColumn() }
/// <summary> public class CustomImageCell : DataGridViewImageCell // mapping between filename and image static public System.Collections.Generic.Dictionary<string, Image> Images public CustomImageCell()
protected override object GetFormattedValue( if (value.GetType() != typeof(string)) // get the image from the string and return it // return the mapped image public static void LoadImage(object value) // read the image file // assign the image mapping public override object DefaultNewRowValue } } |
Updating the Grid
The DataGridView is bound to the Microsoft Access Database via a DataGridAdapter. We can edit the sets, reps, and weight values inside the grid while increasing body strength and the program will persist these values to the database. Sometimes it is useful to override the behavior of the DataGridView for updates and inserts in order to customize how the grid is updated. Below is the ADO.NET code that updates the grid in Microsoft Access without using the OleDb Adapter:
Listing 3 - Updating the Microsoft Access Database after editing the GridView
/// <summary> /// Event Handler called when Cell is finished editing private void UpdateCurrentRow(DataGridViewCellEventArgs e) // construct update command and update from the data set updateCommand.CommandText = String.Format(updateCommand.CommandText, dr["Completed"], updateCommand.CommandType = System.Data.CommandType.Text; updateCommand.Connection.Open(); // execute the update on the database |
Printing the DataGridView by Brute Force.
Once we are satisfied with our exercise parameters, it is time to print out the exercise schedule and head to the gym. Printing is accomplished using the PrintDocument in conjuction with GDI+. Every row and column is painstakingly drawn to the print graphics surface. Listing 4 shows the code that draws the exercise rows below the header. The method loops through each row in the DataSet and draws the text corresponding to the column in the data row. If the row contains a picture, rather than text, the image that is mapped to the text in the cell is printed instead. We also need to track the row and column position after placing each object onto the graphics surface so we know where the next object goes. We do this by incrementing the local variables: columnPosition and rowPosition.
Listing 4 - Printing out the Exercise Rows
private void DrawGridBody(Graphics g, ref int columnPosition, ref int rowPosition) foreach (DataRow dr in scheduleDataSet.Tables[0].Rows) // draw a line to separate the rows g.DrawLine(Pens.Black, new Point(0, rowPosition), new Point(this.Width, rowPosition));
// loop through each column in the row, and g.DrawRectangle(Pens.Black, new Rectangle(columnPosition, rowPosition + 20, 10, 10)); if (dc.DefaultCellStyle.Font != null)
// go to the next column position } // go to the next row position } |
Conclusion
The DataGridView is even more full featured than the DataGrid and provides a good interface for manipulating DataSets. This article explained two ways to print data inside the DataGridView. Hopefully this exercise will give you some insight into printing one of the more powerful controls inside the Windows Form in C# and .NET.
--
Alain Lompo
Excelta - Conseils et services informatiques
MCT
MCSD For Microsoft .Net
MVP Windows Systems Server / Biztalk Server
Certifié ITIL et Microsoft Biztalk Server
Commentaires