Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3118 lines
75 KiB

  1. /*++
  2. Copyright (C) Microsoft Corporation, 1995 - 1999
  3. All rights reserved.
  4. Module Name:
  5. queue.cxx
  6. Abstract:
  7. Manages the print queue.
  8. This module is aware of the ListView.
  9. Author:
  10. Albert Ting (AlbertT) 15-Jun-1995
  11. Revision History:
  12. --*/
  13. #include "precomp.hxx"
  14. #pragma hdrstop
  15. #include "notify.hxx"
  16. #include "data.hxx"
  17. #include "printer.hxx"
  18. #include "dragdrop.hxx"
  19. #include "queue.hxx"
  20. #include "time.hxx"
  21. #include "psetup.hxx"
  22. #include "drvsetup.hxx"
  23. #include "instarch.hxx"
  24. #include "portslv.hxx"
  25. #include "dsinterf.hxx"
  26. #include "prtprop.hxx"
  27. #include "propmgr.hxx"
  28. #include "docdef.hxx"
  29. #include "docprop.hxx"
  30. #include "persist.hxx"
  31. #include "rtlmir.hxx"
  32. #include "guids.h"
  33. #if DBG
  34. //#define DBG_QUEUEINFO DBG_INFO
  35. #define DBG_QUEUEINFO DBG_NONE
  36. #endif
  37. const TQueue::POSINFO TQueue::gPQPos = {
  38. TDataNJob::kColumnFieldsSize,
  39. {
  40. JOB_COLUMN_FIELDS,
  41. 0, 0, 0, 0, 0, 0, 0,
  42. 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
  43. },
  44. {
  45. 200, 80, 80, 60, 100, 140, 120, 80, 80, 80, 80, 80, 80, 80,
  46. 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80, 80
  47. },
  48. {
  49. sizeof( WINDOWPLACEMENT ), 0, SW_SHOW,
  50. { 0, 0 }, { 0, 0 }, { 50, 100, 612, 300 }
  51. },
  52. TRUE,
  53. TRUE,
  54. {
  55. 0, 1, 2, 3, 4, 5, 6, 7,
  56. 8, 9, 10, 11, 12, 13, 14, 15,
  57. 16, 17, 18, 19, 20, 21, 22, 23,
  58. 24, 25, 26, 27, 28, 29, 30, 31,
  59. 0,
  60. }
  61. };
  62. const DWORD
  63. gadwFieldTable[] = {
  64. #define DEFINE( field, x, table, y, offset ) table,
  65. #include "ntfyjob.h"
  66. #undef DEFINE
  67. 0
  68. };
  69. /********************************************************************
  70. Status translation tables:
  71. ********************************************************************/
  72. const STATUS_MAP gaStatusMapPrinter[] = {
  73. { PRINTER_STATUS_PENDING_DELETION, IDS_STATUS_DELETING },
  74. { PRINTER_STATUS_USER_INTERVENTION,IDS_STATUS_USER_INTERVENTION },
  75. { PRINTER_STATUS_PAPER_JAM, IDS_STATUS_PAPER_JAM },
  76. { PRINTER_STATUS_PAPER_OUT, IDS_STATUS_PAPER_OUT },
  77. { PRINTER_STATUS_MANUAL_FEED, IDS_STATUS_MANUAL_FEED },
  78. { PRINTER_STATUS_DOOR_OPEN, IDS_STATUS_DOOR_OPEN },
  79. { PRINTER_STATUS_NOT_AVAILABLE, IDS_STATUS_NOT_AVAILABLE },
  80. { PRINTER_STATUS_PAPER_PROBLEM, IDS_STATUS_PAPER_PROBLEM },
  81. { PRINTER_STATUS_OFFLINE, IDS_STATUS_OFFLINE },
  82. { PRINTER_STATUS_PAUSED, IDS_STATUS_PAUSED },
  83. { PRINTER_STATUS_OUT_OF_MEMORY, IDS_STATUS_OUT_OF_MEMORY },
  84. { PRINTER_STATUS_NO_TONER, IDS_STATUS_NO_TONER },
  85. { PRINTER_STATUS_TONER_LOW, IDS_STATUS_TONER_LOW },
  86. { PRINTER_STATUS_PAGE_PUNT, IDS_STATUS_PAGE_PUNT },
  87. { PRINTER_STATUS_OUTPUT_BIN_FULL, IDS_STATUS_OUTPUT_BIN_FULL },
  88. { PRINTER_STATUS_SERVER_UNKNOWN, IDS_STATUS_SERVER_UNKNOWN },
  89. { PRINTER_STATUS_IO_ACTIVE, IDS_STATUS_IO_ACTIVE },
  90. { PRINTER_STATUS_BUSY, IDS_STATUS_BUSY },
  91. { PRINTER_STATUS_WAITING, IDS_STATUS_WAITING },
  92. { PRINTER_STATUS_PROCESSING, IDS_STATUS_PROCESSING },
  93. { PRINTER_STATUS_INITIALIZING, IDS_STATUS_INITIALIZING },
  94. { PRINTER_STATUS_WARMING_UP, IDS_STATUS_WARMING_UP },
  95. { PRINTER_STATUS_PRINTING, IDS_STATUS_PRINTING },
  96. { PRINTER_STATUS_POWER_SAVE, IDS_STATUS_POWER_SAVE },
  97. { 0, 0 }
  98. };
  99. const STATUS_MAP gaStatusMapJob[] = {
  100. { JOB_STATUS_DELETING, IDS_STATUS_DELETING },
  101. { JOB_STATUS_PAPEROUT, IDS_STATUS_PAPER_OUT },
  102. { JOB_STATUS_ERROR , IDS_STATUS_ERROR },
  103. { JOB_STATUS_OFFLINE , IDS_STATUS_OFFLINE },
  104. { JOB_STATUS_PAUSED , IDS_STATUS_PAUSED },
  105. { JOB_STATUS_SPOOLING, IDS_STATUS_SPOOLING },
  106. { JOB_STATUS_PRINTING, IDS_STATUS_PRINTING },
  107. { JOB_STATUS_PRINTED , IDS_STATUS_PRINTED },
  108. { JOB_STATUS_RESTART , IDS_STATUS_RESTART },
  109. { JOB_STATUS_COMPLETE, IDS_STATUS_COMPLETE },
  110. { 0, 0 }
  111. };
  112. /********************************************************************
  113. MenuHelp
  114. ********************************************************************/
  115. UINT
  116. TQueue::gauMenuHelp[kMenuHelpMax] = {
  117. MH_PRINTER, MH_PRINTER,
  118. 0, 0
  119. };
  120. TQueue::
  121. TQueue(
  122. IN TPrintLib *pPrintLib,
  123. IN LPCTSTR pszPrinter,
  124. IN HANDLE hEventClose
  125. ) : _hwndTB( NULL ), _hwndLV( NULL ), _hwndSB( NULL ),
  126. _idsConnectStatus( 0 ), _dwErrorStatus( 0 ), _dwAttributes( 0 ),
  127. _dwStatusPrinter( 0 ), _hEventClose( hEventClose ),
  128. _bWindowClosing( FALSE ), _cItems( -1 ),
  129. _pDropTarget( NULL )
  130. /*++
  131. Routine Description:
  132. Create the Queue object. The gpPrintLib has already been incremented
  133. for us, so we do not need to do it here.
  134. Must be in UI thread so that all UI is handled by one thread.
  135. Arguments:
  136. hwndOwner - Owning window.
  137. pszPrinter - Printer to open.
  138. nCmdShow - Show command for window.
  139. hEventClose - Event to be triggered when window closes (this
  140. event is _not_ adopted and must be closed by callee). Used
  141. when the callee wants to know when the user dismisses the
  142. Queue UI.
  143. Return Value:
  144. --*/
  145. {
  146. ASSERT(pPrintLib);
  147. SINGLETHREAD(UIThread);
  148. //
  149. // This must always occur, so do not fail before it. We do
  150. // are using a RefLock because we need to store the _pPrintLib
  151. // pointer.
  152. //
  153. _pPrintLib.vAcquire(pPrintLib);
  154. SaveSelection._pSelection = NULL;
  155. _pPrinter = TPrinter::pNew( (TQueue*)this, pszPrinter, 0 );
  156. if( _pPrinter && !VALID_PTR(_pPrinter) )
  157. {
  158. //
  159. // bValid is looking for _pPrinter to determine if the object is valid.
  160. //
  161. _pPrinter->vDelete();
  162. _pPrinter = NULL;
  163. }
  164. _bDefaultPrinter = CheckDefaultPrinter( pszPrinter ) == kDefault;
  165. }
  166. TQueue::
  167. ~TQueue(
  168. VOID
  169. )
  170. {
  171. //
  172. // Delete from our linked list if it's linked.
  173. //
  174. if( Queue_bLinked( )){
  175. CCSLock::Locker CSL( *gpCritSec );
  176. Queue_vDelinkSelf();
  177. }
  178. }
  179. //
  180. // this function is taken from the comctl32 code since shfuson doesn't implement a warpper
  181. // and we need to take care of calling CreateWindowEx appropriately.
  182. //
  183. HWND WINAPI FusionWrapper_CreateStatusWindow(LONG style, LPCTSTR pszText, HWND hwndParent, UINT uID)
  184. {
  185. // remove border styles to fix capone and other apps
  186. return CreateWindowEx(0, STATUSCLASSNAME, pszText, style & ~(WS_BORDER | CCS_NODIVIDER),
  187. -100, -100, 10, 10, hwndParent, INT2PTR(uID, HMENU), ghInst, NULL);
  188. }
  189. BOOL
  190. TQueue::
  191. bInitialize(
  192. IN HWND hwndOwner,
  193. IN INT nCmdShow
  194. )
  195. /*++
  196. Routine Description:
  197. Creates the queue window and thunks it to our object.
  198. Arguments:
  199. Return Value:
  200. --*/
  201. {
  202. SINGLETHREAD(UIThread);
  203. HIMAGELIST himl;
  204. DWORD dwExStyle;
  205. if( !bValid() ) {
  206. goto Error;
  207. }
  208. _hwnd = CreateWindowEx( bIsBiDiLocalizedSystem() ? kExStyleRTLMirrorWnd : 0,
  209. gszClassName,
  210. pPrinter()->strPrinter(),
  211. WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN,
  212. CW_USEDEFAULT, CW_USEDEFAULT,
  213. CW_USEDEFAULT, CW_USEDEFAULT,
  214. hwndOwner,
  215. NULL,
  216. ghInst,
  217. (LPVOID)this );
  218. if( !_hwnd ){
  219. goto Error;
  220. }
  221. _hwndSB = FusionWrapper_CreateStatusWindow( WS_CHILD | SBARS_SIZEGRIP |
  222. WS_CLIPSIBLINGS,
  223. NULL,
  224. _hwnd,
  225. IDD_STATUSBAR );
  226. if( !_hwndSB ){
  227. goto Error;
  228. }
  229. _hwndLV = CreateWindowEx( WS_EX_CLIENTEDGE,
  230. WC_LISTVIEW,
  231. gszNULL,
  232. WS_CHILD | WS_VISIBLE | WS_TABSTOP |
  233. WS_CLIPSIBLINGS | LVS_REPORT | LVS_NOSORTHEADER,
  234. 0, 0, 0, 0,
  235. _hwnd,
  236. (HMENU)IDD_LISTVIEW,
  237. ghInst,
  238. NULL );
  239. if( !_hwndLV ){
  240. goto Error;
  241. }
  242. if( SUCCEEDED(DragDrop::CreatePrintQueueDT(IID_IPrintQueueDT, (void **)&_pDropTarget)) )
  243. {
  244. _pDropTarget->RegisterDragDrop(_hwndLV, _pPrinter);
  245. }
  246. //
  247. // Enable header re-ordering.
  248. //
  249. dwExStyle = ListView_GetExtendedListViewStyle( _hwndLV );
  250. ListView_SetExtendedListViewStyle( _hwndLV, dwExStyle | LVS_EX_HEADERDRAGDROP | LVS_EX_LABELTIP );
  251. //
  252. // !! LATER !! - add toolbar.
  253. //
  254. // NO toolbar for now.
  255. //
  256. himl = ImageList_Create( gcxSmIcon,
  257. gcySmIcon,
  258. ILC_MASK,
  259. 1,
  260. 3 );
  261. if( himl ){
  262. ImageList_SetBkColor( himl, GetSysColor( COLOR_WINDOW ));
  263. HICON hIcon = (HICON)LoadImage( ghInst,
  264. MAKEINTRESOURCE( IDI_DOCUMENT ),
  265. IMAGE_ICON,
  266. gcxSmIcon, gcySmIcon,
  267. LR_DEFAULTCOLOR );
  268. if( hIcon ){
  269. INT iIndex = ImageList_AddIcon( himl, hIcon );
  270. //
  271. // The return value to ImageList_AddIcon should be zero
  272. // since this is the first image we are adding.
  273. //
  274. if( iIndex == 0 ){
  275. SendMessage( _hwndLV, LVM_SETIMAGELIST, LVSIL_SMALL, (LPARAM)himl );
  276. } else {
  277. DBGMSG( DBG_WARN,
  278. ( "Queue.ctr: ImageList_AddIcon failed %d %d\n",
  279. iIndex, GetLastError( )));
  280. }
  281. DestroyIcon( hIcon );
  282. } else {
  283. DBGMSG( DBG_ERROR,
  284. ( "Queue.ctr: Failed to load hIcon %d\n",
  285. GetLastError( )));
  286. }
  287. }
  288. SetWindowLongPtr( _hwnd, DWLP_USER, (LONG_PTR)this );
  289. {
  290. //
  291. // Retrieve the saved windows settings for the printer.
  292. //
  293. POSINFO sPos = gPQPos;
  294. //
  295. // Get the the queue position persistant setting if it does not already
  296. // exist create it. Then read the posistion from the registry.
  297. //
  298. {
  299. TPersist Persist( gszPrinterPositions, TPersist::kCreate|TPersist::kRead );
  300. if( VALID_OBJ( Persist ) )
  301. {
  302. DWORD dwSize = sizeof( sPos );
  303. TStatusB bStatus;
  304. bStatus DBGCHK = Persist.bRead( pPrinter()->strPrinter(), &sPos, dwSize );
  305. }
  306. }
  307. _uColMax = sPos.uColMax;
  308. _bStatusBar = sPos.bStatusBar;
  309. ShowWindow( _hwndSB, _bStatusBar ? SW_SHOW : SW_HIDE );
  310. vAddColumns( &sPos );
  311. LoadPrinterIcons( _pPrinter->strPrinter(),
  312. &_shIconLarge,
  313. &_shIconSmall );
  314. SendMessage( _hwnd, WM_SETICON, ICON_BIG, (LPARAM)(HICON)_shIconLarge );
  315. SendMessage( _hwnd, WM_SETICON, ICON_SMALL, (LPARAM)(HICON)_shIconSmall );
  316. sPos.wp.showCmd = nCmdShow;
  317. SetWindowPlacement( _hwnd, &sPos.wp );
  318. //
  319. // Restore the column order
  320. //
  321. ListView_SetColumnOrderArray( _hwndLV, _uColMax, sPos.anColOrder );
  322. }
  323. //
  324. // Open the printer.
  325. //
  326. _pPrintLib->bJobAdd( _pPrinter,
  327. TPrinter::kExecReopen );
  328. //
  329. // hwndLV is our valid check.
  330. //
  331. //
  332. // Insert into our linked list, but only if valid.
  333. //
  334. {
  335. CCSLock::Locker CSL( *gpCritSec );
  336. SPLASSERT( bValid( ));
  337. _pPrintLib->Queue_vAdd( this );
  338. }
  339. return TRUE;
  340. Error:
  341. return FALSE;
  342. }
  343. VOID
  344. TQueue::
  345. vWindowClosing(
  346. VOID
  347. )
  348. /*++
  349. Routine Description:
  350. Called when window is closing.
  351. Arguments:
  352. Return Value:
  353. --*/
  354. {
  355. SINGLETHREAD(UIThread);
  356. //
  357. // Mark ourselves as closing the windows. This prevents us from
  358. // trying to send more hBlocks to the message queue.
  359. //
  360. _bWindowClosing = TRUE;
  361. SendMessage( _hwnd, WM_SETICON, ICON_SMALL, 0 );
  362. SendMessage( _hwnd, WM_SETICON, ICON_BIG, 0 );
  363. //
  364. // Force cleanup of GenWin.
  365. //
  366. vForceCleanup();
  367. //
  368. // Disassociate the printer from the queue. At this stage, the
  369. // window is marked as closed, so we won't put any more hBlocks into
  370. // the message queue. If we are being accessed by another thread,
  371. // we won't delete ourselves until it has released it's reference
  372. // to us.
  373. //
  374. if( _pPrinter )
  375. {
  376. _pPrinter->vDelete();
  377. }
  378. if( _hEventClose )
  379. {
  380. // notify the caller we are about to go away
  381. SetEvent(_hEventClose);
  382. }
  383. }
  384. VOID
  385. TQueue::
  386. vSaveColumns(
  387. VOID
  388. )
  389. {
  390. //
  391. // Save the position info if we had a valid window.
  392. //
  393. if( bValid( )){
  394. POSINFO sPos = { 0 };
  395. sPos.uColMax = _uColMax;
  396. sPos.wp.length = sizeof( WINDOWPLACEMENT );
  397. GetWindowPlacement( _hwnd, &sPos.wp );
  398. //
  399. // Get the column widths.
  400. //
  401. UINT i;
  402. PFIELD pFields = pGetColFields();
  403. for( i=0; i < _uColMax; ++i ){
  404. sPos.anWidth[i] = ListView_GetColumnWidth( _hwndLV, i );
  405. sPos.aField[i] = pFields[i];
  406. }
  407. sPos.bStatusBar = _bStatusBar;
  408. //
  409. // Get the list views column order.
  410. //
  411. ListView_GetColumnOrderArray( _hwndLV, _uColMax, sPos.anColOrder );
  412. //
  413. // Persist the queue position.
  414. //
  415. TPersist Persist( gszPrinterPositions, TPersist::kOpen|TPersist::kWrite );
  416. if( VALID_OBJ( Persist ) )
  417. {
  418. TCHAR szPrinter[kPrinterBufMax];
  419. TStatusB bStatus;
  420. bStatus DBGCHK = Persist.bWrite( _pPrinter->pszPrinterName( szPrinter ), &sPos, sizeof( sPos ) );
  421. }
  422. }
  423. }
  424. VOID
  425. TQueue::
  426. vAddColumns(
  427. IN const POSINFO* pPosInfo
  428. )
  429. {
  430. SINGLETHREAD(UIThread);
  431. LV_COLUMN col;
  432. TCHAR szColName[kColStrMax];
  433. UINT i;
  434. for( i=0; i < pPosInfo->uColMax; ++i ){
  435. //
  436. // !! SERVERQUEUE !!
  437. //
  438. // Add IDS_HEAD_DELTA if server queue.
  439. //
  440. LoadString( ghInst,
  441. IDS_HEAD + pPosInfo->aField[i],
  442. szColName,
  443. COUNTOF( szColName ));
  444. col.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM;
  445. col.fmt = LVCFMT_LEFT;
  446. col.pszText = (LPTSTR)szColName;
  447. col.cchTextMax = 0;
  448. col.cx = pPosInfo->anWidth[i];
  449. col.iSubItem = pPosInfo->aField[i];
  450. ListView_InsertColumn(_hwndLV, i, &col);
  451. }
  452. }
  453. /********************************************************************
  454. Message handler.
  455. ********************************************************************/
  456. LRESULT
  457. TQueue::
  458. nHandleMessage(
  459. IN UINT uMsg,
  460. IN WPARAM wParam,
  461. IN LPARAM lParam
  462. )
  463. {
  464. switch(uMsg) {
  465. case WM_PRINTLIB_STATUS: {
  466. INFO Info;
  467. Info.dwData = (DWORD)lParam;
  468. //
  469. // Status change request from worker thread.
  470. //
  471. vContainerChangedHandler( (CONTAINER_CHANGE)wParam, Info );
  472. break;
  473. }
  474. case WM_NOTIFY:
  475. if( wParam == IDD_LISTVIEW ){
  476. return lrOnLVNotify( lParam );
  477. }
  478. break;
  479. case WM_SETFOCUS:
  480. SetFocus( _hwndLV );
  481. break;
  482. case WM_CREATE:
  483. //
  484. // The window was successfully created, so increment the
  485. // reference count. The corresponding decrement is when the
  486. // windows is destroyed: WM_NCDESTROY.
  487. //
  488. vIncRef();
  489. break;
  490. case WM_DESTROY:
  491. if( _pDropTarget )
  492. {
  493. // revoke drag & drop
  494. _pDropTarget->RevokeDragDrop();
  495. // this a naked COM pointer - release it.
  496. _pDropTarget->Release();
  497. _pDropTarget = NULL;
  498. }
  499. vSaveColumns();
  500. break;
  501. case WM_NCDESTROY:
  502. {
  503. //
  504. // Deleting ourselves must be the absolute last thing that
  505. // we do; we don't want any more messages to get processed
  506. // after that.
  507. //
  508. // This is necessary because in the synchronous case, we
  509. // (thread A) notifies the waiter (thread W) that the queue
  510. // has gone away. Then we (thread A) call DefWindowProc
  511. // with WM_NCDESTROY, which lets comctl32 acquire it's global
  512. // critical section to destory the image list.
  513. //
  514. // In rundll32, thread W terminates since the queue is gone,
  515. // killing thread A which hold the comctl32 global cs. Then
  516. // thread w tries to call comctl32's DllEntryPoint with
  517. // PROCESS_DETACH, which attempts to acquire the global cs.
  518. // This hangs and the process never terminates.
  519. //
  520. LRESULT lResult = DefWindowProc( hwnd(), uMsg, wParam, lParam );
  521. //
  522. // Save out the settings since we are closing the window.
  523. //
  524. vWindowClosing();
  525. //
  526. // Decrement the reference count since we are deleting.
  527. //
  528. vDecRefDelete();
  529. return lResult;
  530. }
  531. case WM_APPCOMMAND:
  532. {
  533. if( APPCOMMAND_BROWSER_REFRESH == GET_APPCOMMAND_LPARAM(lParam) )
  534. {
  535. //
  536. // Execute refresh.
  537. //
  538. SendMessage( _hwnd, WM_COMMAND, MAKELONG(IDM_REFRESH,0), 0);
  539. }
  540. break;
  541. }
  542. case WM_COMMAND:
  543. {
  544. lParam = 0;
  545. TCHAR szPrinterBuffer[kPrinterBufMax];
  546. LPTSTR pszPrinter;
  547. switch( GET_WM_COMMAND_ID( wParam, lParam )){
  548. case IDM_PRINTER_SET_DEFAULT:
  549. //
  550. // Always write out the default string. User can't
  551. // unset default printer.
  552. //
  553. pszPrinter = _pPrinter->pszPrinterName( szPrinterBuffer );
  554. //
  555. // !! LATER !!
  556. //
  557. // Put up error message if fails.
  558. //
  559. SetDefaultPrinter( szPrinterBuffer );
  560. break;
  561. case IDM_PRINTER_SHARING:
  562. //
  563. // Put up printer properties. If sharing was selected,
  564. // then go directly to that page.
  565. //
  566. lParam = TPrinterData::kPropSharing;
  567. //
  568. // Fall through to printer properties.
  569. //
  570. case IDM_PRINTER_PROPERTIES:
  571. pszPrinter = _pPrinter->pszPrinterName( szPrinterBuffer );
  572. vPrinterPropPages(
  573. NULL,
  574. pszPrinter,
  575. SW_SHOWNORMAL,
  576. lParam );
  577. break;
  578. case IDM_PRINTER_DOCUMENT_DEFAULTS:
  579. pszPrinter = _pPrinter->pszPrinterName( szPrinterBuffer );
  580. vDocumentDefaults(
  581. NULL,
  582. pszPrinter,
  583. SW_SHOWNORMAL,
  584. lParam );
  585. break;
  586. case IDM_PRINTER_CLOSE:
  587. DestroyWindow( _hwnd );
  588. return 0;
  589. case IDM_STATUS_BAR:
  590. {
  591. RECT rc;
  592. _bStatusBar = !_bStatusBar;
  593. ShowWindow( _hwndSB,
  594. _bStatusBar ?
  595. SW_SHOW :
  596. SW_HIDE );
  597. GetClientRect( _hwnd, &rc );
  598. SendMessage( _hwnd,
  599. WM_SIZE,
  600. SIZE_RESTORED,
  601. MAKELONG( rc.right, rc.bottom ));
  602. break;
  603. }
  604. case IDM_PRINTER_INSTALL:
  605. if( !CheckRestrictions(_hwnd, REST_NOPRINTERADD) )
  606. {
  607. TStatusB bStatus;
  608. TCHAR szPrinter[kPrinterBufMax];
  609. LPTSTR pszPrinterName;
  610. pszPrinterName = _pPrinter->pszPrinterName(szPrinter);
  611. bStatus DBGCHK = bPrinterSetup( _hwnd,
  612. MSP_NETPRINTER,
  613. ARRAYSIZE(szPrinter),
  614. pszPrinterName,
  615. NULL,
  616. NULL );
  617. }
  618. break;
  619. case IDM_REFRESH:
  620. _pPrintLib->bJobAdd( _pPrinter,
  621. TPrinter::kExecRefreshAll );
  622. _pPrinter->vCommandRequested();
  623. break;
  624. case IDM_HELP_CONTENTS:
  625. PrintUIHtmlHelp( _hwnd, gszHtmlPrintingHlp, HH_DISPLAY_TOPIC, reinterpret_cast<ULONG_PTR>( gszHelpQueueId ) );
  626. break;
  627. case IDM_HELP_TROUBLESHOOTER:
  628. {
  629. ShellExecute( _hwnd, TEXT("open"), TEXT("helpctr.exe"), gszHelpTroubleShooterURL, NULL, SW_SHOWNORMAL );
  630. }
  631. break;
  632. case IDM_HELP_ABOUT:
  633. {
  634. TString strWindows;
  635. TString strTitle;
  636. LCID lcid;
  637. BOOL bNeedLRM;
  638. //
  639. // if we have Arabic/Hebrew locale we add the text as LRM + Text + LRM
  640. // to force left-to-right reading (LRM is the unicode character 0x200E)
  641. //
  642. lcid = GetUserDefaultLCID();
  643. bNeedLRM = (PRIMARYLANGID(LANGIDFROMLCID(lcid))== LANG_ARABIC) ||
  644. (PRIMARYLANGID(LANGIDFROMLCID(lcid))== LANG_HEBREW);
  645. if (bNeedLRM)
  646. {
  647. strTitle.bUpdate(TEXT("\x200E"));
  648. }
  649. strWindows.bLoadString( ghInst, IDS_WINDOWS );
  650. strTitle.bCat(strWindows);
  651. if (bNeedLRM)
  652. {
  653. strTitle.bCat(TEXT("\x200E"));
  654. }
  655. ShellAbout( _hwnd, strTitle, NULL, NULL );
  656. }
  657. break;
  658. default:
  659. return lrProcessCommand( GET_WM_COMMAND_ID( wParam, lParam ));
  660. }
  661. break;
  662. }
  663. case WM_ACTIVATE:
  664. //
  665. // We must pass the active window to TranslateAccelerator,
  666. // so when the active window for our app changes, make
  667. // a note of it.
  668. //
  669. if( LOWORD( wParam ) & ( WA_ACTIVE | WA_CLICKACTIVE )){
  670. ghwndActive = _hwnd;
  671. }
  672. break;
  673. case WM_INITMENU:
  674. {
  675. HMENU hMenu = GetMenu( _hwnd );
  676. if( hMenu ){
  677. HMENU hTemp = GetSubMenu( hMenu, 0 );
  678. if( hTemp ){
  679. vInitPrinterMenu( hTemp );
  680. }
  681. hTemp = GetSubMenu( hMenu, 1 );
  682. if( hTemp ){
  683. vInitDocMenu( FALSE, hTemp );
  684. }
  685. hTemp = GetSubMenu( hMenu, 2 );
  686. if( hTemp ){
  687. vInitViewMenu( hTemp );
  688. }
  689. }
  690. }
  691. break;
  692. case WM_MENUSELECT:
  693. if( _bStatusBar ){
  694. MenuHelp( WM_MENUSELECT,
  695. wParam,
  696. lParam,
  697. GetMenu( _hwnd ),
  698. ghInst,
  699. _hwndSB,
  700. gauMenuHelp );
  701. }
  702. break;
  703. case WM_SIZE:
  704. if( wParam != SIZE_MINIMIZED ){
  705. UINT dy = 0;
  706. RECT rc;
  707. SendMessage( _hwndSB,
  708. WM_SIZE,
  709. wParam,
  710. lParam );
  711. GetWindowRect( _hwndSB,
  712. &rc );
  713. //
  714. // If the status bar exists, then we must move it to the
  715. // bottom and squeeze the listview slightly higher.
  716. //
  717. if( _bStatusBar ){
  718. dy = rc.bottom - rc.top;
  719. }
  720. INT aiPanes[3];
  721. aiPanes[0] = 0;
  722. aiPanes[1] = (rc.right - rc.left)/2;
  723. aiPanes[2] = -1;
  724. //
  725. // Put three panes there.
  726. //
  727. SendMessage( _hwndSB,
  728. SB_SETPARTS,
  729. 3,
  730. (LPARAM)aiPanes );
  731. //
  732. // Move this list view to match the parent window.
  733. //
  734. MoveWindow( _hwndLV,
  735. 0, 0,
  736. LOWORD( lParam ), HIWORD( lParam ) - dy,
  737. TRUE );
  738. }
  739. break;
  740. case WM_COPYDATA:
  741. return bOnCopyData( wParam, lParam );
  742. case WM_SETTINGCHANGE:
  743. {
  744. //
  745. // Check if the default printer has changed, to update the icon.
  746. //
  747. vCheckDefaultPrinterChanged();
  748. //
  749. // This message is sent when the system date/time format
  750. // has been changed so we need to repaint the list view.
  751. //
  752. InvalidateRect(_hwndLV, NULL, TRUE);
  753. }
  754. break;
  755. default:
  756. return DefWindowProc( hwnd(), uMsg, wParam, lParam );
  757. }
  758. return 0;
  759. }
  760. LRESULT
  761. TQueue::
  762. lrOnLVNotify(
  763. IN LPARAM lParam
  764. )
  765. {
  766. switch( ((LPNMHDR)lParam)->code ){
  767. case LVN_GETDISPINFO:
  768. return lrOnLVGetDispInfo( (LV_DISPINFO*)lParam );
  769. case LVN_BEGINDRAG:
  770. case LVN_BEGINRDRAG:
  771. // return lrOnLVBeginDrag( reinterpret_cast<const NM_LISTVIEW*>(lParam) );
  772. return 0; // turn off this feature for XP, will turn it back on for Blackcomb
  773. case NM_DBLCLK:
  774. return lrOnLVDoubleClick();
  775. #if 0
  776. case LVN_COLUMNCLICK:
  777. return lrOnLVColumnClick( (COLUMN)(( NM_LISTVIEW* )lParam )->iSubItem );
  778. #endif
  779. case NM_RCLICK:
  780. return lrOnLVRClick( (NMHDR*)lParam );
  781. }
  782. return 0;
  783. }
  784. LRESULT
  785. TQueue::
  786. lrOnLVGetDispInfo(
  787. IN const LV_DISPINFO* plvdi
  788. )
  789. /*++
  790. Routine Description:
  791. Process the display info message for list views.
  792. Arguments:
  793. Return Value:
  794. --*/
  795. {
  796. //
  797. // If this message is not going to retrieve the text, return immediately
  798. //
  799. if( !(plvdi->item.mask & LVIF_TEXT) )
  800. {
  801. return 0;
  802. }
  803. LPTSTR pszText = plvdi->item.pszText;
  804. pszText[0] = 0;
  805. FIELD Field = gPQPos.aField[plvdi->item.iSubItem];
  806. INFO Info = _pPrinter->pData()->GetInfo( (HITEM)plvdi->item.lParam,
  807. plvdi->item.iSubItem );
  808. DATA_INDEX DataIndex = 0;
  809. DWORD dwPrinted;
  810. //
  811. // Special case certain fields:
  812. //
  813. // JOB_NOTIFY_FIELD_STATUS_STRING - add STATUS
  814. // JOB_NOTIFY_FIELD_TOTAL_BYTES - add BYTES_PRINTED
  815. // JOB_NOTIFY_FIELD_TOTAL_PAGES - add PAGES_PRINTED
  816. //
  817. switch( Field ){
  818. case JOB_NOTIFY_FIELD_STATUS_STRING:
  819. {
  820. DWORD dwStatus = 0;
  821. COUNT cch = kStrMax;
  822. //
  823. // If the print device wants multiple job status strings or
  824. // current job status string is null then create a job status
  825. // string using the job status bits.
  826. //
  827. if( _pPrinter->eJobStatusStringType() == TPrinter::kMultipleJobStatusString ||
  828. !Info.pszData ||
  829. !Info.pszData[0] ){
  830. dwStatus = _pPrinter->pData()->GetInfo(
  831. (HITEM)plvdi->item.lParam,
  832. TDataNJob::kIndexStatus ).dwData;
  833. pszText = pszStatusString( pszText,
  834. cch, // Note: passed by reference
  835. dwStatus,
  836. FALSE,
  837. FALSE,
  838. gaStatusMapJob );
  839. }
  840. //
  841. // Add the status string, but not if it's PRINTING
  842. // and we already have PRINTING set.
  843. //
  844. if( Info.pszData && Info.pszData[0] ){
  845. TString strPrinting;
  846. TStatusB bStatus = TRUE;
  847. if( dwStatus & JOB_STATUS_PRINTING ){
  848. bStatus DBGCHK = strPrinting.bLoadString(ghInst, IDS_STATUS_PRINTING);
  849. if( bStatus ){
  850. bStatus DBGCHK = lstrcmpi( Info.pszData, strPrinting ) ? TRUE : FALSE;
  851. }
  852. }
  853. if( bStatus ){
  854. //
  855. // Add separator if necessary.
  856. //
  857. if( pszText != plvdi->item.pszText ){
  858. pszText = pszStrCat( pszText, gszStatusSeparator, cch );
  859. }
  860. lstrcpyn( pszText, Info.pszData, cch );
  861. }
  862. }
  863. DBGMSG( DBG_TRACE, ("Job info String: " TSTR "\n", pszText ) );
  864. return 0;
  865. }
  866. case JOB_NOTIFY_FIELD_TOTAL_BYTES:
  867. {
  868. dwPrinted = _pPrinter->pData()->GetInfo(
  869. (HITEM)plvdi->item.lParam,
  870. TDataNJob::kIndexBytesPrinted ).dwData;
  871. if( dwPrinted )
  872. {
  873. TString strForwardSlash;
  874. TStatusB bStatus;
  875. bStatus DBGCHK = strForwardSlash.bLoadString(ghInst, IDS_QUEUE_FORWARD_SLASH);
  876. if( bStatus )
  877. {
  878. TCHAR szPrinted[64];
  879. TCHAR szTotal[64];
  880. StrFormatByteSize(dwPrinted, szPrinted, ARRAYSIZE(szPrinted));
  881. StrFormatByteSize(Info.dwData, szTotal, ARRAYSIZE(szTotal));
  882. wnsprintf(plvdi->item.pszText, plvdi->item.cchTextMax, TEXT("%s%s%s"),
  883. szPrinted, static_cast<LPCTSTR>(strForwardSlash), szTotal);
  884. }
  885. }
  886. else
  887. {
  888. if( Info.dwData )
  889. {
  890. TCHAR szTotal[64];
  891. StrFormatByteSize(Info.dwData, szTotal, ARRAYSIZE(szTotal));
  892. wnsprintf(plvdi->item.pszText, plvdi->item.cchTextMax, TEXT("%s"), szTotal);
  893. }
  894. }
  895. return 0;
  896. }
  897. case JOB_NOTIFY_FIELD_TOTAL_PAGES:
  898. dwPrinted = _pPrinter->pData()->GetInfo(
  899. (HITEM)plvdi->item.lParam,
  900. TDataNJob::kIndexPagesPrinted ).dwData;
  901. //
  902. // when a downlevel document is printed (the doc goes directly to the port) StartDoc/EndDoc
  903. // are not called and the spooler doesn't know the total pages of the document. in this case
  904. // we don't display the pages info since it is not acurate.
  905. //
  906. if( Info.dwData ){
  907. if( dwPrinted ){
  908. AddCommas( dwPrinted, pszText );
  909. lstrcat( pszText, TEXT( "/" ));
  910. }
  911. AddCommas( Info.dwData, pszText + lstrlen( pszText ));
  912. }
  913. else
  914. {
  915. TStatusB bStatus;
  916. TString strText;
  917. bStatus DBGCHK = strText.bLoadString(ghInst, IDS_TEXT_NA);
  918. if( bStatus )
  919. {
  920. //
  921. // just display "N/A" text (we don't know the total pages)
  922. //
  923. lstrcpyn(plvdi->item.pszText, strText, plvdi->item.cchTextMax);
  924. }
  925. }
  926. return 0;
  927. default:
  928. break;
  929. }
  930. switch( gadwFieldTable[Field] ){
  931. case TABLE_STRING:
  932. //
  933. // If we have data, reassign the pointer,
  934. // else leave it pointing to szText, which is "".
  935. //
  936. if( Info.pszData ){
  937. lstrcpyn( pszText, Info.pszData, kStrMax );
  938. }
  939. break;
  940. case TABLE_DWORD:
  941. if( Info.dwData ){
  942. AddCommas( Info.dwData, pszText );
  943. }
  944. break;
  945. case TABLE_TIME:
  946. if( Info.pSystemTime ){
  947. SYSTEMTIME LocalTime;
  948. COUNT cchText;
  949. if ( !SystemTimeToTzSpecificLocalTime(
  950. NULL,
  951. Info.pSystemTime,
  952. &LocalTime )) {
  953. DBGMSG( DBG_MIN, ( "[SysTimeToTzSpecLocalTime failed %d]\n",
  954. ::GetLastError( )));
  955. break;
  956. }
  957. if( !GetTimeFormat( LOCALE_USER_DEFAULT,
  958. 0,
  959. &LocalTime,
  960. NULL,
  961. pszText,
  962. kStrMax )){
  963. DBGMSG( DBG_MIN, ( "[No Time %d], ", ::GetLastError( )));
  964. break;
  965. }
  966. lstrcat( pszText, TEXT(" ") );
  967. cchText = lstrlen( pszText );
  968. pszText += cchText;
  969. if( !GetDateFormat( LOCALE_USER_DEFAULT,
  970. dwDateFormatFlags( hwnd() ),
  971. &LocalTime,
  972. NULL,
  973. pszText,
  974. kStrMax - cchText )){
  975. DBGMSG( DBG_MIN, ( "[No Date %d]\n", ::GetLastError( )));
  976. break;
  977. }
  978. }
  979. break;
  980. default:
  981. DBGMSG( DBG_MIN, ( "[?tab %d %x]\n",
  982. Field,
  983. Info.pvData ));
  984. break;
  985. }
  986. return 0;
  987. }
  988. LRESULT
  989. TQueue::
  990. lrOnLVBeginDrag(
  991. const NM_LISTVIEW *plv
  992. )
  993. /*++
  994. Routine Description:
  995. Initiates drag & drop operation
  996. Arguments:
  997. Standard for LVN_BEGINDRAG & LVN_BEGINRDRAG
  998. Return Value:
  999. Standard for LVN_BEGINDRAG & LVN_BEGINRDRAG
  1000. --*/
  1001. {
  1002. LVITEM lvi = {0};
  1003. lvi.iItem = plv->iItem;
  1004. lvi.iSubItem = plv->iSubItem;
  1005. lvi.mask = LVIF_PARAM;
  1006. if( ListView_GetItem(_hwndLV, &lvi) )
  1007. {
  1008. DragDrop::JOBINFO jobInfo;
  1009. //
  1010. // Initialize job info.
  1011. //
  1012. jobInfo.dwJobID = _pPrinter->pData()->GetId( (HITEM)lvi.lParam );
  1013. jobInfo.hwndLV = _hwndLV;
  1014. jobInfo.iItem = lvi.iItem;
  1015. _pPrinter->pszPrinterName(jobInfo.szPrinterName);
  1016. //
  1017. // Kick off OLE2 drag & drop.
  1018. //
  1019. CRefPtrCOM<IDataObject> spDataObj;
  1020. CRefPtrCOM<IDropSource> spDropSrc;
  1021. if( SUCCEEDED(DragDrop::CreatePrintJobObject(jobInfo, IID_IDataObject, (void **)&spDataObj)) &&
  1022. SUCCEEDED(spDataObj->QueryInterface(IID_IDropSource, (void **)&spDropSrc)) )
  1023. {
  1024. DWORD dwEffect = DROPEFFECT_MOVE | DROPEFFECT_COPY;
  1025. SHDoDragDrop(_hwndLV, spDataObj, spDropSrc, dwEffect, &dwEffect);
  1026. }
  1027. }
  1028. return 0;
  1029. }
  1030. VOID
  1031. TQueue::
  1032. vInitPrinterMenu(
  1033. HMENU hMenu
  1034. )
  1035. {
  1036. //
  1037. // If printer is paused, enable pause, etc.
  1038. //
  1039. BOOL bPaused = _dwStatusPrinter & PRINTER_STATUS_PAUSED;
  1040. //
  1041. // Disable admin functions if not an administrator.
  1042. // We should guard this, but since it's just status, don't bother.
  1043. //
  1044. BOOL bAdministrator = _pPrinter->dwAccess() == PRINTER_ALL_ACCESS;
  1045. BOOL bDirect = _dwAttributes & PRINTER_ATTRIBUTE_DIRECT ? TRUE : FALSE;
  1046. TCHAR szPrinterBuffer[kPrinterBufMax];
  1047. LPTSTR pszPrinter;
  1048. pszPrinter = _pPrinter->pszPrinterName( szPrinterBuffer );
  1049. TCHAR szScratch[2];
  1050. // let's do a quick EnumPrinters call here and see if this printer is
  1051. // locally installed (local printer or printer connection)
  1052. BOOL bInstalled = FALSE;
  1053. DWORD cPrinters = 0, cbBuffer = 0;
  1054. DWORD dwIndex;
  1055. CAutoPtrSpl<PRINTER_INFO_5> spPrinters;
  1056. if( VDataRefresh::bEnumPrinters(PRINTER_ENUM_CONNECTIONS|PRINTER_ENUM_LOCAL,
  1057. NULL, 5, (void **)&spPrinters, &cbBuffer, &cPrinters) )
  1058. {
  1059. // linear search to see if our printer is installed locally,
  1060. for( DWORD dw = 0; dw < cPrinters; dw++ )
  1061. {
  1062. if( 0 == lstrcmp(spPrinters[dw].pPrinterName, pszPrinter) )
  1063. {
  1064. dwIndex = dw;
  1065. bInstalled = TRUE;
  1066. break;
  1067. }
  1068. }
  1069. }
  1070. BOOL bDefault = bInstalled ? CheckDefaultPrinter(pszPrinter) == kDefault : FALSE;
  1071. BOOL bIsLocal = bInstalled ? _pPrinter->pszServerName(szPrinterBuffer) == NULL : FALSE;
  1072. BOOL bIsNowOffline = _dwAttributes & PRINTER_ATTRIBUTE_WORK_OFFLINE;
  1073. CheckMenuItem( hMenu,
  1074. IDM_PRINTER_SET_DEFAULT,
  1075. bDefault ?
  1076. MF_BYCOMMAND|MF_CHECKED :
  1077. MF_BYCOMMAND|MF_UNCHECKED );
  1078. EnableMenuItem( hMenu,
  1079. IDM_PRINTER_SET_DEFAULT,
  1080. bInstalled ?
  1081. MF_BYCOMMAND|MF_ENABLED :
  1082. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED );
  1083. EnableMenuItem( hMenu,
  1084. IDM_PRINTER_INSTALL,
  1085. bInstalled ?
  1086. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED :
  1087. MF_BYCOMMAND|MF_ENABLED );
  1088. CheckMenuItem( hMenu,
  1089. IDM_PRINTER_PAUSE,
  1090. bPaused ?
  1091. MF_BYCOMMAND|MF_CHECKED :
  1092. MF_BYCOMMAND|MF_UNCHECKED );
  1093. EnableMenuItem( hMenu,
  1094. IDM_PRINTER_PAUSE,
  1095. bAdministrator ?
  1096. MF_BYCOMMAND|MF_ENABLED :
  1097. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED );
  1098. EnableMenuItem( hMenu,
  1099. IDM_PRINTER_PURGE,
  1100. bAdministrator ?
  1101. MF_BYCOMMAND|MF_ENABLED :
  1102. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED );
  1103. CheckMenuItem( hMenu,
  1104. IDM_PRINTER_WORKOFFLINE,
  1105. bIsNowOffline ?
  1106. MF_BYCOMMAND|MF_CHECKED :
  1107. MF_BYCOMMAND|MF_UNCHECKED );
  1108. EnableMenuItem( hMenu,
  1109. IDM_PRINTER_WORKOFFLINE,
  1110. bAdministrator ?
  1111. MF_BYCOMMAND|MF_ENABLED :
  1112. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED );
  1113. BOOL bIsRedirected = FALSE;
  1114. BOOL bIsMasq = _dwAttributes & PRINTER_ATTRIBUTE_LOCAL && _dwAttributes & PRINTER_ATTRIBUTE_NETWORK;
  1115. //
  1116. // We only check for redirected port for local printer.
  1117. // If bIsLocal is TRUE, bInstalled must be TRUE
  1118. //
  1119. if( bIsLocal )
  1120. {
  1121. IsRedirectedPort( spPrinters[dwIndex].pPortName, &bIsRedirected );
  1122. }
  1123. //
  1124. // Remove the work offline menu item if this printer is
  1125. // not local or the printer is a masq printer, or if it's a
  1126. // redirected port printer which is not offline now.
  1127. //
  1128. if( !_dwAttributes || !bIsLocal || (bIsRedirected && !bIsNowOffline) || bIsMasq )
  1129. {
  1130. RemoveMenu( hMenu, IDM_PRINTER_WORKOFFLINE, MF_BYCOMMAND );
  1131. }
  1132. }
  1133. VOID
  1134. TQueue::
  1135. vInitDocMenu(
  1136. BOOL bAllowModify,
  1137. HMENU hMenu
  1138. )
  1139. /*++
  1140. Routine Name:
  1141. vInitDocMenu
  1142. Routine Description:
  1143. Enables or disable the document menu selections.
  1144. Arguments:
  1145. bAllowModify - whether we allow deleting the menu item.
  1146. HMENU - Handle to document menu
  1147. Return Value:
  1148. --*/
  1149. {
  1150. INT iSel = ListView_GetNextItem( _hwndLV, -1, LVNI_SELECTED );
  1151. INT iSelCount = ListView_GetSelectedCount( _hwndLV );
  1152. UINT fuFlags = (iSelCount > 1 || iSel >= 0) ?
  1153. MF_BYCOMMAND|MF_ENABLED :
  1154. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED;
  1155. EnableMenuItem(hMenu, IDM_JOB_PAUSE, fuFlags);
  1156. EnableMenuItem(hMenu, IDM_JOB_RESUME, fuFlags);
  1157. EnableMenuItem(hMenu, IDM_JOB_RESTART, fuFlags);
  1158. EnableMenuItem(hMenu, IDM_JOB_CANCEL, fuFlags);
  1159. EnableMenuItem(hMenu, IDM_JOB_PROPERTIES, fuFlags);
  1160. //
  1161. // If more than one item are selected, we will enable all menu items.
  1162. //
  1163. if( iSelCount == 1 && iSel >= 0 && _pPrinter->pData() )
  1164. {
  1165. // let's see if we can un-clutter the menu a little bit...
  1166. LVITEM lvi;
  1167. ZeroMemory(&lvi, sizeof(lvi));
  1168. lvi.iItem = iSel;
  1169. lvi.mask = LVIF_PARAM;
  1170. if( ListView_GetItem(_hwndLV, &lvi) )
  1171. {
  1172. DWORD dwStatus = _pPrinter->pData()->GetInfo(
  1173. reinterpret_cast<HITEM>(lvi.lParam), TDataNJob::kIndexStatus).dwData;
  1174. if( bAllowModify )
  1175. {
  1176. // delete the corresponding menu item
  1177. DeleteMenu(hMenu, (dwStatus & JOB_STATUS_PAUSED) ? IDM_JOB_PAUSE : IDM_JOB_RESUME, MF_BYCOMMAND);
  1178. }
  1179. else
  1180. {
  1181. // disable the corresponding menu item
  1182. EnableMenuItem(hMenu,
  1183. (dwStatus & JOB_STATUS_PAUSED) ? IDM_JOB_PAUSE : IDM_JOB_RESUME,
  1184. MF_BYCOMMAND|MF_DISABLED|MF_GRAYED);
  1185. }
  1186. }
  1187. }
  1188. }
  1189. VOID
  1190. TQueue::
  1191. vInitViewMenu(
  1192. HMENU hMenu
  1193. )
  1194. {
  1195. CheckMenuItem( hMenu,
  1196. IDM_STATUS_BAR,
  1197. _bStatusBar ?
  1198. MF_BYCOMMAND | MF_CHECKED :
  1199. MF_BYCOMMAND | MF_UNCHECKED );
  1200. }
  1201. /********************************************************************
  1202. Support double click context menus.
  1203. ********************************************************************/
  1204. LRESULT
  1205. TQueue::
  1206. lrOnLVDoubleClick(
  1207. VOID
  1208. )
  1209. {
  1210. //
  1211. // We only handle when an item is selected.
  1212. //
  1213. if( ListView_GetNextItem( _hwndLV, -1, LVNI_SELECTED ) < 0 )
  1214. return FALSE;
  1215. //
  1216. // Prevent the selection of multiple jobs
  1217. //
  1218. #if 1
  1219. //
  1220. // If multiple job selections then error.
  1221. //
  1222. if( ListView_GetSelectedCount( _hwndLV ) > 1){
  1223. return FALSE;
  1224. }
  1225. #endif
  1226. //
  1227. // Display the selected job property.
  1228. //
  1229. vProcessItemCommand( IDM_JOB_PROPERTIES );
  1230. return TRUE;
  1231. }
  1232. /********************************************************************
  1233. Support right click context menus.
  1234. ********************************************************************/
  1235. LRESULT
  1236. TQueue::
  1237. lrOnLVRClick(
  1238. NMHDR* pnmhdr
  1239. )
  1240. {
  1241. LRESULT lReturn = TRUE;
  1242. switch( pnmhdr->code ){
  1243. case NM_RCLICK:
  1244. {
  1245. INT iSel;
  1246. HMENU hmContext;
  1247. POINT pt;
  1248. iSel = ListView_GetNextItem( _hwndLV, -1, LVNI_SELECTED );
  1249. hmContext = ShellServices::LoadPopupMenu(ghInst, MENU_PRINTQUEUE, iSel >= 0 ? 1 : 0);
  1250. if( !hmContext ){
  1251. break;
  1252. }
  1253. if( iSel < 0 ){
  1254. //
  1255. // We need to remove the "Close" menu item
  1256. // (and separator).
  1257. //
  1258. iSel = GetMenuItemCount( hmContext ) - 2;
  1259. DeleteMenu( hmContext, iSel, MF_BYPOSITION );
  1260. DeleteMenu( hmContext, iSel, MF_BYPOSITION );
  1261. vInitPrinterMenu( hmContext );
  1262. } else {
  1263. vInitDocMenu( TRUE, hmContext );
  1264. }
  1265. DWORD dw = GetMessagePos();
  1266. pt.x = GET_X_LPARAM(dw);
  1267. pt.y = GET_Y_LPARAM(dw);
  1268. //
  1269. // The command will just get stuck in the regular queue and
  1270. // handled at that time.
  1271. //
  1272. TrackPopupMenu( hmContext,
  1273. TPM_LEFTALIGN|TPM_RIGHTBUTTON,
  1274. pt.x, pt.y,
  1275. 0, _hwnd, NULL);
  1276. DestroyMenu(hmContext);
  1277. break;
  1278. }
  1279. default:
  1280. lReturn = FALSE;
  1281. break;
  1282. }
  1283. return lReturn;
  1284. }
  1285. /********************************************************************
  1286. Commands.
  1287. ********************************************************************/
  1288. LRESULT
  1289. TQueue::
  1290. lrProcessCommand(
  1291. IN UINT uCommand
  1292. )
  1293. /*++
  1294. Routine Description:
  1295. Process an IDM_* command.
  1296. Arguments:
  1297. Return Value:
  1298. LRESULT
  1299. --*/
  1300. {
  1301. //
  1302. // Item (printer) command.
  1303. //
  1304. if( uCommand >= IDM_PRINTER_COMMAND_FIRST && uCommand <= IDM_PRINTER_COMMAND_LAST )
  1305. {
  1306. TSelection* pSelection = new TSelection( this, _pPrinter );
  1307. if( pSelection )
  1308. {
  1309. switch( uCommand ){
  1310. case IDM_PRINTER_PAUSE:
  1311. pSelection->_CommandType = TSelection::kCommandTypePrinter;
  1312. pSelection->_dwCommandAction = _dwStatusPrinter & PRINTER_STATUS_PAUSED ?
  1313. PRINTER_CONTROL_RESUME :
  1314. PRINTER_CONTROL_PAUSE;
  1315. break;
  1316. case IDM_PRINTER_PURGE:
  1317. {
  1318. TCHAR szScratch[kStrMax+kPrinterBufMax] = {0};
  1319. if( CommandConfirmationPurge(_hwnd, _pPrinter->pszPrinterName(szScratch)) )
  1320. {
  1321. pSelection->_CommandType = TSelection::kCommandTypePrinter;
  1322. pSelection->_dwCommandAction = PRINTER_CONTROL_PURGE;
  1323. }
  1324. }
  1325. break;
  1326. case IDM_PRINTER_WORKOFFLINE:
  1327. pSelection->_CommandType = TSelection::kCommandTypePrinterAttributes;
  1328. pSelection->_dwCommandAction = _dwAttributes ^ PRINTER_ATTRIBUTE_WORK_OFFLINE;
  1329. DBGMSG( DBG_WARN, ( "Queue.lrProcessCommand: Workoffline %d\n", uCommand ));
  1330. break;
  1331. default:
  1332. pSelection->_dwCommandAction = 0;
  1333. DBGMSG( DBG_WARN, ( "Queue.lrProcessCommand: unknown command %d\n", uCommand ));
  1334. break;
  1335. }
  1336. //
  1337. // Queue the async command.
  1338. //
  1339. _pPrinter->vCommandQueue( pSelection );
  1340. }
  1341. else
  1342. {
  1343. vShowResourceError( _hwnd );
  1344. }
  1345. }
  1346. else
  1347. {
  1348. //
  1349. // Item (job) command.
  1350. //
  1351. vProcessItemCommand( uCommand );
  1352. }
  1353. return 0;
  1354. }
  1355. VOID
  1356. TQueue::
  1357. vProcessItemCommand(
  1358. IN UINT uCommand
  1359. )
  1360. /*++
  1361. Routine Description:
  1362. Retrieves all selected items and attemps to execute a command
  1363. on them.
  1364. Arguments:
  1365. uCommand - IDM_* command.
  1366. Return Value:
  1367. --*/
  1368. {
  1369. //
  1370. // Declare job menu id to Job command mapping structure.
  1371. //
  1372. static struct {
  1373. UINT idmCommand;
  1374. DWORD dwCommand;
  1375. } aJobCommand[] = {
  1376. { IDM_JOB_CANCEL, JOB_CONTROL_DELETE },
  1377. { IDM_JOB_PAUSE, JOB_CONTROL_PAUSE },
  1378. { IDM_JOB_RESUME, JOB_CONTROL_RESUME },
  1379. { IDM_JOB_RESTART, JOB_CONTROL_RESTART }
  1380. };
  1381. //
  1382. // Get a list of selected Job IDs
  1383. //
  1384. TSelection* pSelection = new TSelection( this,
  1385. _pPrinter );
  1386. //
  1387. // Check for allocation error. We want to put a pop-up on all
  1388. // user actions if we can detect the error immediately after
  1389. // the user issues the command. Otherwise put in status bar.
  1390. //
  1391. if( !VALID_PTR(pSelection) ){
  1392. vShowResourceError( _hwnd );
  1393. goto NoCommand;
  1394. }
  1395. //
  1396. // There a few job related menu selections which are not
  1397. // deferable events i.e Job properties, This event will be done
  1398. // immediately and then release the selection object.
  1399. //
  1400. switch( uCommand ){
  1401. case IDM_JOB_PROPERTIES: {
  1402. TCHAR szPrinter[kPrinterBufMax];
  1403. LPTSTR pszPrinter;
  1404. pszPrinter = _pPrinter->pszPrinterName( szPrinter );
  1405. vDocumentPropSelections( NULL, pszPrinter, pSelection );
  1406. goto NoCommand;
  1407. }
  1408. default:
  1409. break;
  1410. }
  1411. //
  1412. // Ask for confirmation...
  1413. //
  1414. if( IDM_JOB_CANCEL == uCommand &&
  1415. IDYES != iMessage(_hwnd, IDS_PRINTERS_TITLE, IDS_QUEUE_SURE_CANCEL,
  1416. MB_YESNO|MB_ICONQUESTION|MB_DEFBUTTON2, kMsgNone, NULL) )
  1417. {
  1418. goto NoCommand;
  1419. }
  1420. //
  1421. // Map the job menu id to a job command.
  1422. //
  1423. UINT uIndex;
  1424. for( uIndex = 0; uIndex < COUNTOF( aJobCommand ); ++uIndex ){
  1425. //
  1426. // Check for a matching IDM_JOB -> JOB_CONTROL mapping.
  1427. //
  1428. if( aJobCommand[uIndex].idmCommand == uCommand ){
  1429. //
  1430. // Update the command action and job type
  1431. //
  1432. pSelection->_dwCommandAction = aJobCommand[uIndex].dwCommand;
  1433. pSelection->_CommandType = TSelection::kCommandTypeJob;
  1434. //
  1435. // Queue the job commands
  1436. //
  1437. _pPrinter->vCommandQueue( pSelection );
  1438. return;
  1439. }
  1440. }
  1441. //
  1442. // No matches; punt.
  1443. //
  1444. SPLASSERT( FALSE );
  1445. NoCommand:
  1446. delete pSelection;
  1447. return;
  1448. }
  1449. /********************************************************************
  1450. Utils.
  1451. ********************************************************************/
  1452. BOOL
  1453. TQueue::
  1454. bInsertItem(
  1455. HITEM hItem,
  1456. LIST_INDEX ListIndex
  1457. )
  1458. {
  1459. LV_ITEM item;
  1460. //
  1461. // Insert item into listview.
  1462. //
  1463. item.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM;
  1464. item.iSubItem = 0;
  1465. item.pszText = LPSTR_TEXTCALLBACK;
  1466. item.state = 0;
  1467. item.iImage = 0;
  1468. item.iItem = ListIndex;
  1469. item.lParam = (LPARAM)hItem;
  1470. if( ListView_InsertItem( _hwndLV, &item ) < 0 ){
  1471. DBGMSG( DBG_WARN,
  1472. ( "Queue.bInsertItem: Failed to add job %d\n",
  1473. GetLastError( )));
  1474. return FALSE;
  1475. }
  1476. return TRUE;
  1477. }
  1478. /********************************************************************
  1479. Job block processing.
  1480. ********************************************************************/
  1481. VOID
  1482. TQueue::
  1483. vBlockProcess(
  1484. VOID
  1485. )
  1486. /*++
  1487. Routine Description:
  1488. The request has been PostMessage'd and now is ready for
  1489. processing. If we need to save state, get the queue
  1490. selections then notify the pData that something has changed.
  1491. (The pData may call back and request that the list view
  1492. be cleared and reset.)
  1493. Called from UI thread only.
  1494. Arguments:
  1495. Return Value:
  1496. --*/
  1497. {
  1498. SINGLETHREAD(UIThread);
  1499. SPLASSERT( _pPrinter );
  1500. SPLASSERT( _pPrinter->pData( ));
  1501. //
  1502. // Keep track if the number of jobs changes. If it does,
  1503. // then we need to update the status bar.
  1504. //
  1505. COUNT cItems = _cItems;
  1506. //
  1507. // Process all pending blocks.
  1508. //
  1509. _pPrinter->pData()->vBlockProcess();
  1510. if( cItems != _cItems ){
  1511. TCHAR szScratch[kStrMax];
  1512. TCHAR szText[kStrMax];
  1513. szText[0] = 0;
  1514. //
  1515. // Always update the job count.
  1516. //
  1517. if( LoadString( ghInst,
  1518. IDS_SB_JOBS,
  1519. szScratch,
  1520. COUNTOF( szScratch ))){
  1521. wsprintf( szText, szScratch, _cItems );
  1522. }
  1523. SendMessage( _hwndSB,
  1524. SB_SETTEXT,
  1525. kStatusPaneJobs,
  1526. (LPARAM)szText );
  1527. //
  1528. // Check if it's pending deletion and we just printed the
  1529. // last job. The queue window should close.
  1530. //
  1531. bDeletingAndNoJobs();
  1532. }
  1533. }
  1534. /********************************************************************
  1535. Private status helper functions
  1536. ********************************************************************/
  1537. LPTSTR
  1538. TQueue::
  1539. pszStatusString(
  1540. OUT LPTSTR pszDest,
  1541. IN OUT UINT& cchMark,
  1542. IN DWORD dwStatus,
  1543. IN BOOL bInitialSep,
  1544. IN BOOL bFirstOnly,
  1545. IN const STATUS_MAP pStatusMaps[]
  1546. )
  1547. /*++
  1548. Routine Description:
  1549. Builds a status string into pszDest, based on the dwStatus bitfield
  1550. and Type.
  1551. Arguments:
  1552. pszDest - Buffer to receive status string.
  1553. cchMark - Char count of pszDest; on return, holds chars remaining.
  1554. dwStatus - DWORD status field matching Type.
  1555. bInitialSep - Indicates whether an initial separator is needed.
  1556. pStatusMaps - Pointer to array of status maps (bit -> IDS).
  1557. bFirstOnly - Adds only 1 status string.
  1558. Return Value:
  1559. Pointer to the end of the string (ready for next vStrCat.
  1560. cchMark is updated.
  1561. --*/
  1562. {
  1563. TCHAR szStatus[kStrMax];
  1564. LPTSTR pszMark = pszDest;
  1565. UINT i;
  1566. for( i = 0, pszMark[0] = 0;
  1567. pStatusMaps->dwMask;
  1568. ++i, ++pStatusMaps ){
  1569. if( pStatusMaps->dwMask & dwStatus ){
  1570. if( !LoadString( ghInst,
  1571. pStatusMaps->uIDS,
  1572. szStatus,
  1573. COUNTOF( szStatus ))){
  1574. DBGMSG( DBG_ERROR,
  1575. ( "Queue.pszStatusString: unable to load %d, error %d\n",
  1576. pStatusMaps->uIDS,
  1577. GetLastError( )));
  1578. continue;
  1579. }
  1580. //
  1581. // If not at the beginning, we need a separator.
  1582. //
  1583. if( pszMark != pszDest || bInitialSep ){
  1584. //
  1585. // Spit out a separator.
  1586. //
  1587. pszMark = pszStrCat( pszMark,
  1588. gszStatusSeparator,
  1589. cchMark );
  1590. if( !pszMark )
  1591. break;
  1592. }
  1593. //
  1594. // Append the status string.
  1595. //
  1596. pszMark = pszStrCat( pszMark,
  1597. szStatus,
  1598. cchMark );
  1599. if( !pszMark || bFirstOnly ){
  1600. break;
  1601. }
  1602. }
  1603. }
  1604. return pszMark;
  1605. }
  1606. /********************************************************************
  1607. MPrinterClient virtual definitions.
  1608. ********************************************************************/
  1609. VOID
  1610. TQueue::
  1611. vItemChanged(
  1612. IN ITEM_CHANGE ItemChange,
  1613. IN HITEM hItem,
  1614. IN INFO Info,
  1615. IN INFO InfoNew
  1616. )
  1617. /*++
  1618. Routine Description:
  1619. A particular item changed, refresh just part of the window.
  1620. Note: Currently there is no sorting, so the NATURAL_INDEX is
  1621. the same as the LIST_INDEX.
  1622. When a TData* calls this routine, it must also upate its data
  1623. structures before calling this.
  1624. Arguments:
  1625. ItemChange - Indicates what about the job changed.
  1626. hItem - Handle to job that changed.
  1627. Info - Depends on the type of change; generally the old version
  1628. of the info.
  1629. InfoNew - Depends on the type of change; generally the new version
  1630. of the info.
  1631. Return Value:
  1632. --*/
  1633. {
  1634. SINGLETHREAD(UIThread);
  1635. //
  1636. // Fix up one job.
  1637. //
  1638. switch( ItemChange ){
  1639. case kItemCreate:
  1640. //
  1641. // Always insert at the end of the list.
  1642. //
  1643. Info.NaturalIndex = _cItems;
  1644. //
  1645. // How to handle this error?
  1646. //
  1647. // !! SORT !!
  1648. // iItem == NaturalIndex only if no sorting is enabled.
  1649. //
  1650. bInsertItem( hItem, Info.NaturalIndex );
  1651. ++_cItems;
  1652. break;
  1653. case kItemDelete:
  1654. //
  1655. // Delete the item from the listview.
  1656. //
  1657. //
  1658. // !! SORT !!
  1659. // iItem == NaturalIndex only if no sorting is enabled.
  1660. //
  1661. if( !bDeleteItem( Info.NaturalIndex )){
  1662. DBGMSG( DBG_WARN,
  1663. ( "Queue.vItemChanged: Failed to del job %d\n",
  1664. GetLastError( )));
  1665. }
  1666. --_cItems;
  1667. break;
  1668. case kItemName:
  1669. //
  1670. // We must set the item text, or else the width of the
  1671. // label doesn't change.
  1672. //
  1673. //
  1674. // !! SORT !!
  1675. //
  1676. ListView_SetItemText(
  1677. _hwndLV,
  1678. Info.NaturalIndex,
  1679. 0,
  1680. (LPTSTR)_pPrinter->pData()->GetInfo( hItem, 0 ).pszData );
  1681. //
  1682. // Fall through.
  1683. //
  1684. case kItemInfo:
  1685. case kItemAttributes:
  1686. //
  1687. // If it's visible, invalidate the line.
  1688. //
  1689. //
  1690. // !! SORT !!
  1691. // iItem == NaturalIndex only if no sorting is enabled.
  1692. //
  1693. RECT rc;
  1694. if( ListView_GetItemRect( _hwndLV,
  1695. Info.NaturalIndex,
  1696. &rc,
  1697. LVIR_BOUNDS )){
  1698. InvalidateRect( _hwndLV, &rc, TRUE );
  1699. }
  1700. break;
  1701. case kItemPosition:
  1702. vItemPositionChanged( hItem,
  1703. Info.NaturalIndex,
  1704. InfoNew.NaturalIndex );
  1705. break;
  1706. default:
  1707. DBGMSG( DBG_ERROR,
  1708. ( "Queue.vItemChanged: Unknown change %x\n", ItemChange ));
  1709. break;
  1710. }
  1711. }
  1712. VOID
  1713. TQueue::
  1714. vContainerChanged(
  1715. IN CONTAINER_CHANGE ContainerChange,
  1716. IN INFO Info
  1717. )
  1718. {
  1719. DBGMSG( DBG_QUEUEINFO,
  1720. ( "Queue.vContainerChanged: %x %x\n", ContainerChange, Info.dwData ));
  1721. //
  1722. // Some of the commands are synchronous. Handle them first.
  1723. //
  1724. switch( ContainerChange ){
  1725. case kContainerReloadItems:
  1726. vReloadItems( Info.dwData );
  1727. break;
  1728. case kContainerClearItems:
  1729. vClearItems();
  1730. break;
  1731. case kContainerStateVar:
  1732. _pPrintLib->bJobAdd( _pPrinter, Info.dwData );
  1733. break;
  1734. default:
  1735. //
  1736. // All asynchronous commands use PostMessage to get to UI thread.
  1737. //
  1738. PostMessage( _hwnd, WM_PRINTLIB_STATUS, ContainerChange, Info.dwData );
  1739. break;
  1740. }
  1741. }
  1742. VOID
  1743. TQueue::
  1744. vSaveSelections(
  1745. VOID
  1746. )
  1747. {
  1748. SINGLETHREAD(UIThread);
  1749. //
  1750. // State needs to be saved.
  1751. // NOT RE-ENTRANT.
  1752. //
  1753. SPLASSERT( !SaveSelection._pSelection );
  1754. //
  1755. // Determine which item is selected and store the Id.
  1756. //
  1757. SaveSelection._idFocused = kInvalidIdentValue;
  1758. INT iItem = ListView_GetNextItem( _hwndLV, -1, LVNI_FOCUSED );
  1759. if( iItem != -1 ){
  1760. //
  1761. // !! SORTORDER !!
  1762. //
  1763. HANDLE hItem = _pPrinter->pData()->GetItem( iItem );
  1764. SaveSelection._idFocused = _pPrinter->pData()->GetId( hItem );
  1765. }
  1766. //
  1767. // Save selected Items.
  1768. //
  1769. // Can't handle a failure here--we don't want to pop up a
  1770. // message box (since this could be poll-refresh) and we
  1771. // don't want to disturb the current error in the status bar.
  1772. // (The status bar error is only for user-initiated commands:
  1773. // we assume the user will look here before executing another
  1774. // commnad.)
  1775. //
  1776. SaveSelection._pSelection = new TSelection( this,
  1777. _pPrinter );
  1778. }
  1779. VOID
  1780. TQueue::
  1781. vRestoreSelections(
  1782. VOID
  1783. )
  1784. {
  1785. SINGLETHREAD(UIThread);
  1786. if( SaveSelection._idFocused != kInvalidIdentValue ){
  1787. NATURAL_INDEX NaturalIndex;
  1788. LV_ITEM item;
  1789. //
  1790. // Translate ID to DataIndex.
  1791. // !! SORT ORDER !!
  1792. //
  1793. item.stateMask =
  1794. item.state = LVIS_FOCUSED;
  1795. NaturalIndex = _pPrinter->pData()->GetNaturalIndex( SaveSelection._idFocused,
  1796. NULL );
  1797. //
  1798. // The DataIndex value may be gone of the selectd Item was
  1799. // deleted or printed.
  1800. //
  1801. if( NaturalIndex != kInvalidNaturalIndexValue ){
  1802. SendMessage( _hwndLV,
  1803. LVM_SETITEMSTATE,
  1804. NaturalIndex,
  1805. (LPARAM)&item );
  1806. }
  1807. }
  1808. //
  1809. // Don't check using VALID_PTR since the no-selection case will
  1810. // cause it to fail, but we don't want to flag an error.
  1811. //
  1812. if( SaveSelection._pSelection && SaveSelection._pSelection->bValid( )){
  1813. NATURAL_INDEX NaturalIndex;
  1814. COUNT i;
  1815. PIDENT pid;
  1816. LV_ITEM item;
  1817. item.stateMask =
  1818. item.state = LVIS_SELECTED;
  1819. for( i = 0, pid = SaveSelection._pSelection->_pid;
  1820. i < SaveSelection._pSelection->_cSelected;
  1821. ++i, ++pid ){
  1822. //
  1823. // Translate IDENT to DataIndex.
  1824. // !! SORT ORDER !!
  1825. //
  1826. NaturalIndex = _pPrinter->pData()->GetNaturalIndex( *pid,
  1827. NULL );
  1828. //
  1829. // The DataIndex value may be gone of the selected Item was
  1830. // deleted or printed.
  1831. //
  1832. if( NaturalIndex != kInvalidNaturalIndexValue ){
  1833. SendMessage( _hwndLV,
  1834. LVM_SETITEMSTATE,
  1835. NaturalIndex,
  1836. (LPARAM)&item );
  1837. }
  1838. }
  1839. }
  1840. //
  1841. // Cleanup even if we fail.
  1842. //
  1843. delete SaveSelection._pSelection;
  1844. SaveSelection._pSelection = NULL;
  1845. }
  1846. VDataNotify*
  1847. TQueue::
  1848. pNewNotify(
  1849. MDataClient* pDataClient
  1850. ) const
  1851. {
  1852. return new TDataNJob( pDataClient );
  1853. }
  1854. VDataRefresh*
  1855. TQueue::
  1856. pNewRefresh(
  1857. MDataClient* pDataClient
  1858. ) const
  1859. {
  1860. return new TDataRJob( pDataClient );
  1861. }
  1862. /********************************************************************
  1863. Retrieve selected items. Used when processing commands against
  1864. items or saving and restoring the selection during a refresh.
  1865. ********************************************************************/
  1866. COUNT
  1867. TQueue::
  1868. cSelected(
  1869. VOID
  1870. ) const
  1871. {
  1872. SINGLETHREAD(UIThread);
  1873. return ListView_GetSelectedCount( _hwndLV );
  1874. }
  1875. HANDLE
  1876. TQueue::
  1877. GetFirstSelItem(
  1878. VOID
  1879. ) const
  1880. {
  1881. SINGLETHREAD(UIThread);
  1882. INT iItem = ListView_GetNextItem( _hwndLV,
  1883. -1,
  1884. LVNI_SELECTED );
  1885. return INT2PTR(iItem, HANDLE);
  1886. }
  1887. HANDLE
  1888. TQueue::
  1889. GetNextSelItem(
  1890. HANDLE hItem
  1891. ) const
  1892. {
  1893. SINGLETHREAD(UIThread);
  1894. INT iJob = (INT)(ULONG_PTR)hItem;
  1895. SPLASSERT( iJob < 0x8000 );
  1896. iJob = ListView_GetNextItem( _hwndLV,
  1897. iJob,
  1898. LVNI_SELECTED );
  1899. if( iJob == -1 ){
  1900. DBGMSG( DBG_ERROR,
  1901. ( "Queue.hItemNext: LV_GNI failed %d\n",
  1902. GetLastError( )));
  1903. }
  1904. return INT2PTR(iJob, HANDLE);
  1905. }
  1906. IDENT
  1907. TQueue::
  1908. GetId(
  1909. HANDLE hItem
  1910. ) const
  1911. {
  1912. SINGLETHREAD(UIThread);
  1913. INT iJob = (INT)(ULONG_PTR)hItem;
  1914. if( iJob != -1 ){
  1915. hItem = _pPrinter->pData()->GetItem( iJob );
  1916. return _pPrinter->pData()->GetId( hItem );
  1917. }
  1918. return kInvalidIdentValue;
  1919. }
  1920. BOOL
  1921. TQueue::
  1922. bGetPrintLib(
  1923. TRefLock<TPrintLib> &refLock
  1924. ) const
  1925. {
  1926. ASSERT(_pPrintLib.pGet());
  1927. if (_pPrintLib.pGet())
  1928. {
  1929. refLock.vAcquire(_pPrintLib.pGet());
  1930. return TRUE;
  1931. }
  1932. return FALSE;
  1933. }
  1934. VOID
  1935. TQueue::
  1936. vRefZeroed(
  1937. VOID
  1938. )
  1939. {
  1940. if( bValid( )){
  1941. delete this;
  1942. }
  1943. }
  1944. /********************************************************************
  1945. Implementation functions.
  1946. ********************************************************************/
  1947. VOID
  1948. TQueue::
  1949. vContainerChangedHandler(
  1950. IN CONTAINER_CHANGE ContainerChange,
  1951. IN INFO Info
  1952. )
  1953. {
  1954. static DWORD gadwConnectStatusMap[] = CONNECT_STATUS_MAP;
  1955. SINGLETHREAD(UIThread);
  1956. switch( ContainerChange ){
  1957. case kContainerNewBlock:
  1958. vBlockProcess();
  1959. break;
  1960. case kContainerAttributes:
  1961. _dwAttributes = Info.dwData;
  1962. //
  1963. // Update the queue view title, (Note status information is displayed here)
  1964. //
  1965. vUpdateTitle();
  1966. //
  1967. // !! LATER !!
  1968. //
  1969. // Change printer icon.
  1970. //
  1971. break;
  1972. case kContainerName:
  1973. //
  1974. // Update the queue view title, (Note status information is displayed here)
  1975. //
  1976. vUpdateTitle();
  1977. break;
  1978. case kContainerStatus:
  1979. _dwStatusPrinter = Info.dwData;
  1980. //
  1981. // If the printer is pending deletion and has no jobs,
  1982. // then immediately punt.
  1983. //
  1984. if( bDeletingAndNoJobs( )){
  1985. return;
  1986. }
  1987. //
  1988. // Update the queue view title, (Note status information is displayed here)
  1989. //
  1990. vUpdateTitle();
  1991. break;
  1992. case kContainerConnectStatus:
  1993. SPLASSERT( Info.dwData < COUNTOF( gadwConnectStatusMap ));
  1994. _idsConnectStatus = gadwConnectStatusMap[Info.dwData];
  1995. //
  1996. // If the printer isn't found, then put up a message box and
  1997. // dismiss the queue view.
  1998. //
  1999. if( Info.dwData == kConnectStatusInvalidPrinterName ){
  2000. TCHAR szPrinter[kPrinterBufMax];
  2001. LPTSTR pszPrinter = _pPrinter->pszPrinterName( szPrinter );
  2002. iMessage( _hwnd,
  2003. IDS_ERR_PRINTER_NOT_FOUND_TITLE,
  2004. IDS_ERR_PRINTER_NOT_FOUND,
  2005. MB_OK|MB_ICONSTOP,
  2006. ERROR_INVALID_PRINTER_NAME,
  2007. NULL,
  2008. pszPrinter );
  2009. SendMessage( _hwnd,
  2010. WM_CLOSE,
  2011. 0,
  2012. 0 );
  2013. break;
  2014. }
  2015. //
  2016. // Update the queue view title, (Note status information is displayed here)
  2017. //
  2018. vUpdateTitle();
  2019. break;
  2020. case kContainerErrorStatus: {
  2021. _dwErrorStatus = Info.dwData;
  2022. //
  2023. // Scan for known errors and translate into friendly strings.
  2024. //
  2025. static const ERROR_MAP gaErrorMap[] = {
  2026. ERROR_ACCESS_DENIED, IDS_ERR_ACCESS_DENIED,
  2027. };
  2028. COUNT i;
  2029. DWORD idsError = IDS_SB_ERROR;
  2030. for( i=0; i< COUNTOF( gaErrorMap ); ++i ){
  2031. if( _dwErrorStatus == gaErrorMap[i].dwError ){
  2032. idsError = gaErrorMap[i].uIDS;
  2033. break;
  2034. }
  2035. }
  2036. TString strText;
  2037. if( _dwErrorStatus ){
  2038. TStatusB bStatus;
  2039. bStatus DBGCHK = strText.bLoadString( ghInst, idsError );
  2040. }
  2041. SendMessage( _hwndSB, SB_SETTEXT, kStatusPaneError, (LPARAM)(LPCTSTR)strText );
  2042. break;
  2043. }
  2044. case kContainerRefreshComplete:
  2045. break;
  2046. default:
  2047. DBGMSG( DBG_ERROR, ( "Queue.vContainerChanged: unknown ContainerChange %x\n", ContainerChange ));
  2048. break;
  2049. }
  2050. }
  2051. VOID
  2052. TQueue::
  2053. vUpdateTitle(
  2054. VOID
  2055. )
  2056. {
  2057. TCHAR szScratch[kStrMax+kPrinterBufMax];
  2058. TCHAR szText[kStrMax+kPrinterBufMax];
  2059. UINT nSize = COUNTOF( szText );
  2060. //
  2061. // Build the printer status string.
  2062. //
  2063. ConstructPrinterFriendlyName( pszPrinterName( szScratch ), szText, &nSize );
  2064. //
  2065. // Calculate the string text.
  2066. //
  2067. UINT cch = lstrlen( szText );
  2068. LPTSTR pszText = szText + cch;
  2069. cch = COUNTOF( szText ) - cch;
  2070. //
  2071. // Update the title text with the current printer status.
  2072. //
  2073. pszText = pszStatusString( pszText, cch, _dwStatusPrinter, TRUE, FALSE, gaStatusMapPrinter );
  2074. //
  2075. // Special case the work off line status, because work offline is not
  2076. // implemented in the spooler as a printer status rather as a printer attribute.
  2077. //
  2078. if( _dwAttributes & PRINTER_ATTRIBUTE_WORK_OFFLINE )
  2079. {
  2080. if( LoadString( ghInst, IDS_WORK_OFFLINE, szScratch, COUNTOF( szScratch )))
  2081. {
  2082. pszText = pszStrCat( pszText, gszSpace, cch );
  2083. pszText = pszStrCat( pszText, gszStatusSeparator, cch );
  2084. pszText = pszStrCat( pszText, gszSpace, cch );
  2085. pszText = pszStrCat( pszText, szScratch, cch );
  2086. }
  2087. }
  2088. //
  2089. // If the connection status changed then update the text.
  2090. //
  2091. if( _idsConnectStatus ){
  2092. if( LoadString( ghInst, _idsConnectStatus, szScratch, COUNTOF( szScratch )))
  2093. {
  2094. pszText = pszStrCat( pszText, gszSpace, cch );
  2095. pszText = pszStrCat( pszText, szScratch, cch );
  2096. }
  2097. }
  2098. //
  2099. // Set the queu view title bar text.
  2100. //
  2101. SetWindowText( _hwnd, szText );
  2102. }
  2103. BOOL
  2104. TQueue::
  2105. bDeletingAndNoJobs(
  2106. VOID
  2107. )
  2108. /*++
  2109. Routine Description:
  2110. Returns TRUE if the queue is pending deletion and has no jobs.
  2111. In this case it will also close the window.
  2112. Arguments:
  2113. Return Value:
  2114. TRUE - Pending deletion with no Jobs; window also closed.
  2115. FALSE - Not (pending deletion and no jobs).
  2116. --*/
  2117. {
  2118. if( _cItems == 0 && _dwStatusPrinter & PRINTER_STATUS_PENDING_DELETION ){
  2119. PostMessage( _hwnd, WM_CLOSE, 0, 0 );
  2120. return TRUE;
  2121. }
  2122. return FALSE;
  2123. }
  2124. VOID
  2125. TQueue::
  2126. vItemPositionChanged(
  2127. IN HITEM hItem,
  2128. IN NATURAL_INDEX NaturalIndex,
  2129. IN NATURAL_INDEX NaturalIndexNew
  2130. )
  2131. {
  2132. SPLASSERT( NaturalIndexNew < _cItems );
  2133. SPLASSERT( NaturalIndex != NaturalIndexNew );
  2134. DBGMSG( DBG_QUEUEINFO,
  2135. ( "Queue.vItemPositionChanged: Change requested %d %d %x\n",
  2136. NaturalIndex, NaturalIndexNew, hItem ));
  2137. //
  2138. // Get the item state.
  2139. //
  2140. UINT uState = ListView_GetItemState( _hwndLV,
  2141. NaturalIndex,
  2142. LVIS_FOCUSED | LVIS_SELECTED );
  2143. //
  2144. // Move it to the right place.
  2145. //
  2146. if( !bDeleteItem( NaturalIndex )){
  2147. DBGMSG( DBG_WARN,
  2148. ( "Queue.vItemPositionChanged: Moving, delete failed on %d %d\n",
  2149. NaturalIndex, GetLastError( )));
  2150. }
  2151. if( bInsertItem( hItem, NaturalIndexNew )){
  2152. //
  2153. // Set item state.
  2154. //
  2155. ListView_SetItemState( _hwndLV,
  2156. NaturalIndexNew,
  2157. LVIS_FOCUSED | LVIS_SELECTED,
  2158. uState );
  2159. } else {
  2160. DBGMSG( DBG_ERROR,
  2161. ( "Queue.vItemPositionChanged: Moving, insert failed on %d %d\n",
  2162. NaturalIndex, GetLastError( )));
  2163. }
  2164. }
  2165. VOID
  2166. TQueue::
  2167. vClearItems(
  2168. VOID
  2169. )
  2170. /*++
  2171. Routine Description:
  2172. Removes all items from the list view.
  2173. May be called from either UI or worker thread. If called from
  2174. worker thread, must guarantee that there are no synchronization
  2175. problems.
  2176. Arguments:
  2177. Return Value:
  2178. --*/
  2179. {
  2180. DBGMSG( DBG_QUEUEINFO, ( "Queue.vClearItems: Clearing %d %d\n", _hwndLV, this ));
  2181. TStatusB bStatus;
  2182. //
  2183. // Clear out all items from list view.
  2184. //
  2185. bStatus DBGCHK = ListView_DeleteAllItems( _hwndLV );
  2186. }
  2187. VOID
  2188. TQueue::
  2189. vReloadItems(
  2190. COUNT cItems
  2191. )
  2192. /*++
  2193. Routine Description:
  2194. Delete all items in the list view and refresh based on the
  2195. new pData information.
  2196. Arguments:
  2197. Return Value:
  2198. --*/
  2199. {
  2200. SINGLETHREAD(UIThread);
  2201. DBGMSG( DBG_QUEUEINFO,
  2202. ( "Queue.vReloadItems: Reload %d %d %d\n",
  2203. _hwndLV, this, cItems ));
  2204. vClearItems();
  2205. _cItems = cItems;
  2206. //
  2207. // If we have Items, insert them.
  2208. //
  2209. if( _pPrinter->pData() && cItems ){
  2210. LV_ITEM item;
  2211. //
  2212. // Notify the list view of how many jobs we will ultimately insert
  2213. // to avoid reallocs.
  2214. //
  2215. ListView_SetItemCount( _hwndLV, cItems );
  2216. //
  2217. // Add to the listview.
  2218. //
  2219. item.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM;
  2220. item.iSubItem = 0;
  2221. item.pszText = LPSTR_TEXTCALLBACK;
  2222. item.state = 0;
  2223. item.iImage = 0;
  2224. HANDLE hItem;
  2225. COUNT cItemIndex;
  2226. for( cItemIndex = 0, hItem = _pPrinter->pData()->GetItem( 0 );
  2227. cItemIndex < cItems;
  2228. ++cItemIndex, hItem = _pPrinter->pData()->GetNextItem( hItem )){
  2229. item.iItem = cItemIndex;
  2230. item.lParam = (LPARAM)hItem;
  2231. if( ListView_InsertItem( _hwndLV, &item ) < 0 ){
  2232. DBGMSG( DBG_ERROR,
  2233. ( "Queue.vReloadItems: Failed to insert item %d %d\n",
  2234. item, GetLastError( )));
  2235. break;
  2236. }
  2237. }
  2238. }
  2239. }
  2240. BOOL
  2241. TQueue::
  2242. bOnCopyData(
  2243. IN WPARAM wParam,
  2244. IN LPARAM lParam
  2245. )
  2246. /*++
  2247. Routine Description:
  2248. Get the data pased to it and checks if it matches the
  2249. current printer name.
  2250. Arguments:
  2251. Return Value:
  2252. TRUE printer name matches, FALSE match not found.
  2253. --*/
  2254. {
  2255. TCHAR szFullPrinterName[kPrinterBufMax];
  2256. TCHAR szPrinterName[kPrinterBufMax];
  2257. BOOL bStatus = FALSE;
  2258. PCOPYDATASTRUCT pcpds = (PCOPYDATASTRUCT)lParam;
  2259. LPCTSTR pszPrinterName = (LPTSTR)pcpds->lpData;
  2260. //
  2261. // Check for valid queue signature.
  2262. //
  2263. if( pcpds->dwData == static_cast<DWORD>( TQueue::kQueueSignature ) ){
  2264. //
  2265. // Get the queue view printer name, this may not be fully qualified
  2266. // if it's a local printer.
  2267. //
  2268. if( _pPrinter->pszPrinterName( szPrinterName ) ){
  2269. DBGMSG( DBG_TRACE, ("CopyMessage Printer Name " TSTR " Passed Data " TSTR "\n", szPrinterName, pszPrinterName ) );
  2270. //
  2271. // If the printer name passed is fully qualified and the current
  2272. // queue view printer name is not fully qualified then prepend the
  2273. // computer name.
  2274. //
  2275. if( pszPrinterName[0] == TEXT('\\') &&
  2276. pszPrinterName[1] == TEXT('\\') &&
  2277. szPrinterName[0] != TEXT('\\') &&
  2278. szPrinterName[1] != TEXT('\\') ){
  2279. _tcscpy( szFullPrinterName, _pPrintLib->strComputerName() );
  2280. _tcscat( szFullPrinterName, TEXT("\\") );
  2281. _tcscat( szFullPrinterName, szPrinterName );
  2282. //
  2283. // If the passed printer name is not fully qualified and the
  2284. // queue view is fully qualified then strip the computer name.
  2285. //
  2286. } else if( pszPrinterName[0] != TEXT('\\') &&
  2287. pszPrinterName[1] != TEXT('\\') &&
  2288. szPrinterName[0] == TEXT('\\') &&
  2289. szPrinterName[1] == TEXT('\\') ){
  2290. LPCTSTR psz = _tcschr( szPrinterName+2, TEXT('\\') );
  2291. _tcscpy( szFullPrinterName, psz ? psz : TEXT("") );
  2292. //
  2293. // The passed printer name and the queue view printer names are either
  2294. // fully qualified or not, but they match.
  2295. //
  2296. } else {
  2297. _tcscpy( szFullPrinterName, szPrinterName );
  2298. }
  2299. //
  2300. // Check if the printer names match.
  2301. //
  2302. if( !_tcsicmp( szFullPrinterName, pszPrinterName ) ){
  2303. bStatus = TRUE;
  2304. }
  2305. }
  2306. }
  2307. return bStatus;
  2308. }
  2309. BOOL
  2310. TQueue::
  2311. bIsDuplicateWindow(
  2312. IN HWND hwndOwner,
  2313. IN LPCTSTR pszPrinterName,
  2314. IN HWND *phwnd
  2315. )
  2316. /*++
  2317. Routine Description:
  2318. Asks the queue view window if you are the one with the specified
  2319. printer name. This will ensure there is only one queue view per
  2320. printer on the machine.
  2321. Arguments:
  2322. hwndOwner - Handle of parent window
  2323. pszPrinterName - Printer name to check
  2324. *phwnd - Pointer where to return the duplicate previous window handle.
  2325. Return Value:
  2326. TRUE duplicate found and phwnd point to its window handle.
  2327. FALSE duplicate window not found.
  2328. --*/
  2329. {
  2330. BOOL bStatus = FALSE;
  2331. HWND hwnd = NULL;
  2332. DBGMSG( DBG_TRACE, ("Searching for Printer Name " TSTR "\n", pszPrinterName ) );
  2333. for (hwnd = FindWindow(gszClassName, NULL); hwnd; hwnd = GetWindow(hwnd, GW_HWNDNEXT))
  2334. {
  2335. TCHAR szClass[kStrMax];
  2336. COPYDATASTRUCT cpds;
  2337. //
  2338. // Set up the copy data structure.
  2339. //
  2340. cpds.dwData = static_cast<DWORD>( TQueue::kQueueSignature );
  2341. cpds.cbData = _tcslen( pszPrinterName ) * sizeof(TCHAR) + sizeof(TCHAR);
  2342. cpds.lpData = reinterpret_cast<PVOID>( const_cast<LPTSTR>( pszPrinterName ) );
  2343. //
  2344. // Just another check to ensure the class name matches.
  2345. //
  2346. GetClassName(hwnd, szClass, COUNTOF(szClass));
  2347. if(!_tcsicmp( szClass, gszClassName ))
  2348. {
  2349. //
  2350. // Ask this queue window if you are viewing this printer.
  2351. //
  2352. if( SendMessage(hwnd, WM_COPYDATA, (WPARAM)hwndOwner, (LPARAM)&cpds) )
  2353. {
  2354. *phwnd = hwnd;
  2355. bStatus = TRUE;
  2356. break;
  2357. }
  2358. }
  2359. }
  2360. return bStatus;
  2361. }
  2362. VOID
  2363. TQueue::
  2364. vRemove(
  2365. IN LPCTSTR pszPrinterName
  2366. )
  2367. /*++
  2368. Routine Description:
  2369. Remove the printer queue view registry positon information.
  2370. Arguments:
  2371. Pointer to printer fully qualified printer name.
  2372. Return Value:
  2373. Nothing.
  2374. --*/
  2375. {
  2376. TPersist Persist( gszPrinterPositions, TPersist::kOpen|TPersist::kRead|TPersist::kWrite );
  2377. if( VALID_OBJ( Persist ) )
  2378. {
  2379. TStatusB bStatus;
  2380. bStatus DBGCHK = Persist.bRemove( pszPrinterName );
  2381. }
  2382. }
  2383. VOID
  2384. TQueue::
  2385. vCheckDefaultPrinterChanged(
  2386. VOID
  2387. )
  2388. /*++
  2389. Routine Description:
  2390. Update the print queue icon if needed.
  2391. Arguments:
  2392. None.
  2393. Return Value:
  2394. --*/
  2395. {
  2396. BOOL bDefaultPrinter;
  2397. TCHAR szPrinter[kPrinterBufMax];
  2398. _pPrinter->pszPrinterName(szPrinter);
  2399. bDefaultPrinter = CheckDefaultPrinter(szPrinter) == kDefault;
  2400. if( _bDefaultPrinter ^ bDefaultPrinter )
  2401. {
  2402. //
  2403. // If the default printer setting is changed and it related to
  2404. // current printer, change the window icon
  2405. //
  2406. _bDefaultPrinter = bDefaultPrinter;
  2407. CAutoHandleIcon shIconLarge, shIconSmall;
  2408. LoadPrinterIcons(szPrinter, &shIconLarge, &shIconSmall);
  2409. if( shIconLarge && shIconSmall )
  2410. {
  2411. _shIconLarge = shIconLarge.Detach();
  2412. _shIconSmall = shIconSmall.Detach();
  2413. SendMessage( _hwnd, WM_SETICON, ICON_BIG, (LPARAM)(HICON)_shIconLarge );
  2414. SendMessage( _hwnd, WM_SETICON, ICON_SMALL, (LPARAM)(HICON)_shIconSmall );
  2415. }
  2416. }
  2417. }