发个利用EnumJobs监控打印机的源码

标 题:  【分享】发个利用EnumJobs监控打印机的源码
作 者: qydao
时 间: 2009-06-25,11:32:03
链 接: http://bbs.pediy.com/showthread.php?t=92239

办公室里换了个打印机,没两天墨盒完了,很生气,决定监控,网上找了一些,要注册。还有些不能监控网络共享打印文档,决心自己写个简单的,当初想用HOOK,失败,因为我发现打印机共享时,驱动的文件夹都共享了,可见不行,只好用EnumJobs,采用定时器监控。
装那个机子都可监控,只要机子开着,不过最好装在打印报务器上。
代码:
#include <windows.h>  
#include <iostream>
#include "resource.h"

int oldID[10];
typedef struct _PRINTPROCESSOROPENDATA
{
  PDEVMODE  pDevmode;
  LPWSTR  pDatatype;
  LPWSTR  pParameters;
  LPWSTR  pDocumentName;
  DWORD   JobId;
  LPWSTR  pOutputFile;
  LPWSTR  pPrinterName;
}PRINTPROCESSOROPENDATA,*PPRINTPROCESSOROPENDATA;


char * getinic(LPCSTR lb,LPCSTR name)
{
  char FilePath[MAX_PATH];   
  GetModuleFileName(NULL,FilePath,sizeof(FilePath));   
  char drive[_MAX_DRIVE];   
  char dir[_MAX_DIR];   
  char fname[_MAX_FNAME];   
  char ext[_MAX_EXT];   
  _splitpath(FilePath,drive,dir,fname,ext);   
  wsprintf(FilePath,"%s%s",drive,dir);
  strcat(FilePath,"\\date.ini");
  char dd[256];
  GetPrivateProfileString(lb,name,NULL,dd,256,FilePath);
  return dd;
}

int GetdocPages(LPDEVMODE pDevMode,LPSTR pwszDocName)
{
  HINSTANCE hdll=LoadLibrary("gdi32.dll");
  typedef HANDLE(*UGetSpoolFileHandle)(LPSTR pwszPrinterName,LPDEVMODE pDevmode,LPSTR pwszDocName);
  UGetSpoolFileHandle GetSpoolFileHandle=(UGetSpoolFileHandle)GetProcAddress(hdll,"GdiGetSpoolFileHandle");
  HANDLE hPrint=GetSpoolFileHandle(getinic("date","Printer"),pDevMode,pwszDocName);
  typedef DWORD(*UGdiGetPageCount)(HANDLE  SpoolFileHandle);
  UGdiGetPageCount GetPageCount=(UGdiGetPageCount)GetProcAddress(hdll,"GdiGetPageCount");
  int i=GetPageCount(hPrint);
  return i;
}

