Introduction
I was faced with a problem while developing an application on Microsoft .NET Framework. The problem was .NET Framework’s printout support. NET framework has lack of printing and print preview support for some resource reasons. Therefore, I started to develop a print engine for Compact Framework. Actually, Pocket PC 2003 has APIs for printing support to Pocket PC 2003 through serial port which is useful for developers who want to develop an application on Compact Framework with printing support.
I used “P/Invoking Serial APIs in the Compact Framework” sample code on MSDN while developing serial communication class. As we know, dot-matrix printers uses COM or LPT port in order to communicate with the system. If you want to send data through COM port, you should use byte streaming.
In my case, I used following values;
BaudRate = 9600 Size = 8 Parity = 0 StopBits = 0 |
My class has print preview support, too. It is based on dynamically created window as below;
Pic. 1
I used multiline, locked textbox in order to show print preview. Text box object is so necessary for this job because of its nature. Actually, by forming a page from byte array; I can set text property to print buffer in order to show preview, easily. As you see, preview has BACK and NEXT buttons and page number label, too. All of the controls has been created dynamically.
Background
Smart Device Print Engine is formed by two main layers; print class and serial communication class. I will describe in detail and step by step, how it works.
Pic.2
Using the code
As I mentioned before, serial port communication is based on byte streaming. Therefore, you need to send data in a byte array. I used CEWriteFile
API function in order to send data through COM port. You can put your data in a (n) sized array and send it to the port. For example :
int iNumberOfBytesRead=0;
uint iNumberOfBytesToRead=1024;
byte[] Output = new byte[1024];
CEWriteFile(hPort, Output, iNumberOfBytesToRead,
ref iNumberOfBytesRead, IntPtr.Zero);
The API function mentioned, sends 1024 bytes data at a time which is necessary for minimizing trips between smart device and the serial port. For example you want to print “hello world” string from a line printer. How this is going to happen? Firstly, you need to create a page as below;
SDPE = new SmartDevicePrintEngine();
SDPE.Port="COM1:";
SDPE.CreatePage(36,46);
Code block mentioned above, creates a SmartDevicePrintEngine object instance and sets serial port number to COM1 and creates a logical page on memory which is 2 dimensional 36x46 byte array. Then it creates, CommDevice instance in order to communicate with printer. In CommDevice constructor, I set package sizes to 1024 as default value. That means, 1024 bytes will be send at a time for each package. Then, I created Device Control Block instance to set serial port properties. As we know, Compact Framework doesn’t support Abort()
method for threads. Therefore, I used event mechanism in order to stop the thread execution. I used an ArrayList
to store pages for print preview in SmartDevicePrintEngine constructor. These happen at Form_Load
event. Now, we are ready to fill up the logical page with our strings.
How will I write my string and locate on paper? I developed WriteToLogicalPage()
method since it puts a string on the print area. Method has two overloads which are aligned string and unaligned string. I will describe, how it works in the following sections.
Writing and locating a string on a page is quite easy. Look at the following code block for an example;
SDPE.WriteToLogicalPage(17, 2, "Hello World");
X and Y coordinates are starting from value 1. We want to put up “Hello World” string to 17, 2 location. First step in WriteToLogicalPage()
method is checking page bounds and converting the string (“Hello World”) into PageBuffer
which is a byte array. We created a logical page, set and located the string on page. Now, we need to call the most important method which is Print()
.
I initialized serial port by OpenPort
, SetupCommSetDCB
, SetConnectionTimeout
and CreateCommThread
. These methods initializes and opens serial port device communication. After initialization, we need to send byte array to serial port. Therefore, we should call Send()
method. In Send()
method, we need to send 1024 bytes package for each trip. At the end of this process, all of the data is sent to the serial port as 1024 bytes packages. We need to send form feed character (“0x0c”) for page break. If we have more than one page, we need to call NewPage()
method which flushes old page and creates a new one. Now that, print process has completed successfully, we should close port by Close()
method. That’s all.
Now, we will look at the preview support in Smart Device Print Engine. All steps are the same as Print()
method until AddToPreview()
. In AddToPreview()
method; we need to create a StringBuilder
object instance in order to concatenate rows for building a new preview page. We should insert new page into page array. NewPage()
method, flushes old page and creates new one. Preview page is formed dynamically. It has a textbox in order to represent printout of pages.
I want to describe class methods and properties at below;
Class : SmartDevicePrintEngine
|
Signature : SmartDevicePrintEngine() Description : Constructor. |
Signature : public bool CreatePage(int Width, int Height) Description : Creates a logical page which is 2 dimensional byte array. |
Signature : public void WriteToLogicalPage(int x, int y, string Value) Description : Writes a string to given location on page. String shouldn’t overflow from page. |
Signature : public void WriteToLogicalPage(int x, int y, int Length, int Alignment, string Value) Description : Writes a string to given location by alignment and fixed length on page. String shouldn’t overflow from page. For example, you may want to write numeric value which should be right aligned. |
Signature : public void FormFeed() Description : It is simple to understand. Sends 0x0c to serial port. |
Signature : private string ReadLogicalPageByRow(int RowNumber) Description : Reads byte array row by row and returns it in a string variable. |
Signature : public void Dispose() Description : Flushes objects from memory. |
Signature : public void NewPage() Description : Creates new page. |
Signature : public void Print() Description : Prints the document page or pages. |
Signature : public void AddToPreview() Description : Adds your preview pages into array in order to represent at print preview screen. |
Signature : public void Preview() Description : Calls preview window and represents preview of a page. |
Signature : public int PageWidth Description : Page width. |
Signature : public int PageHeight Description : Page height. |
Signature : public string Port Description : COM port number. |
Class : CommDev
|
Signature : public CommDevice() Description : Constructor |
Signature : private IntPtr OpenPort() Description : Opens a port in order to perform serial communication. |
Signature : private void SetupComm() Description : Sets buffer sizes. |
Signature : private void SetConnectionTimeOut() Description : Set connection time-out for serial communication. |
Signature : private void SetDCB() Description : Device control block |
Signature : private void CreateCommThread() Description : Creates a thread in order to processing events. |
Signature : public void Init() Description : Serial port initialization routines. |
Signature : public void Send(char[,] Data) Description : Sends byte array in 1024 bytes segments. |
Signature : public void Send(byte Data) Description : Sends a char to the serial port. |
Signature : public void ClosePort() Description : Closes COM port. |
Signature : public void Dispose() Description : Flushes objects. |
Signature : public string Port Description : COM port number. |
Signature : public uint BaudRate Description : Communication baud rate. |
You can use Smart Device Print Engine in your continuous forms such as sales, order, invoice, etc... Class supports ASCII prints. But, I’m planning to add, Bluetooth support into class, in the near future. I hope it will be necessary for developers who want to add print-out support to their own projects. Also, this article gives information about serial communication through Pocket Devices.
I used Smart Device Print Engine class at mobile sales project which needs to printout invoices on the field. We are still using the class, successfully.
History
- Version 1.0 - First release