source: trunk/dll/notify.c@ 1882

Last change on this file since 1882 was 1877, checked in by Gregg Young, 10 years ago

Remove debug code

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.7 KB
Line 
1/***********************************************************************
2
3 $Id: notify.c 1877 2015-10-11 21:43:27Z gyoung $
4
5 Thread notes window and popup notifications over status line
6
7 Copyright (c) 1993-98 M. Kimes
8 Copyright (c) 2006, 2010 Steven H.Levine
9
10 17 Jul 06 SHL Use Win_Error
11 22 Jul 06 SHL Check more run time errors
12 30 Mar 07 GKY Remove GetPString for window class names
13 06 Aug 07 GKY Reduce DosSleep times (ticket 148)
14 20 Aug 07 GKY Move #pragma alloc_text to end for OpenWatcom compat
15 16 Apr 08 SHL Comment and clean up logic
16 10 Dec 08 SHL Integrate exception handler support
17 11 Jan 09 GKY Replace font names in the string file with global set at compile in init.c
18 07 Feb 09 GKY Allow user to turn off alert and/or error beeps in settings notebook.
19 07 Feb 09 GKY Eliminate Win_Error2 by moving function names to PCSZs used in Win_Error
20 13 Jul 09 SHL Sync with renames
21 16 Jul 09 SHL Stop leaking hptrIcon
22 17 Jan 10 GKY Changes to get working with Watcom 1.9 Beta (1/16/10). Mostly cast CHAR CONSTANT * as CHAR *.
23 12 Aug 12 GKY Add support for changing PresParams in the notify status window
24
25***********************************************************************/
26
27#include <stdlib.h>
28#include <string.h>
29#include <ctype.h>
30#include <stddef.h> // _threadid
31
32#define INCL_DOS
33#define INCL_WIN
34#define INCL_GPI
35#define INCL_LONGLONG // dircnrs.h
36
37#include "fm3dll.h"
38#include "fm3dll2.h" // #define's for UM_*, control id's, etc.
39#include "collect.h" // Data declaration(s)
40#include "grep.h" // Data declaration(s)
41#include "notebook.h" // Data declaration(s)
42#include "init.h" // Data declaration(s)
43#include "fm3dlg.h"
44#include "fm3str.h"
45#include "errutil.h" // Dos_Error...
46#include "strutil.h" // GetPString
47#include "notify.h"
48#include "presparm.h" // SetPresParams
49#include "mainwnd.h" // Data declaration(s)
50#include "wrappers.h" // xmalloc
51#include "misc.h" // PostMsg
52#include "fortify.h"
53#include "excputil.h" // xbeginthread
54
55#pragma data_seg(DATA1)
56
57// Data definitions
58static PSZ pszSrcFile = __FILE__;
59static volatile HWND hwndNotify; // 16 Apr 08 SHL
60static volatile PCSZ pszCachedNote; // 16 Jul 09 SHL
61
62#pragma data_seg(GLOBAL1)
63BOOL fThreadNotes;
64
65static VOID StartNotes(PCSZ pszNote);
66
67/**
68 * Notification message window procedure
69 * Displays timed message over status line
70 */
71
72MRESULT EXPENTRY NotifyWndProc(HWND hwnd, ULONG msg, MPARAM mp1, MPARAM mp2)
73{
74 static ULONG showing = 0;
75
76 switch (msg) {
77 case WM_CREATE:
78 showing++;
79 {
80 MRESULT rc = PFNWPStatic(hwnd, msg, mp1, mp2);
81
82 if (!WinStartTimer(WinQueryAnchorBlock(hwnd), hwnd, ID_NOTIFY_TIMER, 5000)) {
83 Win_Error(hwnd, hwnd, pszSrcFile, __LINE__, "WinStartTimer");
84 WinDestroyWindow(hwnd);
85 }
86 else {
87 if (hwndMain) {
88 if (hwndStatus)
89 WinShowWindow(hwndStatus, FALSE);
90 if (hwndStatus2)
91 WinShowWindow(hwndStatus2, FALSE);
92 }
93 PostMsg(hwnd, UM_SETUP, MPVOID, MPVOID);
94 }
95 return rc;
96 }
97
98 case UM_SETUP:
99 WinSetWindowPos(hwnd, HWND_TOP, 0, 0, 0, 0, SWP_ZORDER | SWP_SHOW);
100 WinInvalidateRect(hwnd, NULL, FALSE);
101 return 0;
102
103 case WM_SETFOCUS:
104 if (mp2)
105 PostMsg(hwnd, UM_REPLACEFOCUS, mp1, MPVOID);
106 break;
107
108 case UM_REPLACEFOCUS:
109 if (mp1)
110 WinSetFocus(HWND_DESKTOP, (HWND) mp1);
111 PostMsg(hwnd, UM_SETUP, MPVOID, MPVOID);
112 return 0;
113
114 case WM_PAINT:
115 {
116 SWP swp;
117 POINTL ptl;
118
119 MRESULT mr = PFNWPStatic(hwnd, msg, mp1, mp2);
120 HPS hps = WinGetPS(hwnd);
121 if (hps) {
122 if (WinQueryWindowPos(hwnd, &swp)) {
123 ptl.x = 0;
124 ptl.y = 0;
125 GpiMove(hps, &ptl);
126 GpiSetColor(hps, CLR_RED);
127 ptl.x = swp.cx - 1;
128 ptl.y = swp.cy - 1;
129 GpiBox(hps, DRO_OUTLINE, &ptl, 2, 2);
130 }
131 WinReleasePS(hwnd);
132 }
133 return mr;
134 }
135
136 case WM_BUTTON1DOWN:
137 case WM_BUTTON2DOWN:
138 case WM_BUTTON3DOWN:
139 case WM_TIMER:
140 case WM_CLOSE:
141 WinStopTimer(WinQueryAnchorBlock(hwnd), hwnd, ID_NOTIFY_TIMER);
142 WinDestroyWindow(hwnd);
143 return 0;
144
145 case WM_DESTROY:
146 showing--;
147 if (!showing && hwndMain) {
148 if (hwndStatus)
149 WinShowWindow(hwndStatus, TRUE);
150 if (hwndStatus2)
151 WinShowWindow(hwndStatus2, TRUE);
152 }
153 break;
154 }
155 return PFNWPStatic(hwnd, msg, mp1, mp2);
156}
157
158/**
159 * Process UM_NOTIFY message to display timed message over status line
160 */
161
162HWND DoNotify(PCSZ str)
163{
164 char *p;
165 HWND hwnd = (HWND)0, hwndP;
166 LONG x, y, cx, cy;
167 SWP swp, swpS, swpS2;
168 static USHORT id = NOTE_FRAME;
169
170 if (str && *str) {
171 // figure out what size the window should be and where it should be
172 hwndP = hwndMain ? WinQueryWindow(hwndMain, QW_PARENT) : HWND_DESKTOP;
173 WinQueryWindowPos(hwndP, &swp);
174 if (hwndStatus)
175 WinQueryWindowPos(hwndStatus, &swpS);
176 if (hwndStatus2)
177 WinQueryWindowPos(hwndStatus2, &swpS2);
178 x = hwndMain ? (hwndStatus ? swpS.x - 1 :
179 WinQuerySysValue(HWND_DESKTOP, SV_CXSIZEBORDER)) : 0;
180 y = hwndMain ? (hwndStatus ? swpS.y - 1 :
181 WinQuerySysValue(HWND_DESKTOP, SV_CYSIZEBORDER)) : 0;
182 if (hwndMain && hwndStatus) {
183 if (hwndStatus2)
184 cx = swpS2.cx + swpS.cx + 8;
185 else
186 cx = swpS.cx + 6;
187 }
188 else
189 cx = (swp.cx - ((x * 2) + 4));
190 cy = hwndMain ? (hwndStatus ? swpS.cy + 2 : 28) : 28;
191
192 // pretty-up the note by putting on a leading space
193 if (*str != ' ') {
194 p = xmalloc(strlen(str) + 2, pszSrcFile, __LINE__);
195 if (!p)
196 p = (PSZ)str;
197 else {
198 strcpy(p + 1, str);
199 *p = ' ';
200 }
201 }
202 else
203 p = (PSZ)str;
204
205 hwnd = WinCreateWindow(hwndP,
206 (CHAR *) WC_ERRORWND,
207 p,
208 SS_TEXT | DT_LEFT | DT_VCENTER | WS_VISIBLE,
209 x, y, cx, cy, hwndP, HWND_TOP, id++, NULL, NULL);
210 if (!hwnd)
211 Win_Error(hwndP, hwndP, pszSrcFile, __LINE__,
212 PCSZ_WINCREATEWINDOW);
213
214 if (p != str)
215 free(p);
216 if (id > NOTE_MAX)
217 id = NOTE_FRAME; // Wrap
218 }
219
220 AddNote(str); // Add thread notes window
221
222 return hwnd;
223}
224
225/**
226 * Display timed notification window over status line
227 */
228
229HWND Notify(PCSZ str)
230{
231 return (HWND)WinSendMsg(MainObjectHwnd, UM_NOTIFY, MPFROMP(str), MPVOID);
232}
233
234/**
235 * Notify on error
236 * Format message and pass to Notify
237 */
238
239VOID NotifyError(PCSZ filename, APIRET status)
240{
241 CHAR errortext[512];
242
243 if (!filename)
244 return;
245
246 sprintf(errortext, GetPString(IDS_ERRORACCESSTEXT), status, filename);
247 if (toupper(*filename) > 'B') {
248 if (status == 21)
249 strcat(errortext, GetPString(IDS_EMPTYREMOVETEXT));
250 else if (status == 27)
251 strcat(errortext, GetPString(IDS_CDMUSICTEXT));
252 else if (status == 19)
253 strcat(errortext, GetPString(IDS_WRITEPROTECTTEXT));
254 else if (status == 108)
255 strcat(errortext, GetPString(IDS_DRIVELOCKEDTEXT));
256 }
257 else {
258 if (status == 21)
259 strcat(errortext, GetPString(IDS_EMPTYFLOPPYTEXT));
260 else if (status == 27)
261 strcat(errortext, GetPString(IDS_UNFORMATEDTEXT));
262 else if (status == 107)
263 sprintf(&errortext[strlen(errortext)],
264 GetPString(IDS_PHANTOMTEXT), toupper(*filename));
265 else if (status == 19)
266 strcat(errortext, GetPString(IDS_DISKWRITEPROTEXTTEXT));
267 else if (status == 108)
268 strcat(errortext, GetPString(IDS_DISKLOCKEDTEXT));
269 }
270 if (!fErrorBeepOff) {
271 DosBeep(250, 10);
272 DosBeep(500, 10);
273 DosBeep(250, 10);
274 DosBeep(500, 10);
275 }
276 Notify(errortext);
277}
278
279/**
280 * Thread notes window dialog procedure
281 */
282
283MRESULT EXPENTRY NoteWndProc(HWND hwnd, ULONG msg, MPARAM mp1, MPARAM mp2)
284{
285 static HPOINTER hptrIcon;
286
287 switch (msg) {
288 case WM_INITDLG:
289 if (hwndNotify != (HWND)0) {
290 // Already have notes dialog - pass message on
291 if (mp2) {
292 AddNote((PCSZ)mp2); // 16 Jul 09 SHL was direct LM_INSERTITEM
293 xfree((PSZ)mp2, pszSrcFile, __LINE__);
294 }
295 WinDismissDlg(hwnd, 0);
296 break;
297 }
298 hwndNotify = hwnd;
299 {
300 USHORT ids[] = {NOTE_FRAME, NOTE_LISTBOX, 0};
301 UINT x;
302 CHAR s[24];
303
304 for (x = 0; ids[x]; x++) {
305 sprintf(s, "%s%i", PCSZ_NOTIFYWND,ids[x]);
306 RestorePresParams(WinWindowFromID(hwnd, ids[x]), s);
307 }
308 }
309 // Remember showing for restart
310 fThreadNotes = TRUE;
311 PrfWriteProfileData(fmprof,
312 FM3Str,
313 "ThreadNotes",
314 &fThreadNotes,
315 sizeof(BOOL));
316 fThreadNotes = FALSE; // Optimize shutdown
317
318 // 16 Jul 09 SHL Added
319 if (pszCachedNote) {
320 PCSZ p = pszCachedNote;
321 pszCachedNote = NULL;
322 AddNote(p);
323 xfree((PSZ)p, pszSrcFile, __LINE__);
324 }
325
326 if (mp2) {
327 AddNote((PCSZ)mp2); // 16 Jul 09 SHL was direct LM_INSERTITEM
328 xfree((PSZ)mp2, pszSrcFile, __LINE__);
329 }
330
331 // Grab focus
332 PostMsg(hwnd,
333 UM_FOCUSME,
334 MPFROMLONG(WinQueryActiveWindow(HWND_DESKTOP)),
335 MPVOID);
336
337 hptrIcon = WinLoadPointer(HWND_DESKTOP, FM3ModHandle, NOTE_FRAME);
338 if (hptrIcon)
339 WinDefDlgProc(hwnd, WM_SETICON, MPFROMLONG(hptrIcon), MPFROMLONG(0L));
340 break;
341
342 case UM_FOCUSME:
343 {
344 ULONG size = sizeof(SWP),
345 fl = SWP_ZORDER | SWP_FOCUSDEACTIVATE | SWP_SHOW;
346 SWP swp;
347
348 if (PrfQueryProfileData(fmprof,
349 FM3Str, "NoteWndSwp", (PVOID) & swp, &size)) {
350 if (swp.fl & (SWP_HIDE | SWP_MINIMIZE)) {
351 fl |= SWP_MINIMIZE;
352 fl &= (~SWP_SHOW);
353 }
354 else
355 fl |= (SWP_MOVE | SWP_SIZE);
356 }
357 WinSetWindowPos(hwnd, HWND_BOTTOM, swp.x, swp.y, swp.cx, swp.cy, fl);
358 if (fl & SWP_MINIMIZE) {
359 WinSetWindowUShort(hwnd, QWS_XRESTORE, (USHORT) swp.x);
360 WinSetWindowUShort(hwnd, QWS_CXRESTORE, (USHORT) swp.cx);
361 WinSetWindowUShort(hwnd, QWS_YRESTORE, (USHORT) swp.y);
362 WinSetWindowUShort(hwnd, QWS_CYRESTORE, (USHORT) swp.cy);
363 }
364 }
365 if (mp1)
366 WinSetActiveWindow(HWND_DESKTOP, (HWND) mp1);
367 return 0;
368
369 case UM_STRETCH:
370 {
371 SWP swp;
372 LONG titl, szbx, szby;
373
374 WinQueryWindowPos(hwnd, &swp);
375 if (!(swp.fl & (SWP_MINIMIZE | SWP_HIDE))) {
376 szbx = SysVal(SV_CXSIZEBORDER);
377 szby = SysVal(SV_CYSIZEBORDER);
378 titl = SysVal(SV_CYTITLEBAR);
379 WinSetWindowPos(WinWindowFromID(hwnd, NOTE_LISTBOX),
380 HWND_TOP,
381 szbx,
382 szby,
383 swp.cx - (szbx * 2L),
384 (swp.cy - titl) - (szby * 2L), SWP_MOVE | SWP_SIZE);
385 }
386 if (!(swp.fl & SWP_MAXIMIZE)) {
387 if (swp.fl & (SWP_MINIMIZE | SWP_HIDE)) {
388 swp.x = WinQueryWindowUShort(hwnd, QWS_XRESTORE);
389 swp.y = WinQueryWindowUShort(hwnd, QWS_YRESTORE);
390 swp.cx = WinQueryWindowUShort(hwnd, QWS_CXRESTORE);
391 swp.cy = WinQueryWindowUShort(hwnd, QWS_CYRESTORE);
392 }
393 PrfWriteProfileData(fmprof,
394 FM3Str, "NoteWndSwp", (PVOID) & swp, sizeof(SWP));
395 }
396 }
397 return 0;
398
399 case WM_ADJUSTWINDOWPOS:
400 PostMsg(hwnd, UM_STRETCH, MPVOID, MPVOID);
401 break;
402
403 case UM_CONTAINER_FILLED:
404 {
405 SHORT y;
406 SHORT x = (SHORT)WinSendDlgItemMsg(hwnd,
407 NOTE_LISTBOX,
408 LM_QUERYITEMCOUNT, MPVOID, MPVOID);
409 if (x > 60) {
410 for (y = 0; y < x - 50; y++) {
411 WinSendDlgItemMsg(hwnd,
412 NOTE_LISTBOX,
413 LM_DELETEITEM, MPFROMSHORT(y), MPVOID);
414 }
415 }
416 }
417 return 0;
418
419 case UM_NOTIFY:
420 {
421 SHORT x = (SHORT) WinSendDlgItemMsg(hwnd,
422 NOTE_LISTBOX,
423 LM_QUERYITEMCOUNT, MPVOID, MPVOID);
424 if (x > 0)
425 WinSendDlgItemMsg(hwnd,
426 NOTE_LISTBOX,
427 LM_SETTOPINDEX, MPFROMSHORT(x), MPVOID);
428 }
429 return 0;
430
431 case UM_SHOWME:
432 WinSetWindowPos(hwnd,
433 HWND_TOP,
434 0,
435 0,
436 0, 0, SWP_SHOW | SWP_RESTORE | SWP_ZORDER | SWP_ACTIVATE);
437 return 0;
438
439 case WM_COMMAND:
440 return 0;
441
442 case WM_CLOSE:
443 if (pszCachedNote)
444 Runtime_Error(pszSrcFile, __LINE__, "pszCachedNote %p unexpected", pszCachedNote);
445 else {
446 // Cache last item for next open
447 SHORT ndx = (SHORT)WinSendDlgItemMsg(hwnd, NOTE_LISTBOX,
448 LM_QUERYITEMCOUNT, MPVOID, MPVOID);
449 if (ndx != LIT_NONE) {
450 SHORT len;
451 ndx--;
452 len = (SHORT)WinSendDlgItemMsg(hwnd, NOTE_LISTBOX,
453 LM_QUERYITEMTEXTLENGTH,
454 MPFROMSHORT(ndx), MPVOID);
455 if (len != LIT_ERROR) {
456 PSZ p;
457 len++;
458 p = xmalloc(len, pszSrcFile, __LINE__);
459 if (p) {
460 SHORT len2 = (SHORT)WinSendDlgItemMsg(hwnd, NOTE_LISTBOX,
461 LM_QUERYITEMTEXT,
462 MPFROM2SHORT(ndx, len), MPFROMP(p));
463 len--;
464 if (len2 != len) {
465 xfree((PSZ)p, pszSrcFile, __LINE__);
466 }
467 else
468 pszCachedNote = p;
469 }
470 }
471 }
472 }
473 WinDismissDlg(hwnd, 0);
474 return 0;
475
476 case WM_DESTROY:
477 if (hwndNotify == hwnd) {
478 USHORT ids[] = {NOTE_FRAME, NOTE_LISTBOX, 0};
479 UINT x;
480 CHAR s[24];
481
482 for (x = 0; ids[x]; x++) {
483 sprintf(s, "%s%i", PCSZ_NOTIFYWND,ids[x]);
484 SavePresParams(WinWindowFromID(hwnd, ids[x]), s);
485 }
486 fThreadNotes = FALSE; // Remember not open
487 PrfWriteProfileData(fmprof,
488 FM3Str, "ThreadNotes", &fThreadNotes, sizeof(BOOL));
489 hwndNotify = (HWND)0;
490 }
491 if (hptrIcon) {
492 WinDestroyPointer(hptrIcon);
493 hptrIcon = (HPOINTER)0; // 16 Jul 09 SHL
494 }
495 if (!PostMsg((HWND)0, WM_QUIT, MPVOID, MPVOID))
496 WinSendMsg((HWND)0, WM_QUIT, MPVOID, MPVOID);
497 break;
498 }
499 return WinDefDlgProc(hwnd, msg, mp1, mp2);
500}
501
502/**
503 * Thread notes dialog window thread
504 */
505
506static VOID NoteThread(VOID *args)
507{
508 HAB hab = WinInitialize(0);
509# ifdef FORTIFY
510 Fortify_EnterScope();
511# endif
512 if (hab) {
513 HMQ hmq = WinCreateMsgQueue(hab, 0);
514 if (hmq) {
515 if (!hwndNotify)
516 WinDlgBox(HWND_DESKTOP,
517 HWND_DESKTOP,
518 NoteWndProc, FM3ModHandle, NOTE_FRAME, args);
519 WinDestroyMsgQueue(hmq);
520 }
521 WinTerminate(hab);
522 }
523# ifdef FORTIFY
524 Fortify_LeaveScope();
525# endif
526}
527
528/**
529 * Start thread notes dialog window thread
530 */
531
532static VOID StartNotes(PCSZ note)
533{
534 if (!hwndNotify) {
535 if (xbeginthread(NoteThread,
536 65536,
537 (VOID*)note,
538 pszSrcFile,
539 __LINE__) != -1)
540 {
541 USHORT i;
542 for (i = 0; !hwndNotify && i < 10; i++)
543 DosSleep(10);
544 if (!hwndNotify)
545 Runtime_Error(pszSrcFile, __LINE__, "Can not create Notify window");
546 }
547 }
548}
549
550/**
551 * Add note to thread notes window or popup status window
552 * Open window if was open and first messages after restart
553 * Cache last note until window opened
554 */
555
556VOID AddNote(PCSZ note)
557{
558 PSZ s;
559 PCSZ p;
560 BOOL once = FALSE;
561
562 // Cache last note until window opened
563 // 16 Jul 09 SHL fixme to avoid FORTIFY complaints
564 if (!fThreadNotes && !hwndNotify && note) {
565 p = note + strspn(note, " \t"); // Skip leading white
566 if (*p) {
567 if (pszCachedNote)
568 xfree((PSZ)pszCachedNote, pszSrcFile, __LINE__);
569 pszCachedNote = xstrdup(p, pszSrcFile, __LINE__);
570 }
571 return;
572 }
573
574 if ((fThreadNotes || hwndNotify) && note) {
575 p = note + strspn(note, " \t"); // Skip leading white
576 if (*p) {
577 // If have cached note, output it first
578 if (pszCachedNote) {
579 PCSZ psz = pszCachedNote;
580 pszCachedNote = NULL;
581 AddNote(psz);
582 free((VOID*)psz);
583 }
584 if (!hwndNotify) {
585 fThreadNotes = FALSE; // 16 Jul 09 SHL fixme to be gone?
586 StartNotes(NULL);
587 }
588 if (hwndNotify) {
589 s = xmalloc(strlen(p) + 14, pszSrcFile, __LINE__);
590 if (s) {
591 sprintf(s, "%08lx %s", _threadid, p);
592 while (!once) {
593 if ((SHORT) WinSendDlgItemMsg(hwndNotify,
594 NOTE_LISTBOX,
595 LM_INSERTITEM,
596 MPFROM2SHORT(LIT_END, 0),
597 MPFROMP(s)) >= 0) {
598 PostMsg(hwndNotify, UM_NOTIFY, MPVOID, MPVOID);
599 break;
600 }
601 PostMsg(hwndNotify, UM_CONTAINER_FILLED, MPVOID, MPVOID);
602 once = TRUE;
603 }
604 free(s);
605 }
606 }
607 }
608 }
609}
610
611/**
612 * Close thread notes window
613 */
614
615VOID EndNote(VOID)
616{
617 if (hwndNotify)
618 if (!PostMsg(hwndNotify, WM_CLOSE, MPVOID, MPVOID))
619 WinSendMsg(hwndNotify, WM_CLOSE, MPVOID, MPVOID);
620}
621
622/**
623 * Pop up thread notes window
624 */
625
626VOID ShowNote(VOID)
627{
628 if (!hwndNotify)
629 StartNotes(NULL);
630 if (hwndNotify)
631 PostMsg(hwndNotify, UM_SHOWME, MPVOID, MPVOID);
632}
633
634/**
635 * Hide thread notes window
636 */
637
638VOID HideNote(VOID)
639{
640 if (hwndNotify)
641 WinSetWindowPos(hwndNotify,
642 HWND_BOTTOM,
643 0,
644 0, 0, 0, SWP_MINIMIZE | SWP_ZORDER | SWP_FOCUSDEACTIVATE);
645}
646
647#pragma alloc_text(NOTIFY,Notify,NotifyWndProc,StartNotify,NotifyError)
648#pragma alloc_text(NOTIFY2,AddNote,NoteThread,NoteWndProc)
649#pragma alloc_text(NOTIFY3,StartNotes,EndNote,HideNote,ShowNote)
Note: See TracBrowser for help on using the repository browser.