VOID CALLBACK TimerProc ( HWND hwnd, UINT message, UINT iTimerID, DWORD dwTime)
{
  HANDLE hPrinter; 
  //int ret=OpenPrinter("HP LaserJet M1005",&hPrinter,NULL);
  int ret=OpenPrinter(getinic("date","Printer"),&hPrinter,NULL);
  DWORD   pcbNeeded=0;   //所有打印字节数   
  DWORD   pcReturned=0;   //打印任务数   
  ret=EnumJobs(hPrinter,0,127,2,0,0,&pcbNeeded,&pcReturned);   //必要的一步,先取出打印机里任务的字节数,然后决定jobs1变量大小。//--------------------------①   
  int actNeed=0;  
  actNeed=pcbNeeded;   
    JOB_INFO_2 *jobs1=NULL;
  jobs1=new JOB_INFO_2[actNeed];
  ret=EnumJobs(hPrinter,0,127,2,(LPBYTE)jobs1,actNeed,&pcbNeeded,&pcReturned);



  for(int i=0;i<pcReturned;i++)
  {
  bool isfine=false;
  for(int n=0;n<oldID[0];n++)
  {
    if(jobs1[i].JobId==oldID[n+1]) isfine=true;
  }
  if(isfine) break;
  FILE *fp;
  if((fp=fopen(getinic("date","outfile"),"r"))==NULL)
  {
    fp=fopen(getinic("date","outfile"),"a");
    fputs("<table width=\"700\" border=\"0\" align=\"center\" cellspacing=\"0\">\n",fp);
    fputs("<tr><th><font size=\"+4\" color=\"#0000CC\" face=\"华文新魏\">打印机监控记录</font></th>\n",fp);
    fputs("</tr></table>\n",fp);
    fputs("<table width=\"700\" border=\"1\" align=\"center\" cellspacing=\"0\" bordercolor=\"#99CC33\"><tr>\n",fp);
    fputs("<th width=\"35\" bgcolor=\"#99CC33\">序号</th>\n",fp);
    fputs("<th width=\"273\" bgcolor=\"#99CC33\">文档名</th>\n",fp);
    fputs("<th width=\"73\" bgcolor=\"#99CC33\">用户</th>\n",fp);
    fputs("<th width=\"164\" bgcolor=\"#99CC33\">计算机</th>\n",fp);
    fputs("<th width=\"42\" bgcolor=\"#99CC33\">纸张</th>\n",fp);
    fputs("<th width=\"41\" bgcolor=\"#99CC33\">页数</th>\n",fp);
    fputs("<th width=\"42\" bgcolor=\"#99CC33\">份数</th></tr>\n",fp);

    fclose(fp);

  }else
  {
  fclose(fp);
  fp=fopen(getinic("date","outfile"),"a");
  fputs("<tr><td>",fp);
  char tmpBuf[260];
  _ultoa(jobs1[i].JobId,tmpBuf,10);
  fputs(tmpBuf,fp);
  fputs("</td><td>\n",fp);
  fputs(jobs1[i].pDocument,fp);

  fputs("</td><td>\n",fp);
  fputs(jobs1[i].pUserName,fp);

  fputs("</td><td>\n",fp);
  fputs(jobs1[i].pMachineName,fp);

  fputs("</td><td>\n",fp);
  if(jobs1[i].pDevMode->dmPaperSize==DMPAPER_A4)  fputs("A4",fp);
  if(jobs1[i].pDevMode->dmPaperSize==DMPAPER_B5)  fputs("B5",fp);
  if(jobs1[i].pDevMode->dmPaperSize==263)  fputs("16K",fp);
  fputs("</td><td>\n",fp);
  _ultoa(jobs1[i].TotalPages,tmpBuf,10);
  fputs(tmpBuf,fp);
  fputs("</td><td>\n",fp);
  //int pp=GetdocPages(jobs1[i].pDevMode,jobs1[i].pDocument);
  //_ultoa(pp,tmpBuf,10);
  _ultoa(jobs1[i].pDevMode->dmCopies,tmpBuf,10);
  fputs(tmpBuf,fp);
  fputs("</td></tr>\n",fp);
  fputs("\n",fp);
  fclose(fp);
  }
  }
  for(int i=0;i<pcReturned;i++)
  {
  oldID[i+1]=jobs1[i].JobId;
  }
  oldID[0]=pcReturned;
  delete   jobs1;   

}

void showmenu(HWND hdwnd)
{
  POINT point;//定义一个指针变量
  ::GetCursorPos(&point);
  SetForegroundWindow(hdwnd);//让菜单自动消失
  HMENU htaskmenu=LoadMenu(GetModuleHandle(NULL),MAKEINTRESOURCE(IDR_MENU1));//读入菜单,获得菜单句柄。
    htaskmenu=GetSubMenu(htaskmenu,0);//设置为右键点击弹出菜单
  TrackPopupMenuEx(htaskmenu,TPM_VERTICAL|TPM_LEFTALIGN,point.x,point.y,hdwnd,NULL);//在单击的位置上显示弹出菜单
}

void taskico(HWND hdwnd,bool ifshow)
{
NOTIFYICONDATA nid;
nid.cbSize=sizeof(NOTIFYICONDATA);
strcpy(nid.szTip,"打印机监控精灵");
nid.hWnd=hdwnd;
nid.uID=3;
nid.uCallbackMessage=WM_USER+10;
HICON hIcon = LoadIcon(GetModuleHandle(NULL),(LPCTSTR)IDI_ICON1);   
nid.hIcon=hIcon;
nid.uFlags=NIF_MESSAGE|NIF_ICON|NIF_TIP;
if(ifshow) Shell_NotifyIcon(NIM_ADD,&nid);    //在任务栏中加入一个图标
else Shell_NotifyIcon(NIM_DELETE,&nid);
}

void autorun()
{
LPCSTR str;
HKEY hRegKey;
BOOL bResult;
str="Software\\Microsoft\\Windows\\CurrentVersion\\Run";
if(RegOpenKey(HKEY_LOCAL_MACHINE, str, &hRegKey) != ERROR_SUCCESS) 
bResult=FALSE;
else
{
  char FilePath[MAX_PATH];   
  GetModuleFileName(NULL,FilePath,sizeof(FilePath));   
  char drive[_MAX_DRIVE];   
  char dir[_MAX_DIR];   
  char fname[_MAX_FNAME];   
  char ext[_MAX_EXT];   
  _splitpath(FilePath,drive,dir,fname,ext);   
::RegSetValueEx( hRegKey,fname,0,REG_SZ,(CONST BYTE *)FilePath,255);
}
}

