直接上代码,将下面的代码复制到工程中保存为main.cpp,然后在工程目录下放一张miku.bmp图片,直接就可以编译并且执行了。
下面是代码
#include <windows.h>
#include <stdio.h>
//这个函数用于读取bmp图像文件,用于给打印机打印的时候使用
//info是位图信息结构
//file是文件名
//dib_ptr是位图rgb像素数据指针,输出用的,所以请提供一个void**
bool read_bmp(BITMAPINFO& info, const char* file, void** dib_ptr)
{
BITMAPFILEHEADER file_handle;
FILE* f = fopen(file, "rb");
if(f == 0) return false;
memset(&info, 0, sizeof(info));
fread(&file_handle, 1, sizeof(file_handle), f);
fread(&info.bmiHeader, 1, sizeof(info.bmiHeader), f);
fseek(f, file_handle.bfOffBits, SEEK_SET);
void* dib = malloc(info.bmiHeader.biSizeImage);
fread(dib, 1, info.bmiHeader.biSizeImage, f);
fclose(f);
*dib_ptr = dib;
return true;
}
//释放位图的数据指针,释放空间
void release_bmp(void** dib_ptr)
{
if(dib_ptr != 0)
{
if(*dib_ptr != 0)
{
free(*dib_ptr);
*dib_ptr = 0;
}
}
}
int main(int argc, char* argv[])
{
PRINTDLG printInfo = {0};
printInfo.lStructSize = sizeof(printInfo);
printInfo.Flags = PD_RETURNDC | PD_RETURNDEFAULT | PD_ALLPAGES;
//PD_RETURNDEFAULT 意味着直接返回当前系统默认的打印机设置,若没有这个标识,则会弹出对话框让用户自己选择
//PD_RETURNDC 意味着返回的是dc而不是ic(information context)
//PD_ALLPAGES 指定“全部”单选钮在初始时被选中(缺省标志)
//对于错误的时候,若需要知道更多的错误消息,请执行CommDlgError来查看返回值
//PrintDlg目的是获取当前系统设置的默认打印机相关信息,供后面使用
if(!PrintDlg(&printInfo))
{
printInfo.Flags = 0; //清除标志,然后执行将会弹出对话框让用户选择打印机
if(!PrintDlg(&printInfo))
{
printf("没有选择打印机。\n");
return 0;
}
}
//获取打印的时候的dc,然后往这个dc上绘图就是打印出来的样子了
HDC hPrintDC = printInfo.hDC;
//锁定全局对象,获取对象指针。 devmode是有关设备初始化和打印机环境的信息
DEVMODE* devMode = (DEVMODE*)GlobalLock(printInfo.hDevMode);
if(devMode == 0)
{
printf("获取打印机设置时发生了错误.\n");
return 0;
}
devMode->dmPaperSize = DMPAPER_A4; //打印机纸张设置为A4。
devMode->dmOrientation = DMORIENT_PORTRAIT; //打印方向设置成纵向打印
//DMORIENT_LANDSCAPE 是横向打印
//对打印方向的设置,会影响hPrintDC的大小,假设宽度为1024,高度为300
//则在横向打印的时候dc大小会是宽1024 * 高300
//而纵向打印的时候dc大小会是宽300 * 高1024
int printQuality = devMode->dmPrintQuality; //获取打印机的打印质量
//devMode->dmPrintQuality = DMRES_MEDIUM;
//设置打印质量的,因为像素被打印到纸上的时候是有做转换的
//单位是dpi,意为像素每英寸(dots per inch)。就是一英寸的纸张上
//打印多少个像素点,意味着这个质量越高,打印结果越精细,越低,越粗糙
//设置的质量可以是具体数值,也可以是宏DMRES_MEDIUM
//一般我们选择300,或者600,DMRES_MEDIUM = 600dpi
//应用我们修改过后的设置.
ResetDC(hPrintDC, devMode);
//解锁全局对象,对应GlobalLock
GlobalUnlock(printInfo.hDevMode);
//设置绘图模式,以保证绘图可以不失真的绘制上去,因为StretchDIBits函数要求设置这个才能够不是失真的绘图
//当你用StretchDIBits绘图往窗口显示的时候就会发现,24位图,若没有这个设置,是会失真的
SetStretchBltMode(hPrintDC, HALFTONE);
//读取位图,待会画在hPrintDC上面去
BITMAPINFO bmp_info;
int image_width = 0, image_height = 0;
void* dib_ptr = 0;
if(!read_bmp(bmp_info, "miku.bmp", &dib_ptr))
{
printf("读取位图miku.bmp失败了.\n");
return 0;
}
image_width = bmp_info.bmiHeader.biWidth;
image_height = bmp_info.bmiHeader.biHeight;
printf("位图大小:%d x %d\n", image_width, image_height);
//设置打印时候的字体
LOGFONT lf;
lf.lfHeight = -printQuality * 1 / 2.54;
//打印出来的字像素高度有n个,注意是像素高度,打印到纸上的时候是需要将
//像素转换成实际尺寸单位,比如你需要在纸上打印高度为1cm的字,当你打印质量为600dpi的时候
//这里就设置为 -236, -600dpi * 1cm / 2.54 = -236pix
lf.lfWidth = 0;
lf.lfEscapement = 0;
lf.lfOrientation = 0;
lf.lfWeight = 5; //这里设置字体重量,意味着字体的厚度
lf.lfItalic = false; //斜体
lf.lfUnderline = false; //下划线
lf.lfStrikeOut = 0;
lf.lfCharSet = DEFAULT_CHARSET;
lf.lfOutPrecision = 0;
lf.lfClipPrecision = 0;
lf.lfQuality = PROOF_QUALITY;
lf.lfPitchAndFamily = 0;
strcpy (lf.lfFaceName, "宋体"); //使用宋体进行打印
//实际上这一步并不是必须的,因为默认打印机设置已经配置好了,这里只是我们自己固定好字体
HFONT hUseFont = CreateFontIndirect(&lf); //创建字体
HFONT hOldFont = (HFONT)SelectObject(hPrintDC, hUseFont); //选用创建的字体
//获取dc的大小,实际上还有一种HORZRES和VERTRES就是宽度和高度,但是我查得到的结果说计算下来准确的
//HORZSIZE 是Horizontal size in millimeters,页面宽度(水平),单位毫米mm
//VERTSIZE 是Vertical size in millimeters,页面高度(垂直),单位毫米mm
//LOGPIXELSX 是Logical pixels/inch in X,x方向的逻辑像素每英寸.单位 pix / inch,像素每英寸
//LOGPIXELSY 是Logical pixels/inch in Y,y方向的逻辑像素每英寸.单位 pix / inch,像素每英寸
//不用管逻辑是个什么东西,不理会他,知道单位是pix / inch就行了
//1 inch = 2.54 cm,所以这里是 mm / 25.4 * pix / inch,得到的结果就是dc大小像素数为单位
int dc_page_width = GetDeviceCaps(hPrintDC, HORZSIZE) / 25.4 * GetDeviceCaps(hPrintDC, LOGPIXELSX);
int dc_page_height = GetDeviceCaps(hPrintDC, VERTSIZE) / 25.4 * GetDeviceCaps(hPrintDC, LOGPIXELSY);
//好了,可以开始打印了
DOCINFO doc_info = {sizeof(DOCINFO), "测试打印机"};
//cbSize
//结构体的字节大小
//lpszDocName
//指明文档名的字符串指针,该字符串以null为尾。
//lpszOutput
//指明输出文件的名称的字符串指针,该字符串以null为尾。如果指针为null,那么输出将会发送至某个设备,该设备将由传递至 StartDoc 函数的‘设备上下文句柄’HDC来指明。
//lpszDatatype
//指针指向代表某种数据类型的字符串,而数据用于记录打印工作,该字符串以null为尾。合法值可通过调用函数 EnumPrintProcessorDatatypes 可得到,亦可为 NULL。
//fwType
//指明打印工作的其它信息。可为0或者以下值之一:
//DI_APPBANDING
//DI_ROPS_READ_DESTINATION
//开始一个档案,打印的时候是按照档案来区分的,返回作业编号(大于0的为正常)
int doc_idd = StartDoc(hPrintDC, &doc_info);
if(doc_idd <= 0)
{
printf("StartDoc 错误,错误代码:%d\n", GetLastError());
goto last_code;
}
printf("作业编号:%d\n", doc_idd);
//开始一个打印页面,大于等于0为正确
if(StartPage(hPrintDC) < 0)
{
printf("StartPage 错误\n");
AbortDoc(hPrintDC); //结束doc打印
goto last_code;
}
//设定文字和图像绘制的区域
RECT rcText = {0, 30, dc_page_width, 300};
RECT rcImage = {30, 300, dc_page_width - 30, dc_page_height - 30};
//写下文字
DrawText(hPrintDC, "miku,测试打印图片", -1, &rcText, DT_VCENTER | DT_SINGLELINE | DT_CENTER);
//将图片绘制到dc上,给打印机打印出来
//这个函数带拉伸功能的,若dest的宽度高度和src不一致的时候他会拉伸
StretchDIBits(hPrintDC, rcImage.left, rcImage.top, rcImage.right - rcImage.left,
rcImage.bottom - rcImage.top, 0, 0, image_width, image_height,
dib_ptr, &bmp_info, DIB_RGB_COLORS, SRCCOPY);
//结束这一页,其实可以循环的startpage、endpage,这样在一个文档中进行多个页面的打印
EndPage(hPrintDC);
EndDoc(hPrintDC);
printf("打印顺利完成了. o(∩_∩)o \n");
last_code:
//选取旧的字体
SelectObject(hPrintDC, hOldFont);
//删除gdi对象,释放内存
DeleteObject(hUseFont);
//释放位图内存
release_bmp(&dib_ptr);
return 0;
}