Document/View框架之外的打印
熟悉VC的程序员都知道Microsoft 的AppWizard生成的应用程序框架中,可以选择免费的打印及打印预览功能,但前提是必须选择Document/View结构。然而我们的数据库应用中一般都不需要一个文档类来保存文档,因为数据库(源)就是我们的文档,数据一般是在一知凯穗个对话框或视中与用户交互,我或修改结果直接通过数据库引擎写回数据库中。我们的程序主框架要么是基于对话框的、要么是基于无文档类的单视(或多视)结构,在这种情况下,AppWizard 在打印控制部分并不能给我们任何帮助,只能自己负责完成打印控制。
总结起来,一次打印 *** 作要遵循以下步骤:
1.得到或生成打印设备场景,可通过显示打印对话框让用户选择打印机与纸张等设置,也可在程序中直接取系统缺省打印机设置,然后根据报表格式设置纸张大小和打印方向;
2.开始在该设备场景中的一次打印作业,实际打印报表内容,终止打印作业;
3.清除打印设备场景,完成本次打印 *** 作。
在下面的例子中,我们在一个对话框中让用户选择打印某个报表,没有显示打印设置对话框,而是直接取系统缺省打印机设置,然后根据报表格式设置纸张大小和打印方向。之所以这样做是因为各行业、部门的报表格式一般都是至上而下的统一格式。函数DoPreparePrintDC()、DoPrint()和DoClearPrintDC()分别对应上述的三个步骤:
BOOL CMyDialog::DoPreparePrintDC()
// 准备打印场景
{
#defineFONTSIZE 14
// 获取打印机的设备属性
CPrintDialog dlgPrint( FALSE )
// 得到当前系统缺省打印机设置
if(!dlgPrint.GetDefaults()) return FALSE
LPDEVMODE pDM=dlgPrint.GetDevMode()
if(pDM==NULL) return FALSE
::GlobalUnlock(pDM)
// 联结打印DC,m—hDC是定义为HDC m—hDC的类成员变量
m—hDC=dlgPrint.CreatePrinterDC()
// m—DC是定义为CDC m—DC的类成员变量
if(!m—DC.Attach(m—hDC) return FALSE
// 设置打印标志
m—DC.m—bPrinting=TRUE
short cxInch=m—DC.GetDeviceCaps(LOGPIXELSX)
short cyInch=m—DC.GetDeviceCaps(LOGPIXELSY)
// 建立打印字体,m—fontPrint 是定义为CFont m_fontPrint的类成员变量
if(!m—fontPrint.CreateFont(MulDiv(FONTSIZE, -cyInch, 72),0,0,0, FW—NORMAL,0,0,0, GB2312—CHARSET, OUT—CHARACTER—PRECIS, CLIP—CHARACTER—PRECIS,DEFAULT—QUALITY, DEFAULT—PITCH|FF—DONTCARE, HFONTNAME)
return FALSE
return TRUE}
void CDlgDataPrint::DoClearPrintDC()
// 清除打印场景
{ m—fontPrint.DeleteObject()
m—DC.Detach()
::DeleteDC( m—hDC )}
void CDlgDataPrint::DoPrint()
// 实际打印输出
{ if(DoPreparePrintDC()==FALSE) return
// 开始一次打印作业孙誉
CString str
str.LoadString(AFX—IDS—APP—TITLE)
DOCINFO di
::ZeroMemory (&di, sizeof (DOCINFO))
di.cbSize=sizeof (DOCINFO)
di.lpszDocName=str
m—DC.StartDoc(&di)
m—DC.StartPage()
// 将打印字体选进设备场景
CFont* pOldFont=m—DC.SelectObject(&m—fontPrint)
// 输出报表,建议用CDC::DrawText()函数便于控制打印范围
...
// 终止打印作业
m—DC.EndPage()
m—DC.EndDoc()
m—DC.SelectObject(pOldFont)
// 清除打印场景
DoClearPrintDC()}
表格问题
对格式固定或表头、表体比较复杂的情况,特别是有嵌套表格的情况下,我们认为先将打印结果以文本形式写入一临时文件,然后再对文件进行打印输出的方法会使事情简单一些。因为这样在程序中就可以只关心数据(表体)的输出,而表头则可以固定格式存入数据库。
但是,我们在对文本形式的报表进行打印输出时却发现一个有趣的现象,有时输出结果的相邻两行表格竖线不能对齐,而有时又可以。原来是 Windows 图形输出的问题,只要我们在建立打印字体时,字体的高度是偶数就可保证竖线对齐,如果是奇数则不能对齐。知道了原因,在建立打印字体时只需加入如下代码即可:
...
#defineFONTSIZE 14
#defineHFONTNAME ″微软雅黑″
// 建立打印字体,m—fontPrint是定义为CFont m—fontPrint的类成员变量
int nFontHeight=MulDiv(FONTSIZE,-cyInch, 72)
if(nFontHeight % 2) nFontHeight++
if(!m—fontPrint.CreateFont(nFontHeight, 0, 0, 0, FW—NORMAL, 0, 0,0,GB2312—CHARSET, OUT—CHARACTER—PRECIS, CLIP—CHARACTER—PRECIS,DEFAULT—QUALITY, DEFAULT—PITCH | FF—DONTCARE, HFONTNAME))
return FALSE
...
还有一点小技巧,输出文本形式的报表时,对报表的横线只需要输出其正常高度的二分之一或三分之一,这样报表看起来更紧凑美观。这也是我们在前面建议用CDC::DrawText()进行绘制输出的原因。
使用ShellExecute打印,它会通过文件扩展名调用默认的打开脊橡轮程序进行打印。换句话说,如果是doc文档,就会使用word打印,你的参数为SHOW,还可以看到一个word的打开过程(会自动关闭)。
而不使用默认打印机,就是因如唤为要打印的文档或者文件关联程序指定了使用的打印机(这个和文件格式有关,樱信比如doc文件格式中,是可以指定打印机的),所以,如果没有指定,那么一定会使用默认打印机,而不使用默认,和程序无关(也管理不了),只能看是文档问题还是关联程序问题。
使雹贺用数据库读取出数据来以后直接自己写打印的代码就可以了啊我以前做过一个相似功能的先从数据库里读取数据填充了列表框控件然后调用了我自己写的一个函数打印这个列表框中的所有数据
根据你现在的贴图假设点击打印按钮的时候列表框里已经被你通过访问数据库腊衫填充进了数据,那么就是从该列表框中获取数据按顺序打印就可以了。以下这个函数基本就能实现这个需求。
//打印列表
void PrintList(CListCtrl &listCtrl)
{
if(!listCtrl.GetItemCount())
{
AfxMessageBox("没有数据可以打印")
return
}
// 建立打印对话框对象
CPrintDialog
d1(FALSE,PD_ALLPAGES|PD_ALLPAGES|PD_NOPAGENUMS,NULL)
HDC dc1
DOCINFO di={sizeof(DOCINFO),printName,NULL}
if(d1.DoModal()==IDOK)
{
// 返回一个打印DC句柄
dc1=d1.GetPrinterDC()
CDC testDC
testDC.Attach(dc1)
testDC.SetBkMode(TRANSPARENT)
int nListItem = listCtrl.GetItemCount(),//需要打印多少条目
nListColumn =
((CHeaderCtrl*)listCtrl.GetHeaderCtrl())->GetItemCount(),//列数
nPageNeedPrint = nListItem/70 + ((nListItem%70) ? 1 : 0),//需要打印的页数
nPageCount = 0, //已打印的页数
nCount = 0, //已打印条目的计数
nItemPerPage = 0//每页应该打印的条目数
//列表框宽度,用于计算比例
int nHeaderWidth = 0
for (int k=0k<nListColumnk++)
{
nHeaderWidth += listCtrl.GetColumnWidth(k)
}
// 获取打印设备横、纵坐标像素值
int xHor = testDC.GetDeviceCaps(HORZRES)
int yVer = testDC.GetDeviceCaps(VERTRES)
int nStepX = xHor/25, nStepY = yVer/80
// 启动打印工作
StartDoc(dc1,&di)
while(nPageCount <nPageNeedPrint)
{
// 一页开始
StartPage(dc1)
///////////////////////////////////////////////////
//具体的针对参数listCtrl的打印 *** 作
//就类似dc绘源局派图输出文字一样,根据需求输出即可
// 一页结束
EndPage(dc1)
nPageCount++
}
// 终止打印工作
EndDoc(dc1)
// 删除句柄
if(DeleteDC(dc1))
{
return
}
else
{
//出错信息
EndDoc(dc1)
// 删除句柄
if(DeleteDC(dc1))
{
return
}
else
{
//出错信息
}
}
}
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)