Leaked source code of windows server 2003
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.

2028 lines
58 KiB

  1. /*++
  2. Copyright (c) 1995-2000 Microsoft Corporation
  3. Module Name:
  4. fileq5.c
  5. Abstract:
  6. Default queue callback function.
  7. Author:
  8. Ted Miller (tedm) 24-Feb-1995
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. #define QUEUECONTEXT_SIGNATURE (DWORD)(0x43515053) // 'CQPS'
  14. typedef struct _QUEUECONTEXT {
  15. DWORD Signature; // an attempt to catch re-use of deleted queuecontext
  16. HWND OwnerWindow;
  17. DWORD MainThreadId;
  18. HWND ProgressDialog;
  19. HWND ProgressBar;
  20. BOOL Cancelled;
  21. PTSTR CurrentSourceName;
  22. BOOL ScreenReader;
  23. BOOL MessageBoxUp;
  24. WPARAM PendingUiType;
  25. PVOID PendingUiParameters;
  26. UINT CancelReturnCode;
  27. BOOL DialogKilled;
  28. //
  29. // If the SetupInitDefaultQueueCallbackEx is used, the caller can
  30. // specify an alternate handler for progress. This is useful to
  31. // get the default behavior for disk prompting, error handling, etc,
  32. // but to provide a gas gauge embedded, say, in a wizard page.
  33. //
  34. // The alternate window is sent ProgressMsg once when the copy queue
  35. // is started (wParam = 0. lParam = number of files to copy).
  36. // It is then also sent once per file copied (wParam = 1. lParam = 0).
  37. //
  38. // NOTE: a silent installation (i.e., no progress UI) can be accomplished
  39. // by specifying an AlternateProgressWindow handle of INVALID_HANDLE_VALUE.
  40. //
  41. HWND AlternateProgressWindow;
  42. UINT ProgressMsg;
  43. UINT NoToAllMask;
  44. HANDLE UiThreadHandle;
  45. //
  46. // instead of posting responses to main thread, use an event with flags
  47. //
  48. HANDLE hEvent;
  49. BOOL bDialogExited;
  50. LPARAM lParam;
  51. #ifdef NOCANCEL_SUPPORT
  52. BOOL AllowCancel;
  53. #endif
  54. } QUEUECONTEXT, *PQUEUECONTEXT;
  55. typedef struct _VERDLGCONTEXT {
  56. PQUEUECONTEXT QueueContext;
  57. UINT Notification;
  58. UINT_PTR Param1;
  59. UINT_PTR Param2;
  60. } VERDLGCONTEXT, *PVERDLGCONTEXT;
  61. #define WMX_PROGRESSTHREAD (WM_APP+0)
  62. #define WMX_KILLDIALOG (WM_APP+1)
  63. #define WMX_HELLO (WM_APP+2)
  64. #define WMX_PERFORMUI (WM_APP+3)
  65. #define UI_NONE 0
  66. #define UI_COPYERROR 1
  67. #define UI_DELETEERROR 2
  68. #define UI_RENAMEERROR 3
  69. #define UI_NEEDMEDIA 4
  70. #define UI_MISMATCHERROR 5
  71. #define UI_BACKUPERROR 6
  72. typedef struct _COPYERRORUI {
  73. TCHAR Buffer[MAX_PATH];
  74. PTCHAR Filename;
  75. PFILEPATHS FilePaths;
  76. DWORD Flags;
  77. PTSTR PathOut;
  78. } COPYERRORUI, *PCOPYERRORUI;
  79. typedef struct _NEEDMEDIAUI {
  80. PSOURCE_MEDIA SourceMedia;
  81. DWORD Flags;
  82. PTSTR PathOut;
  83. } NEEDMEDIAUI, *PNEEDMEDIAUI;
  84. PCTSTR DialogPropName = TEXT("_context");
  85. INT_PTR
  86. pSetupProgressDlgProc(
  87. IN HWND hdlg,
  88. IN UINT msg,
  89. IN WPARAM wParam,
  90. IN LPARAM lParam
  91. );
  92. LPARAM
  93. pPerformUi (
  94. IN PQUEUECONTEXT Context,
  95. IN UINT UiType,
  96. IN PVOID UiParameters
  97. );
  98. VOID
  99. __cdecl
  100. pSetupProgressThread(
  101. IN PVOID Context
  102. )
  103. /*++
  104. Routine Description:
  105. Thread entry point for setup file progress indicator.
  106. Puts up a dialog box.
  107. Arguments:
  108. Context - supplies queue context.
  109. Return Value:
  110. 0 if unsuccessful, non-0 if successful.
  111. --*/
  112. {
  113. PQUEUECONTEXT context;
  114. INT_PTR i;
  115. MSG msg;
  116. //
  117. // Force this thread to have a message queue, just in case.
  118. //
  119. PeekMessage(&msg,NULL,0,0,PM_NOREMOVE);
  120. //
  121. // The thread parameter is the queue context.
  122. //
  123. context = Context;
  124. //
  125. // Create the progress dialog box.
  126. //
  127. i = DialogBoxParam(
  128. MyDllModuleHandle,
  129. MAKEINTRESOURCE(IDD_FILEPROGRESS),
  130. context->OwnerWindow,
  131. pSetupProgressDlgProc,
  132. (LPARAM)context
  133. );
  134. //
  135. // flag that this is the very last time hEvent will be set
  136. //
  137. context->bDialogExited = TRUE;
  138. SetEvent(context->hEvent);
  139. //
  140. // Done.
  141. //
  142. _endthread();
  143. }
  144. BOOL
  145. pWaitForUiResponse(
  146. IN OUT PQUEUECONTEXT Context
  147. )
  148. /*++
  149. Routine Description:
  150. Waits for UI event to be set
  151. Arguments:
  152. Context - supplies queue-context structure
  153. Return Value:
  154. FALSE = failure
  155. --*/
  156. {
  157. BOOL KeepWaiting = TRUE;
  158. DWORD WaitProcStatus;
  159. if (Context->hEvent == NULL) {
  160. MYASSERT(Context->hEvent);
  161. return FALSE;
  162. }
  163. if (Context->bDialogExited) {
  164. //
  165. // dialog has already exited, we wont get another event
  166. //
  167. return FALSE;
  168. }
  169. while (KeepWaiting) {
  170. WaitProcStatus = MyMsgWaitForMultipleObjectsEx(
  171. 1,
  172. &Context->hEvent,
  173. INFINITE,
  174. QS_ALLINPUT,
  175. MWMO_ALERTABLE | MWMO_INPUTAVAILABLE);
  176. switch (WaitProcStatus) {
  177. case WAIT_OBJECT_0 + 1: { // Process gui messages
  178. MSG msg;
  179. while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
  180. TranslateMessage(&msg);
  181. DispatchMessage(&msg);
  182. }
  183. // fall through ...
  184. }
  185. case WAIT_IO_COMPLETION:
  186. break;
  187. case WAIT_OBJECT_0:
  188. case WAIT_TIMEOUT:
  189. default:
  190. KeepWaiting = FALSE;
  191. break;
  192. }
  193. }
  194. return TRUE;
  195. }
  196. UINT
  197. PostUiMessage (
  198. IN PQUEUECONTEXT Context,
  199. IN UINT UiType,
  200. IN UINT CancelCode,
  201. IN OUT PVOID UiParameters
  202. )
  203. {
  204. MSG msg;
  205. if(IsWindow(Context->ProgressDialog)) {
  206. //
  207. // Let progress ui thread handle it.
  208. //
  209. Context->lParam = FILEOP_ABORT; // in case nobody gets chance to set Context->lParam
  210. PostMessage(
  211. Context->ProgressDialog,
  212. WMX_PERFORMUI,
  213. MAKEWPARAM(UiType,CancelCode),
  214. (LPARAM)UiParameters
  215. );
  216. pWaitForUiResponse(Context);
  217. return (UINT)Context->lParam;
  218. } else {
  219. //
  220. // There is no progress thread so do it synchronously.
  221. //
  222. return (UINT)pPerformUi(Context,UiType,UiParameters);
  223. }
  224. return 0;
  225. }
  226. UINT
  227. pNotificationStartQueue(
  228. IN PQUEUECONTEXT Context
  229. )
  230. /*++
  231. Routine Description:
  232. Handle SPFILENOTIFY_STARTQUEUE.
  233. Creates a progress dialog in a separate thread.
  234. Arguments:
  235. Context - supplies queue context.
  236. Return Value:
  237. 0 if unsuccessful, non-0 if successful.
  238. --*/
  239. {
  240. ULONG_PTR Thread;
  241. MSG msg;
  242. //
  243. // SetupCommitFileQueue could have been called in a different
  244. // thread. Adjust the thread id.
  245. //
  246. Context->MainThreadId = GetCurrentThreadId();
  247. //
  248. // Force this thread to have a message queue. If we don't do this,
  249. // then PostMessage, PostThreadMessage, etc can fail, which results
  250. // in hangs in some cases since we rely heavily on these for
  251. // progress, synchronization, etc.
  252. //
  253. PeekMessage(&msg,NULL,0,0,PM_NOREMOVE);
  254. if(Context->AlternateProgressWindow) {
  255. //
  256. // Either the caller is supplying their own window for progress UI,
  257. // or this is a silent install (AlternateProgressWindow is
  258. // INVALID_HANDLE_VALUE).
  259. //
  260. return(TRUE);
  261. } else {
  262. //
  263. // Fire up the progress dialog in a separate thread.
  264. // This allows it to be responsive without suspending
  265. // the file operations.
  266. //
  267. Thread = _beginthread(
  268. pSetupProgressThread,
  269. 0,
  270. Context
  271. );
  272. if(Thread == -1) {
  273. //
  274. // assume OOM
  275. //
  276. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  277. return(0);
  278. }
  279. //
  280. // Wait for notification from the thread about the state
  281. // of the dialog. Assume out of memory if we fail
  282. //
  283. if(!pWaitForUiResponse(Context) || Context->bDialogExited) {
  284. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  285. return FALSE;
  286. } else {
  287. return TRUE;
  288. }
  289. }
  290. }
  291. UINT
  292. pNotificationStartEndSubqueue(
  293. IN PQUEUECONTEXT Context,
  294. IN BOOL Start,
  295. IN UINT_PTR Operation,
  296. IN UINT_PTR OpCount
  297. )
  298. /*++
  299. Routine Description:
  300. Handle SPFILENOTIFY_STARTSUBQUEUE, SPFILENOTIFY_ENDSUBQUEUE.
  301. Initializes/terminates a progress control.
  302. Also sets progress dialog caption.
  303. Arguments:
  304. Context - supplies queue context.
  305. Start - if TRUE, then this routine is being called to handle
  306. a subqueue start notification. Otherwise it's supposed to
  307. handle a subqueue end notification.
  308. Operation - one of FILEOP_COPY, FILEOP_DELETE, FILEOP_RENAME.
  309. OpCount - supplies number of copies, renames, or deletes.
  310. Return Value:
  311. 0 if unsuccessful, non-0 if successful.
  312. --*/
  313. {
  314. UINT rc;
  315. UINT CaptionStringId;
  316. TCHAR ParentText[256];
  317. BOOL GotParentText;
  318. PCTSTR CaptionText;
  319. UINT AnimationId;
  320. HWND Animation;
  321. rc = 1; // assume success.
  322. if(Context->Cancelled) {
  323. SetLastError(ERROR_CANCELLED);
  324. return(0);
  325. }
  326. if(Start) {
  327. if(IsWindow(Context->OwnerWindow)
  328. && GetWindowText(Context->OwnerWindow,ParentText,256)) {
  329. GotParentText = TRUE;
  330. } else {
  331. GotParentText = FALSE;
  332. }
  333. //
  334. // Clean out the text fields first.
  335. //
  336. if(IsWindow(Context->ProgressDialog)) {
  337. SetDlgItemText(Context->ProgressDialog,IDT_TEXT1,TEXT(""));
  338. SetDlgItemText(Context->ProgressDialog,IDT_TEXT2,TEXT(""));
  339. }
  340. switch(Operation) {
  341. case FILEOP_COPY:
  342. //
  343. // IDT_TEXT1 = target name sans path
  344. // IDT_TEXT2 = target name with path
  345. //
  346. if(IsWindow(Context->ProgressDialog)) {
  347. ShowWindow(GetDlgItem(Context->ProgressDialog,IDT_TEXT2),SW_SHOW);
  348. }
  349. CaptionStringId = GotParentText ? IDS_COPY_CAPTION1 : IDS_COPY_CAPTION2;
  350. AnimationId = IDA_FILECOPY;
  351. break;
  352. case FILEOP_RENAME:
  353. if(IsWindow(Context->ProgressDialog)) {
  354. ShowWindow(GetDlgItem(Context->ProgressDialog,IDT_TEXT2),SW_SHOW);
  355. }
  356. CaptionStringId = GotParentText ? IDS_RENAME_CAPTION1 : IDS_RENAME_CAPTION2;
  357. AnimationId = IDA_FILECOPY;
  358. break;
  359. case FILEOP_DELETE:
  360. //
  361. // IDT_TEXT1 = target name sans path
  362. // IDT_TEXT2 = path
  363. //
  364. if(IsWindow(Context->ProgressDialog)) {
  365. ShowWindow(GetDlgItem(Context->ProgressDialog,IDT_TEXT2),SW_HIDE);
  366. }
  367. CaptionStringId = GotParentText ? IDS_DELETE_CAPTION1 : IDS_DELETE_CAPTION2;
  368. AnimationId = IDA_FILEDEL;
  369. break;
  370. case FILEOP_BACKUP:
  371. // handle the new backup case (this is for a backup queue as opposed to on demand
  372. if(IsWindow(Context->ProgressDialog)) {
  373. ShowWindow(GetDlgItem(Context->ProgressDialog,IDT_TEXT2),SW_SHOW);
  374. }
  375. CaptionStringId = GotParentText ? IDS_BACKUP_CAPTION1 : IDS_BACKUP_CAPTION2;
  376. AnimationId = IDA_FILECOPY;
  377. break;
  378. default:
  379. SetLastError(ERROR_INVALID_PARAMETER);
  380. rc = 0;
  381. break;
  382. }
  383. if(rc) {
  384. //
  385. // Set dialog caption.
  386. //
  387. if(GotParentText) {
  388. CaptionText = FormatStringMessage(CaptionStringId,ParentText);
  389. } else {
  390. CaptionText = MyLoadString(CaptionStringId);
  391. }
  392. if(CaptionText) {
  393. if(IsWindow(Context->ProgressDialog)) {
  394. if(!SetWindowText(Context->ProgressDialog,CaptionText)) {
  395. SetWindowText(Context->ProgressDialog,TEXT(""));
  396. }
  397. }
  398. MyFree(CaptionText);
  399. }
  400. if(Context->AlternateProgressWindow) {
  401. //
  402. // If this is really an alternate progress window, notify it
  403. // about the number of operations. Copy only.
  404. //
  405. if((Operation == FILEOP_COPY) &&
  406. (Context->AlternateProgressWindow != INVALID_HANDLE_VALUE)) {
  407. SendMessage(Context->AlternateProgressWindow,
  408. Context->ProgressMsg,
  409. 0,
  410. OpCount
  411. );
  412. }
  413. } else {
  414. //
  415. // Set up the progress control. Each file will be 1 tick.
  416. //
  417. if(IsWindow(Context->ProgressBar)) {
  418. SendMessage(Context->ProgressBar,PBM_SETRANGE,0,MAKELPARAM(0,OpCount));
  419. SendMessage(Context->ProgressBar,PBM_SETSTEP,1,0);
  420. SendMessage(Context->ProgressBar,PBM_SETPOS,0,0);
  421. }
  422. //
  423. // And set up the animation control based on the operation type.
  424. //
  425. if(OpCount && IsWindow(Context->ProgressDialog)) {
  426. Animation = GetDlgItem(Context->ProgressDialog,IDA_ANIMATION);
  427. if(Animation) {
  428. Animate_Open(Animation,MAKEINTRESOURCE(AnimationId));
  429. Animate_Play(Animation,0,-1,-1);
  430. }
  431. }
  432. }
  433. }
  434. } else {
  435. //
  436. // Stop the animation control. Note that if the op count was 0
  437. // then we never started it, so stopping/unloading will give an error,
  438. // which we ignore. It's not harmful.
  439. //
  440. if(!Context->AlternateProgressWindow && IsWindow(Context->ProgressDialog)) {
  441. Animation = GetDlgItem(Context->ProgressDialog,IDA_ANIMATION);
  442. if (Animation) {
  443. Animate_Stop(Animation);
  444. Animate_Close(Animation);
  445. }
  446. }
  447. }
  448. return(rc);
  449. }
  450. UINT
  451. pNotificationStartOperation(
  452. IN PQUEUECONTEXT Context,
  453. IN PFILEPATHS FilePaths,
  454. IN UINT_PTR Operation
  455. )
  456. /*++
  457. Routine Description:
  458. Handle SPFILENOTIFY_STARTRENAME, SPFILENOTIFY_STARTDELETE,
  459. SPFILENOTIFY_STARTCOPY or SPFILENOTIFY_STARTBACKUP.
  460. Updates text in the progress dialog to indicate the files
  461. involved in the operation.
  462. Arguments:
  463. Context - supplies queue context.
  464. Start - if TRUE, then this routine is being called to handle
  465. a subqueue start notification. Otherwise it's supposed to
  466. handle a subqueue end notification.
  467. Operation - one of FILEOP_COPY, FILEOP_DELETE, FILEOP_RENAME.
  468. OpCount - supplies number of copies, renames, or deletes.
  469. Return Value:
  470. FILEOP_ABORT if error, otherwise FILEOP_DOIT.
  471. --*/
  472. {
  473. PCTSTR Text1,Text2;
  474. PTSTR p;
  475. TCHAR Target[MAX_PATH];
  476. UINT rc;
  477. DWORD ec;
  478. if(Context->Cancelled) {
  479. SetLastError(ERROR_CANCELLED);
  480. return(FILEOP_ABORT);
  481. }
  482. Text1 = Text2 = NULL;
  483. rc = FILEOP_ABORT;
  484. ec = ERROR_NOT_ENOUGH_MEMORY;
  485. switch(Operation) {
  486. case FILEOP_COPY:
  487. lstrcpyn(Target,FilePaths->Target,MAX_PATH);
  488. if(p = _tcsrchr(Target,TEXT('\\'))) {
  489. //
  490. // Ignore the source filename completely.
  491. //
  492. *p++ = 0;
  493. Text1 = DuplicateString(p);
  494. Text2 = FormatStringMessage(IDS_FILEOP_TO,Target);
  495. } else {
  496. //
  497. // Assume not full path -- strange case, but deal with it anyway.
  498. //
  499. Text1 = DuplicateString(FilePaths->Target);
  500. Text2 = DuplicateString(TEXT(""));
  501. }
  502. break;
  503. case FILEOP_RENAME:
  504. Text1 = DuplicateString(FilePaths->Source);
  505. if(p = _tcsrchr(FilePaths->Target,TEXT('\\'))) {
  506. p++;
  507. } else {
  508. p = (PTSTR)FilePaths->Target;
  509. }
  510. Text2 = FormatStringMessage(IDS_FILEOP_TO,p);
  511. break;
  512. case FILEOP_DELETE:
  513. lstrcpyn(Target,FilePaths->Target,MAX_PATH);
  514. if(p = _tcsrchr(Target,TEXT('\\'))) {
  515. *p++ = 0;
  516. Text1 = DuplicateString(p);
  517. Text2 = FormatStringMessage(IDS_FILEOP_FROM,Target);
  518. } else {
  519. //
  520. // Assume not full path -- strange case, but deal with it anyway.
  521. //
  522. Text1 = DuplicateString(FilePaths->Target);
  523. Text2 = DuplicateString(TEXT(""));
  524. }
  525. break;
  526. case FILEOP_BACKUP:
  527. lstrcpyn(Target,FilePaths->Source,MAX_PATH);
  528. if(p = _tcsrchr(Target,TEXT('\\'))) {
  529. //
  530. // FilePaths->Source = what we're backing up (which is target of a restore)
  531. // FilePaths->Target = where we're backing up to (block backup) or NULL (demand backup)
  532. //
  533. *p++ = 0;
  534. if (FilePaths->Target == NULL) {
  535. // Backing up <Filename>
  536. Text1 = FormatStringMessage(IDS_FILEOP_BACKUP,p);
  537. } else {
  538. // <Filename> (title already says Backing up)
  539. Text1 = DuplicateString(p);
  540. }
  541. // From <Directory>
  542. Text2 = FormatStringMessage(IDS_FILEOP_FROM,Target);
  543. } else {
  544. //
  545. // Assume not full path -- strange case, but deal with it anyway.
  546. //
  547. if (FilePaths->Source == NULL) {
  548. Text1 = FormatStringMessage(IDS_FILEOP_BACKUP,Target);
  549. } else {
  550. Text1 = DuplicateString(Target);
  551. }
  552. Text2 = DuplicateString(TEXT(""));
  553. }
  554. break;
  555. default:
  556. ec = ERROR_INVALID_PARAMETER;
  557. break;
  558. }
  559. if(Text1 && Text2) {
  560. if(IsWindow(Context->ProgressDialog)) {
  561. SetDlgItemText(Context->ProgressDialog,IDT_TEXT1,Text1);
  562. SetDlgItemText(Context->ProgressDialog,IDT_TEXT2,Text2);
  563. }
  564. rc = FILEOP_DOIT;
  565. }
  566. if(Text1) {
  567. MyFree(Text1);
  568. }
  569. if(Text2) {
  570. MyFree(Text2);
  571. }
  572. SetLastError(ec);
  573. return(rc);
  574. }
  575. UINT
  576. pNotificationErrorCopy(
  577. IN PQUEUECONTEXT Context,
  578. IN PFILEPATHS FilePaths,
  579. OUT PTSTR PathOut
  580. )
  581. {
  582. UINT rc;
  583. COPYERRORUI CopyError;
  584. CopyError.FilePaths = FilePaths;
  585. CopyError.PathOut = PathOut;
  586. //
  587. // Buffer gets the pathname part of the source
  588. // and p points to the filename part of the source.
  589. //
  590. lstrcpyn(CopyError.Buffer,FilePaths->Source,MAX_PATH);
  591. CopyError.Filename = _tcsrchr(CopyError.Buffer,TEXT('\\'));
  592. *CopyError.Filename++ = 0;
  593. //
  594. // The noskip and warnifskip flags are really mutually exclusive
  595. // but we don't try to enforce that here. Just pass through as
  596. // appropriate.
  597. //
  598. CopyError.Flags = 0;
  599. if(FilePaths->Flags & SP_COPY_NOSKIP) {
  600. CopyError.Flags |= IDF_NOSKIP;
  601. }
  602. if(FilePaths->Flags & SP_COPY_WARNIFSKIP) {
  603. CopyError.Flags |= IDF_WARNIFSKIP;
  604. }
  605. //
  606. // Also pass through the 'no browse' flag.
  607. //
  608. if(FilePaths->Flags & SP_COPY_NOBROWSE) {
  609. CopyError.Flags |= IDF_NOBROWSE;
  610. }
  611. rc = PostUiMessage (Context, UI_COPYERROR, DPROMPT_CANCEL, &CopyError);
  612. switch(rc) {
  613. case DPROMPT_SUCCESS:
  614. //
  615. // If a new path is indicated, verify that it actually changed.
  616. //
  617. if(CopyError.PathOut[0] &&
  618. !lstrcmpi(CopyError.Buffer,CopyError.PathOut)) {
  619. CopyError.PathOut[0] = 0;
  620. }
  621. rc = FILEOP_RETRY;
  622. break;
  623. case DPROMPT_SKIPFILE:
  624. rc = FILEOP_SKIP;
  625. break;
  626. case DPROMPT_CANCEL:
  627. SetLastError(ERROR_CANCELLED);
  628. rc = FILEOP_ABORT;
  629. break;
  630. default:
  631. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  632. rc = FILEOP_ABORT;
  633. break;
  634. }
  635. return(rc);
  636. }
  637. UINT
  638. pNotificationStartRegistration(
  639. IN PQUEUECONTEXT Context,
  640. IN PSP_REGISTER_CONTROL_STATUS ControlStatus,
  641. IN BOOL Register
  642. )
  643. {
  644. return FILEOP_DOIT;
  645. }
  646. UINT
  647. pNotificationErrorDelete(
  648. IN PQUEUECONTEXT Context,
  649. IN PFILEPATHS FilePaths
  650. )
  651. {
  652. UINT rc;
  653. //
  654. // Certain errors are not actually errors.
  655. //
  656. if((FilePaths->Win32Error == ERROR_FILE_NOT_FOUND)
  657. || (FilePaths->Win32Error == ERROR_PATH_NOT_FOUND)) {
  658. return(FILEOP_SKIP);
  659. }
  660. rc = PostUiMessage (Context, UI_DELETEERROR, DPROMPT_CANCEL, FilePaths);
  661. switch(rc) {
  662. case DPROMPT_SUCCESS:
  663. return(FILEOP_RETRY);
  664. case DPROMPT_SKIPFILE:
  665. return(FILEOP_SKIP);
  666. case DPROMPT_CANCEL:
  667. SetLastError(ERROR_CANCELLED);
  668. return(FILEOP_ABORT);
  669. default:
  670. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  671. return(FILEOP_ABORT);
  672. }
  673. }
  674. UINT
  675. pNotificationErrorRename(
  676. IN PQUEUECONTEXT Context,
  677. IN PFILEPATHS FilePaths
  678. )
  679. {
  680. UINT rc;
  681. rc = PostUiMessage (Context, UI_RENAMEERROR, DPROMPT_CANCEL, FilePaths);
  682. switch(rc) {
  683. case DPROMPT_SUCCESS:
  684. return(FILEOP_RETRY);
  685. case DPROMPT_SKIPFILE:
  686. return(FILEOP_SKIP);
  687. case DPROMPT_CANCEL:
  688. SetLastError(ERROR_CANCELLED);
  689. return(FILEOP_ABORT);
  690. default:
  691. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  692. return(FILEOP_ABORT);
  693. }
  694. }
  695. UINT
  696. pNotificationErrorBackup(
  697. IN PQUEUECONTEXT Context,
  698. IN PFILEPATHS FilePaths
  699. )
  700. {
  701. UINT rc;
  702. if(!(FilePaths->Flags & SP_BACKUP_SPECIAL)) {
  703. return FILEOP_SKIP;
  704. }
  705. rc = PostUiMessage (Context, UI_BACKUPERROR, DPROMPT_CANCEL, FilePaths);
  706. switch(rc) {
  707. case DPROMPT_SUCCESS:
  708. return(FILEOP_RETRY);
  709. case DPROMPT_SKIPFILE:
  710. return(FILEOP_SKIP);
  711. case DPROMPT_CANCEL:
  712. SetLastError(ERROR_CANCELLED);
  713. return(FILEOP_ABORT);
  714. default:
  715. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  716. return(FILEOP_ABORT);
  717. }
  718. }
  719. UINT
  720. pNotificationNeedMedia(
  721. IN PQUEUECONTEXT Context,
  722. IN PSOURCE_MEDIA SourceMedia,
  723. OUT PTSTR PathOut
  724. )
  725. {
  726. UINT rc;
  727. TCHAR Buffer[MAX_PATH];
  728. NEEDMEDIAUI NeedMedia;
  729. if(Context->Cancelled) {
  730. SetLastError(ERROR_CANCELLED);
  731. return(FILEOP_ABORT);
  732. }
  733. NeedMedia.SourceMedia = SourceMedia;
  734. NeedMedia.PathOut = PathOut;
  735. //
  736. // Remember the name of this media.
  737. //
  738. if(Context->CurrentSourceName) {
  739. MyFree(Context->CurrentSourceName);
  740. Context->CurrentSourceName = NULL;
  741. }
  742. if(SourceMedia->Description) {
  743. Context->CurrentSourceName = DuplicateString(SourceMedia->Description);
  744. if(!Context->CurrentSourceName) {
  745. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  746. return(FILEOP_ABORT);
  747. }
  748. }
  749. //
  750. // Set the source file in the progress dialog
  751. // so it matches the file being sought.
  752. //
  753. if(!(SourceMedia->Flags & SP_FLAG_CABINETCONTINUATION)) {
  754. if(IsWindow(Context->ProgressDialog) && !Context->ScreenReader) {
  755. DWORD chars;
  756. lstrcpyn(Buffer,SourceMedia->SourcePath,MAX_PATH);
  757. pSetupConcatenatePaths(Buffer,SourceMedia->SourceFile,MAX_PATH,NULL);
  758. SetTruncatedDlgItemText(Context->ProgressDialog,IDT_TEXT1,Buffer);
  759. SetDlgItemText(Context->ProgressDialog,IDT_TEXT2,TEXT(""));
  760. }
  761. }
  762. //
  763. // The noskip and warnifskip flags are really mutually exclusive
  764. // but we don't try to enforce that here. Just pass through as
  765. // appropriate.
  766. //
  767. // Allow skip if this is not a cabinet continuation and
  768. // the noskip flag is not set.
  769. //
  770. NeedMedia.Flags = IDF_CHECKFIRST;
  771. if(SourceMedia->Flags & (SP_FLAG_CABINETCONTINUATION | SP_COPY_NOSKIP)) {
  772. NeedMedia.Flags |= IDF_NOSKIP;
  773. }
  774. if(SourceMedia->Flags & SP_COPY_WARNIFSKIP) {
  775. NeedMedia.Flags |= IDF_WARNIFSKIP;
  776. }
  777. if(SourceMedia->Flags & SP_COPY_NOBROWSE) {
  778. NeedMedia.Flags |= IDF_NOBROWSE;
  779. }
  780. rc = PostUiMessage (Context, UI_NEEDMEDIA, DPROMPT_CANCEL, &NeedMedia);
  781. switch(rc) {
  782. case DPROMPT_SUCCESS:
  783. //
  784. // If the path really has changed, then return NEWPATH.
  785. // Otherwise return DOIT. Account for trailing backslash
  786. // differences.
  787. //
  788. lstrcpyn(Buffer,SourceMedia->SourcePath,MAX_PATH);
  789. rc = lstrlen(Buffer);
  790. if(rc && (*CharPrev(Buffer,Buffer+rc) == TEXT('\\'))) {
  791. Buffer[rc-1] = TEXT('\0'); // valid to do if last char is '\'
  792. }
  793. rc = lstrlen(NeedMedia.PathOut);
  794. if(rc && (*CharPrev(NeedMedia.PathOut,NeedMedia.PathOut+rc) == TEXT('\\'))) {
  795. NeedMedia.PathOut[rc-1] = TEXT('\0'); // valid to do if last char is '\'
  796. }
  797. rc = (lstrcmpi(SourceMedia->SourcePath,NeedMedia.PathOut) ?
  798. FILEOP_NEWPATH : FILEOP_DOIT);
  799. //
  800. // Make sure <drive>: ends with a \.
  801. //
  802. if(NeedMedia.PathOut[0] && (NeedMedia.PathOut[1] == TEXT(':')) &&
  803. !NeedMedia.PathOut[2]) {
  804. NeedMedia.PathOut[2] = TEXT('\\');
  805. NeedMedia.PathOut[3] = TEXT('\0');
  806. }
  807. break;
  808. case DPROMPT_SKIPFILE:
  809. rc = FILEOP_SKIP;
  810. break;
  811. case DPROMPT_CANCEL:
  812. SetLastError(ERROR_CANCELLED);
  813. rc = FILEOP_ABORT;
  814. break;
  815. default:
  816. SetLastError(ERROR_NOT_ENOUGH_MEMORY);
  817. rc = FILEOP_ABORT;
  818. break;
  819. }
  820. return(rc);
  821. }
  822. INT_PTR
  823. pNotificationVersionDlgProc(
  824. IN HWND hdlg,
  825. IN UINT msg,
  826. IN WPARAM wParam,
  827. IN LPARAM lParam
  828. )
  829. {
  830. PVERDLGCONTEXT context;
  831. PFILEPATHS filePaths;
  832. HWND hwnd;
  833. TCHAR text[128];
  834. TCHAR Buffer[MAX_PATH*2];
  835. PCTSTR message;
  836. int i;
  837. TCHAR SourceLangName[128];
  838. TCHAR TargetLangName[128];
  839. switch(msg) {
  840. case WM_INITDIALOG:
  841. context = (PVERDLGCONTEXT)lParam;
  842. MYASSERT(context != NULL);
  843. filePaths = (PFILEPATHS)context->Param1;
  844. if(!SetProp(hdlg,DialogPropName,(HANDLE)context)) {
  845. goto no_memory;
  846. }
  847. //
  848. // Set the source and target filenames. Hack: if the source filename
  849. // looks like one of our temporary filenames from a cabinet extraction,
  850. // hide it.
  851. //
  852. message = pSetupGetFileTitle(filePaths->Source);
  853. i = lstrlen(message);
  854. if((i > 8)
  855. && (_totupper(message[0]) == TEXT('S'))
  856. && (_totupper(message[1]) == TEXT('E'))
  857. && (_totupper(message[2]) == TEXT('T'))
  858. && (_totupper(message[3]) == TEXT('P'))
  859. && !_tcsicmp(message+(i-4),TEXT(".TMP"))) {
  860. ShowWindow(GetDlgItem(hdlg,IDT_TEXT7),SW_HIDE);
  861. } else {
  862. GetDlgItemText(hdlg,IDT_TEXT7,text,SIZECHARS(text));
  863. message = FormatStringMessageFromString(text,filePaths->Source);
  864. if (message == NULL) goto no_memory;
  865. SetTruncatedDlgItemText(hdlg,IDT_TEXT7,message);
  866. MyFree(message);
  867. }
  868. GetDlgItemText(hdlg,IDT_TEXT8,text,SIZECHARS(text));
  869. message = FormatStringMessageFromString(text,filePaths->Target);
  870. if (message == NULL) goto no_memory;
  871. SetTruncatedDlgItemText(hdlg,IDT_TEXT8,message);
  872. MyFree(message);
  873. if (context->Notification & SPFILENOTIFY_LANGMISMATCH) {
  874. //
  875. // Language mismatch has the highest priority.
  876. //
  877. context->Notification = SPFILENOTIFY_LANGMISMATCH; // force other bits off, for NoToAll
  878. //
  879. // Format the overwrite question.
  880. //
  881. if(PRIMARYLANGID(LOWORD(context->Param2))==LANG_NEUTRAL) {
  882. LoadString(
  883. MyDllModuleHandle,
  884. IDS_LANG_NEUTRAL,
  885. SourceLangName,
  886. SIZECHARS(SourceLangName)
  887. );
  888. } else {
  889. i = GetLocaleInfo(
  890. MAKELCID(LOWORD(context->Param2),SORT_DEFAULT),
  891. LOCALE_SLANGUAGE,
  892. SourceLangName,
  893. SIZECHARS(SourceLangName)
  894. );
  895. if(!i) {
  896. LoadString(
  897. MyDllModuleHandle,
  898. IDS_LANG_UNKNOWN,
  899. SourceLangName,
  900. SIZECHARS(SourceLangName)
  901. );
  902. }
  903. }
  904. if(PRIMARYLANGID(HIWORD(context->Param2))==LANG_NEUTRAL) {
  905. LoadString(
  906. MyDllModuleHandle,
  907. IDS_LANG_NEUTRAL,
  908. TargetLangName,
  909. SIZECHARS(TargetLangName)
  910. );
  911. } else {
  912. i = GetLocaleInfo(
  913. MAKELCID(HIWORD(context->Param2),SORT_DEFAULT),
  914. LOCALE_SLANGUAGE,
  915. TargetLangName,
  916. SIZECHARS(TargetLangName)
  917. );
  918. if(!i) {
  919. LoadString(
  920. MyDllModuleHandle,
  921. IDS_LANG_UNKNOWN,
  922. TargetLangName,
  923. SIZECHARS(TargetLangName)
  924. );
  925. }
  926. }
  927. GetDlgItemText(hdlg,IDT_TEXT4,text,SIZECHARS(text));
  928. message = FormatStringMessageFromString(text,TargetLangName,SourceLangName);
  929. if (message == NULL) goto no_memory;
  930. SetDlgItemText(hdlg,IDT_TEXT4,message);
  931. MyFree(message);
  932. //
  933. // Turn off the TARGETNEWER and TARGETEXISTS messages.
  934. //
  935. hwnd = GetDlgItem(hdlg,IDT_TEXT2);
  936. ShowWindow(hwnd,SW_HIDE);
  937. hwnd = GetDlgItem(hdlg,IDT_TEXT3);
  938. ShowWindow(hwnd,SW_HIDE);
  939. hwnd = GetDlgItem(hdlg,IDT_TEXT5);
  940. ShowWindow(hwnd,SW_HIDE);
  941. hwnd = GetDlgItem(hdlg,IDT_TEXT6);
  942. ShowWindow(hwnd,SW_HIDE);
  943. } else if (context->Notification & SPFILENOTIFY_TARGETNEWER) {
  944. //
  945. // Target being newer has second highest priority.
  946. //
  947. context->Notification = SPFILENOTIFY_TARGETNEWER; // force other bits off, for NoToAll
  948. //
  949. // Turn off the LANGMISMATCH and TARGETEXISTS messages.
  950. //
  951. hwnd = GetDlgItem(hdlg,IDT_TEXT1);
  952. ShowWindow(hwnd,SW_HIDE);
  953. hwnd = GetDlgItem(hdlg,IDT_TEXT3);
  954. ShowWindow(hwnd,SW_HIDE);
  955. hwnd = GetDlgItem(hdlg,IDT_TEXT4);
  956. ShowWindow(hwnd,SW_HIDE);
  957. hwnd = GetDlgItem(hdlg,IDT_TEXT6);
  958. ShowWindow(hwnd,SW_HIDE);
  959. } else { // must be exactly SPFILENOTIFY_TARGETEXISTS
  960. //
  961. // Target existing has the lowest priority.
  962. //
  963. // Turn off the LANGMISMATCH and TARGETNEWER messages.
  964. //
  965. hwnd = GetDlgItem(hdlg,IDT_TEXT1);
  966. ShowWindow(hwnd,SW_HIDE);
  967. hwnd = GetDlgItem(hdlg,IDT_TEXT2);
  968. ShowWindow(hwnd,SW_HIDE);
  969. hwnd = GetDlgItem(hdlg,IDT_TEXT4);
  970. ShowWindow(hwnd,SW_HIDE);
  971. hwnd = GetDlgItem(hdlg,IDT_TEXT5);
  972. ShowWindow(hwnd,SW_HIDE);
  973. }
  974. PostMessage(hdlg,WMX_HELLO,0,0);
  975. break;
  976. case WMX_HELLO:
  977. //
  978. // If this guy has no owner force him to the foreground.
  979. // This catches cases where people are using a series of
  980. // dialogs and then some setup apis, because when they
  981. // close a dialog focus switches away from them.
  982. //
  983. hwnd = GetWindow(hdlg,GW_OWNER);
  984. if(!IsWindow(hwnd)) {
  985. SetForegroundWindow(hdlg);
  986. }
  987. break;
  988. case WM_COMMAND:
  989. context = (PVERDLGCONTEXT)GetProp(hdlg,DialogPropName);
  990. MYASSERT(context != NULL);
  991. switch (GET_WM_COMMAND_ID(wParam,lParam)) {
  992. case IDYES:
  993. EndDialog(hdlg,IDYES); // copy this file
  994. break;
  995. case IDNO:
  996. EndDialog(hdlg,IDNO); // skip this file
  997. break;
  998. case IDB_NOTOALL:
  999. //
  1000. // No to All was selected. Add this notification type to the
  1001. // NoToAllMask so that we don't ask about it again.
  1002. //
  1003. context->QueueContext->NoToAllMask |= context->Notification;
  1004. EndDialog(hdlg,IDNO); // skip this file
  1005. break;
  1006. }
  1007. break;
  1008. default:
  1009. return FALSE;
  1010. }
  1011. return TRUE;
  1012. no_memory:
  1013. pSetupOutOfMemory(
  1014. IsWindow(context->QueueContext->ProgressDialog) ?
  1015. context->QueueContext->ProgressDialog : context->QueueContext->OwnerWindow
  1016. );
  1017. EndDialog(hdlg,IDNO); // skip this file
  1018. return TRUE;
  1019. }
  1020. LPARAM
  1021. pPerformUi (
  1022. IN PQUEUECONTEXT Context,
  1023. IN UINT UiType,
  1024. IN PVOID UiParameters
  1025. )
  1026. {
  1027. PCOPYERRORUI CopyError;
  1028. PFILEPATHS FilePaths;
  1029. PNEEDMEDIAUI NeedMedia;
  1030. INT_PTR rc;
  1031. HWND Animation;
  1032. if(!Context->AlternateProgressWindow && IsWindow(Context->ProgressDialog)) {
  1033. Animation = GetDlgItem(Context->ProgressDialog,IDA_ANIMATION);
  1034. } else {
  1035. Animation = NULL;
  1036. }
  1037. switch (UiType) {
  1038. case UI_COPYERROR:
  1039. CopyError = (PCOPYERRORUI)UiParameters;
  1040. if (Animation) {
  1041. Animate_Stop(Animation);
  1042. }
  1043. rc = SetupCopyError(
  1044. IsWindow(Context->ProgressDialog) ? Context->ProgressDialog : Context->OwnerWindow,
  1045. NULL,
  1046. Context->CurrentSourceName,
  1047. CopyError->Buffer,
  1048. CopyError->Filename,
  1049. CopyError->FilePaths->Target,
  1050. CopyError->FilePaths->Win32Error,
  1051. CopyError->Flags,
  1052. CopyError->PathOut,
  1053. MAX_PATH,
  1054. NULL
  1055. );
  1056. if (Animation) {
  1057. Animate_Play(Animation,0,-1,-1);
  1058. }
  1059. break;
  1060. case UI_DELETEERROR:
  1061. FilePaths = (PFILEPATHS)UiParameters;
  1062. if (Animation) {
  1063. Animate_Stop(Animation);
  1064. }
  1065. rc = SetupDeleteError(
  1066. IsWindow(Context->ProgressDialog) ? Context->ProgressDialog : Context->OwnerWindow,
  1067. NULL,
  1068. FilePaths->Target,
  1069. FilePaths->Win32Error,
  1070. 0
  1071. );
  1072. if (Animation) {
  1073. Animate_Play(Animation,0,-1,-1);
  1074. }
  1075. break;
  1076. case UI_RENAMEERROR:
  1077. FilePaths = (PFILEPATHS)UiParameters;
  1078. if(Animation) {
  1079. Animate_Stop(Animation);
  1080. }
  1081. rc = SetupRenameError(
  1082. IsWindow(Context->ProgressDialog) ? Context->ProgressDialog : Context->OwnerWindow,
  1083. NULL,
  1084. FilePaths->Source,
  1085. FilePaths->Target,
  1086. FilePaths->Win32Error,
  1087. 0
  1088. );
  1089. if(Animation) {
  1090. Animate_Play(Animation,0,-1,-1);
  1091. }
  1092. break;
  1093. case UI_BACKUPERROR:
  1094. FilePaths = (PFILEPATHS)UiParameters;
  1095. if(Animation) {
  1096. Animate_Stop(Animation);
  1097. }
  1098. rc = SetupBackupError(
  1099. IsWindow(Context->ProgressDialog) ? Context->ProgressDialog : Context->OwnerWindow,
  1100. NULL,
  1101. FilePaths->Source,
  1102. FilePaths->Target,
  1103. FilePaths->Win32Error,
  1104. 0
  1105. );
  1106. if(Animation) {
  1107. Animate_Play(Animation,0,-1,-1);
  1108. }
  1109. break;
  1110. case UI_NEEDMEDIA:
  1111. NeedMedia = (PNEEDMEDIAUI)UiParameters;
  1112. if(Animation) {
  1113. Animate_Stop(Animation);
  1114. }
  1115. rc = SetupPromptForDisk(
  1116. IsWindow(Context->ProgressDialog) ? Context->ProgressDialog : Context->OwnerWindow,
  1117. NULL,
  1118. NeedMedia->SourceMedia->Description,
  1119. NeedMedia->SourceMedia->SourcePath,
  1120. NeedMedia->SourceMedia->SourceFile,
  1121. NeedMedia->SourceMedia->Tagfile,
  1122. NeedMedia->Flags,
  1123. NeedMedia->PathOut,
  1124. MAX_PATH,
  1125. NULL
  1126. );
  1127. if(Animation) {
  1128. Animate_Play(Animation,0,-1,-1);
  1129. }
  1130. break;
  1131. case UI_MISMATCHERROR:
  1132. if(Animation) {
  1133. Animate_Stop(Animation);
  1134. }
  1135. if(GlobalSetupFlags & (PSPGF_NONINTERACTIVE|PSPGF_UNATTENDED_SETUP)) {
  1136. rc = DPROMPT_CANCEL;
  1137. } else {
  1138. rc = DialogBoxParam(
  1139. MyDllModuleHandle,
  1140. MAKEINTRESOURCE(IDD_REPLACE),
  1141. IsWindow(Context->ProgressDialog) ?
  1142. Context->ProgressDialog : Context->OwnerWindow,
  1143. pNotificationVersionDlgProc,
  1144. (LPARAM)UiParameters
  1145. );
  1146. }
  1147. if(Animation) {
  1148. Animate_Play(Animation,0,-1,-1);
  1149. }
  1150. break;
  1151. default:
  1152. MYASSERT (0);
  1153. rc = 0;
  1154. }
  1155. return rc;
  1156. }
  1157. INT_PTR
  1158. pSetupProgressDlgProc(
  1159. IN HWND hdlg,
  1160. IN UINT msg,
  1161. IN WPARAM wParam,
  1162. IN LPARAM lParam
  1163. )
  1164. {
  1165. BOOL b;
  1166. PQUEUECONTEXT Context;
  1167. HWND hwnd;
  1168. PTSTR p;
  1169. INT_PTR i;
  1170. MSG m;
  1171. BOOL Cancelled;
  1172. HANDLE h;
  1173. static UINT uQueryCancelAutoPlay = 0;
  1174. switch(msg) {
  1175. case WM_INITDIALOG:
  1176. #ifdef PRERELEASE
  1177. if (GuiSetupInProgress) {
  1178. MYASSERT( FALSE && TEXT("bringing up file progress dialog (IDD_FILEPROGRESS) during gui-mode setup, which is a UI violation. Click yes and retrive a stack trace to determine errant caller.\n"));
  1179. }
  1180. #endif
  1181. Context = (PQUEUECONTEXT)lParam;
  1182. MYASSERT(Context != NULL);
  1183. if(!SetProp(hdlg,DialogPropName,(HANDLE)Context)) {
  1184. //
  1185. // running out of resources, kill this dialog
  1186. //
  1187. EndDialog(hdlg, -1);
  1188. return TRUE; // TRUE or FALSE, it really doesn't matter
  1189. }
  1190. #ifdef NOCANCEL_SUPPORT
  1191. //
  1192. // If cancel is not allowed, disable the cancel button.
  1193. //
  1194. if(!Context->AllowCancel) {
  1195. RECT rect;
  1196. RECT rect2;
  1197. hwnd = GetDlgItem(hdlg,IDCANCEL);
  1198. ShowWindow(hwnd,SW_HIDE);
  1199. EnableWindow(hwnd,FALSE);
  1200. GetWindowRect(hdlg,&rect);
  1201. GetWindowRect(hwnd,&rect2);
  1202. SetWindowPos(
  1203. hdlg,
  1204. NULL,
  1205. 0,0,
  1206. rect.right - rect.left,
  1207. (rect.bottom - rect.top) - (rect.bottom - rect2.top),
  1208. SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER
  1209. );
  1210. }
  1211. #endif
  1212. //
  1213. // Center the progress dialog relative to the parent window.
  1214. //
  1215. pSetupCenterWindowRelativeToParent(hdlg);
  1216. SetFocus(GetDlgItem(hdlg,IDCANCEL));
  1217. //
  1218. // The main thread is processing SPFILENOTIFY_STARTQUEUE and is
  1219. // waiting for some notification about the state of the UI thread.
  1220. // Let the main thread know we succeeded, and pass back a real
  1221. // handle to this thread that can be used to wait for it to terminate.
  1222. //
  1223. b = DuplicateHandle(
  1224. GetCurrentProcess(), // source process
  1225. GetCurrentThread(), // source handle
  1226. GetCurrentProcess(), // target process
  1227. &h, // new handle
  1228. 0, // ignored with DUPLICATE_SAME_ACCESS
  1229. FALSE, // not inheritable
  1230. DUPLICATE_SAME_ACCESS
  1231. );
  1232. if(!b) {
  1233. //
  1234. // Since we cannot duplicate the handle, we must kill this dialog,
  1235. // because otherwise the inter-thread communication will be broken
  1236. // and things go horribly wrong.
  1237. //
  1238. EndDialog(hdlg, -1);
  1239. return TRUE; // TRUE or FALSE, it really doesn't matter
  1240. }
  1241. //
  1242. // Store the dialog and progress bar handles in the context structure.
  1243. //
  1244. Context->ProgressDialog = hdlg;
  1245. Context->ProgressBar = GetDlgItem(hdlg,IDC_PROGRESS);
  1246. Context->UiThreadHandle = h;
  1247. PostMessage(hdlg,WMX_HELLO,0,0); // put WMX_HELLO on our message queue
  1248. SetEvent(Context->hEvent); // inform caller we've done initialization
  1249. b = FALSE;
  1250. break;
  1251. case WMX_HELLO:
  1252. //
  1253. // If this guy has no owner force him to the foreground.
  1254. // This catches cases where people are using a series of
  1255. // dialogs and then some setup apis, because when they
  1256. // close a dialog focus switches away from them.
  1257. //
  1258. hwnd = GetWindow(hdlg,GW_OWNER);
  1259. if(!IsWindow(hwnd)) {
  1260. SetForegroundWindow(hdlg);
  1261. }
  1262. b = TRUE;
  1263. break;
  1264. case WMX_PERFORMUI:
  1265. b = TRUE;
  1266. Context = (PQUEUECONTEXT)GetProp(hdlg,DialogPropName);
  1267. MYASSERT(Context != NULL);
  1268. //
  1269. // We'd better not already have any UI pending...
  1270. //
  1271. MYASSERT(Context->PendingUiType == UI_NONE);
  1272. if (Context->MessageBoxUp == TRUE) {
  1273. Context->PendingUiType = LOWORD (wParam);
  1274. Context->CancelReturnCode = HIWORD (wParam);
  1275. Context->PendingUiParameters = (PVOID)lParam;
  1276. } else {
  1277. Context->lParam = pPerformUi (Context, LOWORD(wParam), (PVOID)lParam);
  1278. SetEvent(Context->hEvent); // wakeup main thread (lParam has Ui result)
  1279. }
  1280. break;
  1281. case WM_COMMAND:
  1282. Context = (PQUEUECONTEXT)GetProp(hdlg,DialogPropName);
  1283. MYASSERT(Context != NULL);
  1284. if((HIWORD(wParam) == BN_CLICKED) && (LOWORD(wParam) == IDCANCEL)) {
  1285. p = MyLoadString(IDS_CANCELFILEOPS);
  1286. Cancelled = FALSE;
  1287. if(p) {
  1288. //
  1289. // While the message box is up, the main thread is still copying files,
  1290. // and it might just complete. If that happens, the main thread will
  1291. // post us WMX_KILLDIALOG, which would cause this dialog to nuke itself
  1292. // out from under the message box. The main thread would then continue
  1293. // executing while the message box is sitting there. Some components
  1294. // actually unload setupapi.dll at that point, and so when the user
  1295. // dismisses the message box, an AV results.
  1296. //
  1297. // We can't freeze the main thread via SuspendThread because then
  1298. // that thread, which probably owns this dialog's parent window,
  1299. // will not be able to process messages that result from the message box's
  1300. // creation. Result is that the message box never comes up and the process
  1301. // is deadlocked.
  1302. //
  1303. // We'd better not already have a message box up!
  1304. //
  1305. MYASSERT(!Context->MessageBoxUp);
  1306. Context->MessageBoxUp = TRUE;
  1307. i = MessageBox(
  1308. hdlg,
  1309. p,
  1310. TEXT(""),
  1311. MB_YESNO | MB_APPLMODAL | MB_DEFBUTTON2 | MB_SETFOREGROUND | MB_ICONQUESTION
  1312. );
  1313. Context->MessageBoxUp = FALSE;
  1314. //
  1315. // We set b to TRUE if the dialog is going away.
  1316. // We set Cancelled to TRUE if the user clicked the CANCEL button.
  1317. //
  1318. if(Context->DialogKilled) {
  1319. b = TRUE;
  1320. Cancelled = (i == IDYES);
  1321. } else {
  1322. b = (i == IDYES);
  1323. Cancelled = b;
  1324. }
  1325. MyFree(p);
  1326. } else {
  1327. pSetupOutOfMemory(hdlg);
  1328. Cancelled = TRUE;
  1329. b = TRUE;
  1330. }
  1331. if(b) {
  1332. if(Cancelled) {
  1333. Context->Cancelled = TRUE;
  1334. }
  1335. PostMessage(hdlg,WMX_KILLDIALOG,0,0);
  1336. if (Context->PendingUiType != UI_NONE) {
  1337. //
  1338. // We now allow the main thread to continue. Once we do
  1339. // so, the UI parameters that we passed to us are invalid.
  1340. // Cancel the pending UI.
  1341. //
  1342. Context->PendingUiType = UI_NONE;
  1343. Context->lParam = Context->CancelReturnCode;
  1344. SetEvent(Context->hEvent); // wake up main thread (lParam has UI result)
  1345. }
  1346. } else {
  1347. if (Context->PendingUiType != UI_NONE) {
  1348. Context->lParam = pPerformUi(Context,
  1349. (UINT)Context->PendingUiType,
  1350. Context->PendingUiParameters);
  1351. Context->PendingUiType = UI_NONE;
  1352. SetEvent(Context->hEvent); // wake up main thread (lParam has UI result)
  1353. }
  1354. }
  1355. b = TRUE;
  1356. } else {
  1357. b = FALSE;
  1358. }
  1359. break;
  1360. case WMX_KILLDIALOG:
  1361. //
  1362. // Exit unconditionally. Clean up first.
  1363. //
  1364. b = TRUE;
  1365. Context = (PQUEUECONTEXT)GetProp(hdlg, DialogPropName);
  1366. MYASSERT(Context != NULL);
  1367. if(Context->MessageBoxUp) {
  1368. //
  1369. // The user was still interacting with the "are you sure you
  1370. // want to cancel" dialog and the copying finished. So we don't want
  1371. // to nuke the dialog out from under the message box.
  1372. //
  1373. Context->DialogKilled = TRUE;
  1374. break;
  1375. }
  1376. DestroyWindow(Context->ProgressBar);
  1377. EndDialog(hdlg, 0);
  1378. break;
  1379. default:
  1380. //
  1381. // we disable autorun because it confuses the user.
  1382. //
  1383. if (!uQueryCancelAutoPlay) {
  1384. uQueryCancelAutoPlay = RegisterWindowMessage(TEXT("QueryCancelAutoPlay"));
  1385. }
  1386. if (msg == uQueryCancelAutoPlay) {
  1387. SetWindowLongPtr( hdlg, DWLP_MSGRESULT, 1 );
  1388. return 1; // cancel auto-play
  1389. }
  1390. b = FALSE;
  1391. break;
  1392. }
  1393. return(b);
  1394. }
  1395. PVOID
  1396. SetupInitDefaultQueueCallbackEx(
  1397. IN HWND OwnerWindow,
  1398. IN HWND AlternateProgressWindow, OPTIONAL
  1399. IN UINT ProgressMessage,
  1400. IN DWORD Reserved1,
  1401. IN PVOID Reserved2
  1402. )
  1403. {
  1404. PQUEUECONTEXT Context;
  1405. BOOL b;
  1406. Context = MyMalloc(sizeof(QUEUECONTEXT));
  1407. if(Context) {
  1408. ZeroMemory(Context,sizeof(QUEUECONTEXT));
  1409. Context->Signature = QUEUECONTEXT_SIGNATURE;
  1410. Context->hEvent = CreateEvent(NULL,FALSE,FALSE,NULL);
  1411. if (Context->hEvent == NULL) {
  1412. MyFree(Context);
  1413. return NULL;
  1414. }
  1415. Context->OwnerWindow = OwnerWindow;
  1416. Context->MainThreadId = GetCurrentThreadId();
  1417. Context->ProgressMsg = ProgressMessage;
  1418. Context->NoToAllMask = 0;
  1419. //
  1420. // If the caller specified NULL for the alternate progress window, and
  1421. // we're running non-interactively, we want to treat this just as if
  1422. // they'd specified INVALID_HANDLE_VALUE (i.e., suppress all progress
  1423. // UI).
  1424. //
  1425. if((GlobalSetupFlags & PSPGF_NONINTERACTIVE) && !AlternateProgressWindow) {
  1426. Context->AlternateProgressWindow = INVALID_HANDLE_VALUE;
  1427. } else {
  1428. Context->AlternateProgressWindow = AlternateProgressWindow;
  1429. }
  1430. if(SystemParametersInfo(SPI_GETSCREENREADER,0,&b,0) && b) {
  1431. Context->ScreenReader = TRUE;
  1432. } else {
  1433. Context->ScreenReader = FALSE;
  1434. }
  1435. #ifdef PRERELEASE
  1436. //
  1437. // if we're running in gui-mode setup, we suppress all setupapi progress UI
  1438. // and only allow AlternateProgressWindow UI
  1439. //
  1440. if (GuiSetupInProgress
  1441. && (Context->AlternateProgressWindow != (HWND)INVALID_HANDLE_VALUE)
  1442. && !IsWindow(Context->AlternateProgressWindow)) {
  1443. MYASSERT( FALSE && TEXT("SetupInitDefaultQueueCallbackEx() called in gui-setup without INVALID_HANDLE_VALUE, which means UI may be presented. Click yes and retrieve the stack trace to detect errant caller.\n"));
  1444. }
  1445. #endif
  1446. }
  1447. return(Context);
  1448. }
  1449. PVOID
  1450. SetupInitDefaultQueueCallback(
  1451. IN HWND OwnerWindow
  1452. )
  1453. {
  1454. #ifdef PRERELEASE
  1455. if (GuiSetupInProgress) {
  1456. MYASSERT( FALSE && TEXT("SetupInitDefaultQueueCallback() called in gui-setup, which means UI may be presented. Click yes and retrieve the stack trace to detect the errant caller.\n"));
  1457. }
  1458. #endif
  1459. return(SetupInitDefaultQueueCallbackEx(OwnerWindow,NULL,0,0,NULL));
  1460. }
  1461. VOID
  1462. SetupTermDefaultQueueCallback(
  1463. IN PVOID Context
  1464. )
  1465. {
  1466. PQUEUECONTEXT context;
  1467. context = Context;
  1468. try {
  1469. if(context && context->Signature == QUEUECONTEXT_SIGNATURE) {
  1470. if(context->CurrentSourceName) {
  1471. MyFree(context->CurrentSourceName);
  1472. }
  1473. if (context->hEvent) {
  1474. CloseHandle(context->hEvent);
  1475. }
  1476. context->Signature = 0;
  1477. }
  1478. MyFree(Context);
  1479. } except(EXCEPTION_EXECUTE_HANDLER) {
  1480. ;
  1481. }
  1482. }
  1483. #ifdef UNICODE
  1484. //
  1485. // ANSI version
  1486. //
  1487. UINT
  1488. SetupDefaultQueueCallbackA(
  1489. IN PVOID Context,
  1490. IN UINT Notification,
  1491. IN UINT_PTR Param1,
  1492. IN UINT_PTR Param2
  1493. )
  1494. {
  1495. UINT u;
  1496. u = pSetupCallDefaultMsgHandler(
  1497. Context,
  1498. Notification,
  1499. Param1,
  1500. Param2
  1501. );
  1502. return(u);
  1503. }
  1504. #else
  1505. //
  1506. // Unicode stub
  1507. //
  1508. UINT
  1509. SetupDefaultQueueCallbackW(
  1510. IN PVOID Context,
  1511. IN UINT Notification,
  1512. IN UINT_PTR Param1,
  1513. IN UINT_PTR Param2
  1514. )
  1515. {
  1516. UNREFERENCED_PARAMETER(Context);
  1517. UNREFERENCED_PARAMETER(Notification);
  1518. UNREFERENCED_PARAMETER(Param1);
  1519. UNREFERENCED_PARAMETER(Param2);
  1520. SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
  1521. return(0);
  1522. }
  1523. #endif
  1524. UINT
  1525. SetupDefaultQueueCallback(
  1526. IN PVOID Context,
  1527. IN UINT Notification,
  1528. IN UINT_PTR Param1,
  1529. IN UINT_PTR Param2
  1530. )
  1531. {
  1532. UINT rc;
  1533. DWORD err;
  1534. PQUEUECONTEXT context = Context;
  1535. MSG msg;
  1536. VERDLGCONTEXT dialogContext;
  1537. DWORD waitResult;
  1538. //
  1539. // a little sanity check on context
  1540. //
  1541. err = NO_ERROR;
  1542. try {
  1543. if (context == NULL || context->Signature != QUEUECONTEXT_SIGNATURE) {
  1544. err = ERROR_INVALID_PARAMETER;
  1545. }
  1546. } except(EXCEPTION_EXECUTE_HANDLER) {
  1547. err = ERROR_INVALID_PARAMETER;
  1548. }
  1549. if (err != NO_ERROR) {
  1550. return pGetCallbackErrorReturn(Notification,err);
  1551. }
  1552. switch(Notification) {
  1553. case SPFILENOTIFY_STARTQUEUE:
  1554. rc = pNotificationStartQueue(context);
  1555. break;
  1556. case SPFILENOTIFY_ENDQUEUE:
  1557. //
  1558. // Make sure the progress dialog is dead.
  1559. //
  1560. if(context->AlternateProgressWindow) {
  1561. //
  1562. // If we have an alternate progress window, then we'd better not
  1563. // have our own progress dialog, nor should we have a UI thread
  1564. // handle.
  1565. //
  1566. MYASSERT(!context->ProgressDialog);
  1567. MYASSERT(!context->UiThreadHandle);
  1568. } else {
  1569. if(IsWindow(context->ProgressDialog)) {
  1570. //
  1571. // Post a message to the dialog, instructing it to terminate,
  1572. // then wait for it to confirm.
  1573. //
  1574. PostMessage(context->ProgressDialog, WMX_KILLDIALOG, 0, 0);
  1575. }
  1576. //
  1577. // The dialog may have been marked for delete (thus IsWindow() fails),
  1578. // and yet the dialog has not yet been destroyed. Therefore, we always
  1579. // want to wait for the thread message that assures us that everything
  1580. // has been cleaned up in the other thread.
  1581. //
  1582. // Also, before returning we need to make sure that the UI thread is
  1583. // really gone, or else there's a hole where our caller could unload
  1584. // the library and then the UI thread would fault.
  1585. // We have seen this happen in stress.
  1586. //
  1587. while (pWaitForUiResponse(context)) /* nothing */;
  1588. if(context->UiThreadHandle) {
  1589. waitResult = WaitForSingleObject(context->UiThreadHandle,INFINITE);
  1590. MYASSERT(waitResult != WAIT_FAILED);
  1591. CloseHandle(context->UiThreadHandle);
  1592. }
  1593. }
  1594. rc = TRUE; // return value for this notification is actually ignored.
  1595. break;
  1596. case SPFILENOTIFY_STARTSUBQUEUE:
  1597. rc = pNotificationStartEndSubqueue(context,TRUE,Param1,Param2);
  1598. break;
  1599. case SPFILENOTIFY_ENDSUBQUEUE:
  1600. rc = pNotificationStartEndSubqueue(context,FALSE,Param1,0);
  1601. break;
  1602. case SPFILENOTIFY_STARTDELETE:
  1603. case SPFILENOTIFY_STARTRENAME:
  1604. case SPFILENOTIFY_STARTCOPY:
  1605. case SPFILENOTIFY_STARTBACKUP:
  1606. //
  1607. // Update display to indicate the files involved
  1608. // in the operation, unless a screen reader is active.
  1609. //
  1610. if(context->ScreenReader) {
  1611. rc = FILEOP_DOIT;
  1612. } else {
  1613. rc = pNotificationStartOperation(context,(PFILEPATHS)Param1,Param2);
  1614. }
  1615. break;
  1616. case SPFILENOTIFY_ENDDELETE:
  1617. case SPFILENOTIFY_ENDRENAME:
  1618. case SPFILENOTIFY_ENDCOPY:
  1619. case SPFILENOTIFY_ENDBACKUP:
  1620. case SPFILENOTIFY_ENDREGISTRATION:
  1621. if(context->AlternateProgressWindow) {
  1622. //
  1623. // If this is really is an alternate progress window, then 'tick' it.
  1624. // Copy only.
  1625. //
  1626. if((Notification == SPFILENOTIFY_ENDCOPY) &&
  1627. (context->AlternateProgressWindow != INVALID_HANDLE_VALUE)) {
  1628. SendMessage(context->AlternateProgressWindow, context->ProgressMsg, 1, 0);
  1629. }
  1630. } else {
  1631. if(IsWindow(context->ProgressBar)) {
  1632. //
  1633. // Update gas gauge.
  1634. //
  1635. SendMessage(context->ProgressBar,PBM_STEPIT,0,0);
  1636. }
  1637. }
  1638. rc = TRUE; // return value for these notifications is actually ignored.
  1639. break;
  1640. case SPFILENOTIFY_DELETEERROR:
  1641. rc = pNotificationErrorDelete(context,(PFILEPATHS)Param1);
  1642. break;
  1643. case SPFILENOTIFY_RENAMEERROR:
  1644. rc = pNotificationErrorRename(context,(PFILEPATHS)Param1);
  1645. break;
  1646. case SPFILENOTIFY_BACKUPERROR:
  1647. rc = pNotificationErrorBackup(context,(PFILEPATHS)Param1);
  1648. break;
  1649. case SPFILENOTIFY_COPYERROR:
  1650. rc = pNotificationErrorCopy(context,(PFILEPATHS)Param1,(PTSTR)Param2);
  1651. break;
  1652. case SPFILENOTIFY_NEEDMEDIA:
  1653. //
  1654. // Perform prompt.
  1655. //
  1656. rc = pNotificationNeedMedia(context,(PSOURCE_MEDIA)Param1,(PTSTR)Param2);
  1657. break;
  1658. case SPFILENOTIFY_STARTREGISTRATION:
  1659. rc = pNotificationStartRegistration(context,(PSP_REGISTER_CONTROL_STATUS)Param1,(BOOL)Param2);
  1660. break;
  1661. default:
  1662. //
  1663. // The notification is either an unknown ordinal or a version mismatch.
  1664. //
  1665. if(Notification & (SPFILENOTIFY_LANGMISMATCH | SPFILENOTIFY_TARGETNEWER | SPFILENOTIFY_TARGETEXISTS)) {
  1666. //
  1667. // It's one or more of our known version mismatches. First
  1668. // check to see whether No to All has already been specified
  1669. // for the mismatch(es). Turn off the bits in the notification
  1670. // that are set in NoToAllMask; if there are still bits set,
  1671. // we need to notify about this mismatch. If there are no
  1672. // longer any bits set, then don't copy this file.
  1673. //
  1674. Notification &= ~context->NoToAllMask;
  1675. if (Notification != 0) {
  1676. //
  1677. // Notify about this mismatch.
  1678. //
  1679. dialogContext.QueueContext = context;
  1680. dialogContext.Notification = Notification;
  1681. dialogContext.Param1 = Param1;
  1682. dialogContext.Param2 = Param2;
  1683. rc = PostUiMessage (
  1684. context, UI_MISMATCHERROR, DPROMPT_CANCEL, &dialogContext);
  1685. rc = (rc == IDYES);
  1686. } else {
  1687. //
  1688. // No To All has already been specified for this notification type.
  1689. // Skip the file.
  1690. //
  1691. rc = 0;
  1692. }
  1693. } else {
  1694. //
  1695. // Unknown notification. Skip the file.
  1696. //
  1697. rc = 0;
  1698. }
  1699. break;
  1700. }
  1701. return(rc);
  1702. }