LRESULT CALLBACK DBFunc(HWND hdwnd,UINT message,WPARAM wParam,LPARAM lParam) 
    { 
    switch(message){
  case WM_CREATE:
    autorun();
    taskico(hdwnd,true);
    oldID[0]=0;
    SetTimer(hdwnd,1,1000,TimerProc);
    break;
  case WM_USER+10:
    {
    UINT uMsg=lParam;//用户对任务栏图标进行的是什么操作
    switch(uMsg)
    {
    case WM_RBUTTONDOWN://如果是单击右键
    showmenu(hdwnd);
    break;
    }
    }
    break;
    case WM_COMMAND: 
    switch(LOWORD(wParam)){ 
    case ID_Menu:
    taskico(hdwnd,false);
    PostQuitMessage(0);
  break;
    case ID_MENU_40002:
    ShellExecute(hdwnd,"open",getinic("date","outfile"),NULL,NULL,SW_SHOW);
  break;
  }
  break;
    }
return DefWindowProc(hdwnd,message,wParam,lParam);
}


int APIENTRY WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPSTR lpCmdLine,int nCmdShow) 
{


static TCHAR szAppName[]=TEXT ("UUUUUU");
HWND hwnd;
MSG msg;
WNDCLASS wndclass;
wndclass.style =0;
wndclass.lpfnWndProc =DBFunc;
wndclass.cbClsExtra =0;
wndclass.cbWndExtra =0;
wndclass.hInstance =hInstance;
wndclass.hIcon =0;
wndclass.hCursor =0;
wndclass.hbrBackground =0;
wndclass.lpszMenuName =NULL;
wndclass.lpszClassName =szAppName;
if (!RegisterClass (&wndclass))
{
MessageBox (NULL,TEXT("Program requires Windows NT!"),szAppName, MB_ICONERROR);
return 0;
}
hwnd = CreateWindow (szAppName, NULL,WS_DISABLED,0, 0,0, 0,NULL, NULL, hInstance, NULL);
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage (&msg);
DispatchMessage (&msg);
}
return msg.wParam;
} 
其实有问题,就是天杀的word份数总是1,除非打sp3,水的很,后来有人说解析SPL,要使用GdiGetPageCount,但找不到DDK,麻烦。写的乱莫笑!
下面是搞好的,注意修改ini中的内容 
上传的附件
文件类型: rar Release.rar (29.0 KB, 175 次下载) [谁下载?]
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
客户端安装个软件。负责监控该电脑的打印动作。并把内容转成图片储存。并将图片传送到服务器机器上,并将打印内容传到服务器上面. 解决方案: 1. 先用api打印函数连接到指定的打印机.再试着用枚举函数()获得打印作业信息.根据信息得知打印的内容,及当前的状态. 2. 获知内容,得知打印内容所在的位置,再某种方式将数据导在图片;再传到服务器上. 技术问题: 1. 打印枚举函数中找不到JOB_INFO_1 或 JOB_INFO_2结构的定义.(已解决) 2. 怎么样获得打印内容.是通过原本的驱动还是其它办法, 其它: 文件传给pdf打印机之前已经存成raw格式了,应该是从缓冲池中直接读取数据 接下去要完成的应该是怎么把raw格式读出来, 具我推测在获得打印信息的时候肯定有某个参数跟这个RAW格式是对应的.读出某个参数后才能再继续读取对RAW文件读取 具微软件网站显示,打印机的格式应该分成5种.raw的三种格式,text,emf(增强型图元文件) RAW格式指是最原始的数据 CreateDC("WINSPOOL", printer, null, ref dv);//用DISPLAY,是获取整个屏幕的设备场景;2、用WINSPOOL,则是访问打印驱动 返回新设备场景句柄,若出错则为零 EMFStreamPrintDocument 实力问题:就算获得句柄也没有办法接下去要做什么.(想错了) 目前状态:EMF图片取出来,监控也可以实现了.但监控的打印作业跟EMF图片不知道怎么产生关联.而且EMF图片读起出来比较慢. 取EMF图片本身spl就已经读入内存,但是有一种办法为了要读取图片只能将spl文件考出来再做成emf文件. 新的思路能不能将文件 shd文件中包含了一个作业ID RPC 命名管道 Server(服务器) Server 系统服务提供 RPC 支持以及文件、打印和命名管道在网络上的共享。Server 服务允许本地资源(如磁盘和打印机)共享,因此网络上的其他用户可以访问它们。它还允许在其他计算机上的应用程序与您计算机上的应用程序之间进行命名管道通信,这是用于 RPC 的。命名管道通信是为一个进程的输出(此输出用作另外一个进程的输入)而保留的内存。接受输入的进程不必是本地进程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值