xref: /reactos/base/applications/notepad/main.c (revision 463784c5)
1 /*
2  *  Notepad
3  *
4  *  Copyright 2000 Mike McCormack <Mike_McCormack@looksmart.com.au>
5  *  Copyright 1997,98 Marcel Baur <mbaur@g26.ethz.ch>
6  *  Copyright 2002 Sylvain Petreolle <spetreolle@yahoo.fr>
7  *  Copyright 2002 Andriy Palamarchuk
8  *  Copyright 2020 Katayama Hirofumi MZ
9  *
10  * This library is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU Lesser General Public
12  * License as published by the Free Software Foundation; either
13  * version 2.1 of the License, or (at your option) any later version.
14  *
15  * This library is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18  * Lesser General Public License for more details.
19  *
20  * You should have received a copy of the GNU Lesser General Public
21  * License along with this library; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
23  *
24  */
25 
26 #include "notepad.h"
27 
28 #include <shlobj.h>
29 #include <strsafe.h>
30 
31 NOTEPAD_GLOBALS Globals;
32 static ATOM aFINDMSGSTRING;
33 
34 VOID NOTEPAD_EnableSearchMenu()
35 {
36     EnableMenuItem(Globals.hMenu, CMD_SEARCH,
37                    MF_BYCOMMAND | ((GetWindowTextLength(Globals.hEdit) == 0) ? MF_DISABLED | MF_GRAYED : MF_ENABLED));
38     EnableMenuItem(Globals.hMenu, CMD_SEARCH_NEXT,
39                    MF_BYCOMMAND | ((GetWindowTextLength(Globals.hEdit) == 0) ? MF_DISABLED | MF_GRAYED : MF_ENABLED));
40 }
41 
42 /***********************************************************************
43  *
44  *           SetFileName
45  *
46  *  Sets Global File Name.
47  */
48 VOID SetFileName(LPCTSTR szFileName)
49 {
50     StringCchCopy(Globals.szFileName, ARRAY_SIZE(Globals.szFileName), szFileName);
51     Globals.szFileTitle[0] = 0;
52     GetFileTitle(szFileName, Globals.szFileTitle, ARRAY_SIZE(Globals.szFileTitle));
53 
54     if (szFileName && szFileName[0])
55         SHAddToRecentDocs(SHARD_PATHW, szFileName);
56 }
57 
58 /***********************************************************************
59  *
60  *           NOTEPAD_MenuCommand
61  *
62  *  All handling of main menu events
63  */
64 static int NOTEPAD_MenuCommand(WPARAM wParam)
65 {
66     switch (wParam)
67     {
68     case CMD_NEW:        DIALOG_FileNew(); break;
69     case CMD_NEW_WINDOW: DIALOG_FileNewWindow(); break;
70     case CMD_OPEN:       DIALOG_FileOpen(); break;
71     case CMD_SAVE:       DIALOG_FileSave(); break;
72     case CMD_SAVE_AS:    DIALOG_FileSaveAs(); break;
73     case CMD_PRINT:      DIALOG_FilePrint(); break;
74     case CMD_PAGE_SETUP: DIALOG_FilePageSetup(); break;
75     case CMD_EXIT:       DIALOG_FileExit(); break;
76 
77     case CMD_UNDO:       DIALOG_EditUndo(); break;
78     case CMD_CUT:        DIALOG_EditCut(); break;
79     case CMD_COPY:       DIALOG_EditCopy(); break;
80     case CMD_PASTE:      DIALOG_EditPaste(); break;
81     case CMD_DELETE:     DIALOG_EditDelete(); break;
82     case CMD_SELECT_ALL: DIALOG_EditSelectAll(); break;
83     case CMD_TIME_DATE:  DIALOG_EditTimeDate(); break;
84 
85     case CMD_SEARCH:      DIALOG_Search(); break;
86     case CMD_SEARCH_NEXT: DIALOG_SearchNext(); break;
87     case CMD_REPLACE:     DIALOG_Replace(); break;
88     case CMD_GOTO:        DIALOG_GoTo(); break;
89 
90     case CMD_WRAP: DIALOG_EditWrap(); break;
91     case CMD_FONT: DIALOG_SelectFont(); break;
92 
93     case CMD_STATUSBAR: DIALOG_ViewStatusBar(); break;
94 
95     case CMD_HELP_CONTENTS: DIALOG_HelpContents(); break;
96     case CMD_HELP_ABOUT_NOTEPAD: DIALOG_HelpAboutNotepad(); break;
97 
98     default:
99         break;
100     }
101     return 0;
102 }
103 
104 /***********************************************************************
105  *
106  *           NOTEPAD_FindTextAt
107  */
108 
109 static BOOL
110 NOTEPAD_FindTextAt(FINDREPLACE *pFindReplace, LPCTSTR pszText, int iTextLength, DWORD dwPosition)
111 {
112     BOOL bMatches;
113     size_t iTargetLength;
114 
115     if ((!pFindReplace) || (!pszText))
116     {
117         return FALSE;
118     }
119 
120     iTargetLength = _tcslen(pFindReplace->lpstrFindWhat);
121 
122     /* Make proper comparison */
123     if (pFindReplace->Flags & FR_MATCHCASE)
124         bMatches = !_tcsncmp(&pszText[dwPosition], pFindReplace->lpstrFindWhat, iTargetLength);
125     else
126         bMatches = !_tcsnicmp(&pszText[dwPosition], pFindReplace->lpstrFindWhat, iTargetLength);
127 
128     if (bMatches && pFindReplace->Flags & FR_WHOLEWORD)
129     {
130         if ((dwPosition > 0) && !_istspace(pszText[dwPosition-1]))
131             bMatches = FALSE;
132         if ((dwPosition < (DWORD) iTextLength - 1) && !_istspace(pszText[dwPosition+1]))
133             bMatches = FALSE;
134     }
135 
136     return bMatches;
137 }
138 
139 /***********************************************************************
140  *
141  *           NOTEPAD_FindNext
142  */
143 
144 BOOL NOTEPAD_FindNext(FINDREPLACE *pFindReplace, BOOL bReplace, BOOL bShowAlert)
145 {
146     int iTextLength, iTargetLength;
147     size_t iAdjustment = 0;
148     LPTSTR pszText = NULL;
149     DWORD dwPosition, dwBegin, dwEnd;
150     BOOL bMatches = FALSE;
151     TCHAR szResource[128], szText[128];
152     BOOL bSuccess;
153 
154     iTargetLength = (int) _tcslen(pFindReplace->lpstrFindWhat);
155 
156     /* Retrieve the window text */
157     iTextLength = GetWindowTextLength(Globals.hEdit);
158     if (iTextLength > 0)
159     {
160         pszText = (LPTSTR) HeapAlloc(GetProcessHeap(), 0, (iTextLength + 1) * sizeof(TCHAR));
161         if (!pszText)
162             return FALSE;
163 
164         GetWindowText(Globals.hEdit, pszText, iTextLength + 1);
165     }
166 
167     SendMessage(Globals.hEdit, EM_GETSEL, (WPARAM) &dwBegin, (LPARAM) &dwEnd);
168     if (bReplace && ((dwEnd - dwBegin) == (DWORD) iTargetLength))
169     {
170         if (NOTEPAD_FindTextAt(pFindReplace, pszText, iTextLength, dwBegin))
171         {
172             SendMessage(Globals.hEdit, EM_REPLACESEL, TRUE, (LPARAM) pFindReplace->lpstrReplaceWith);
173             iAdjustment = _tcslen(pFindReplace->lpstrReplaceWith) - (dwEnd - dwBegin);
174         }
175     }
176 
177     if (pFindReplace->Flags & FR_DOWN)
178     {
179         /* Find Down */
180         dwPosition = dwEnd;
181         while(dwPosition < (DWORD) iTextLength)
182         {
183             bMatches = NOTEPAD_FindTextAt(pFindReplace, pszText, iTextLength, dwPosition);
184             if (bMatches)
185                 break;
186             dwPosition++;
187         }
188     }
189     else
190     {
191         /* Find Up */
192         dwPosition = dwBegin;
193         while(dwPosition > 0)
194         {
195             dwPosition--;
196             bMatches = NOTEPAD_FindTextAt(pFindReplace, pszText, iTextLength, dwPosition);
197             if (bMatches)
198                 break;
199         }
200     }
201 
202     if (bMatches)
203     {
204         /* Found target */
205         if (dwPosition > dwBegin)
206             dwPosition += (DWORD) iAdjustment;
207         SendMessage(Globals.hEdit, EM_SETSEL, dwPosition, dwPosition + iTargetLength);
208         SendMessage(Globals.hEdit, EM_SCROLLCARET, 0, 0);
209         bSuccess = TRUE;
210     }
211     else
212     {
213         /* Can't find target */
214         if (bShowAlert)
215         {
216             LoadString(Globals.hInstance, STRING_CANNOTFIND, szResource, ARRAY_SIZE(szResource));
217             _sntprintf(szText, ARRAY_SIZE(szText), szResource, pFindReplace->lpstrFindWhat);
218             LoadString(Globals.hInstance, STRING_NOTEPAD, szResource, ARRAY_SIZE(szResource));
219             MessageBox(Globals.hFindReplaceDlg, szText, szResource, MB_OK);
220         }
221         bSuccess = FALSE;
222     }
223 
224     if (pszText)
225         HeapFree(GetProcessHeap(), 0, pszText);
226     return bSuccess;
227 }
228 
229 /***********************************************************************
230  *
231  *           NOTEPAD_ReplaceAll
232  */
233 
234 static VOID NOTEPAD_ReplaceAll(FINDREPLACE *pFindReplace)
235 {
236     BOOL bShowAlert = TRUE;
237 
238     SendMessage(Globals.hEdit, EM_SETSEL, 0, 0);
239 
240     while (NOTEPAD_FindNext(pFindReplace, TRUE, bShowAlert))
241     {
242         bShowAlert = FALSE;
243     }
244 }
245 
246 /***********************************************************************
247  *
248  *           NOTEPAD_FindTerm
249  */
250 
251 static VOID NOTEPAD_FindTerm(VOID)
252 {
253     Globals.hFindReplaceDlg = NULL;
254 }
255 
256 /***********************************************************************
257  * Data Initialization
258  */
259 static VOID NOTEPAD_InitData(VOID)
260 {
261     LPTSTR p = Globals.szFilter;
262     static const TCHAR txt_files[] = _T("*.txt");
263     static const TCHAR all_files[] = _T("*.*");
264 
265     p += LoadString(Globals.hInstance, STRING_TEXT_FILES_TXT, p, MAX_STRING_LEN) + 1;
266     _tcscpy(p, txt_files);
267     p += ARRAY_SIZE(txt_files);
268 
269     p += LoadString(Globals.hInstance, STRING_ALL_FILES, p, MAX_STRING_LEN) + 1;
270     _tcscpy(p, all_files);
271     p += ARRAY_SIZE(all_files);
272     *p = '\0';
273     Globals.find.lpstrFindWhat = NULL;
274 
275     Globals.hDevMode = NULL;
276     Globals.hDevNames = NULL;
277 }
278 
279 /***********************************************************************
280  * Enable/disable items on the menu based on control state
281  */
282 static VOID NOTEPAD_InitMenuPopup(HMENU menu, LPARAM index)
283 {
284     int enable;
285 
286     UNREFERENCED_PARAMETER(index);
287 
288     CheckMenuItem(GetMenu(Globals.hMainWnd), CMD_WRAP,
289         MF_BYCOMMAND | (Globals.bWrapLongLines ? MF_CHECKED : MF_UNCHECKED));
290     if (!Globals.bWrapLongLines)
291     {
292         CheckMenuItem(GetMenu(Globals.hMainWnd), CMD_STATUSBAR,
293             MF_BYCOMMAND | (Globals.bShowStatusBar ? MF_CHECKED : MF_UNCHECKED));
294     }
295     EnableMenuItem(menu, CMD_UNDO,
296         SendMessage(Globals.hEdit, EM_CANUNDO, 0, 0) ? MF_ENABLED : MF_GRAYED);
297     EnableMenuItem(menu, CMD_PASTE,
298         IsClipboardFormatAvailable(CF_TEXT) ? MF_ENABLED : MF_GRAYED);
299     enable = (int) SendMessage(Globals.hEdit, EM_GETSEL, 0, 0);
300     enable = (HIWORD(enable) == LOWORD(enable)) ? MF_GRAYED : MF_ENABLED;
301     EnableMenuItem(menu, CMD_CUT, enable);
302     EnableMenuItem(menu, CMD_COPY, enable);
303     EnableMenuItem(menu, CMD_DELETE, enable);
304 
305     EnableMenuItem(menu, CMD_SELECT_ALL,
306         GetWindowTextLength(Globals.hEdit) ? MF_ENABLED : MF_GRAYED);
307     DrawMenuBar(Globals.hMainWnd);
308 }
309 
310 LRESULT CALLBACK EDIT_WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
311 {
312     switch (msg)
313     {
314         case WM_KEYDOWN:
315         case WM_KEYUP:
316         {
317             switch (wParam)
318             {
319                 case VK_UP:
320                 case VK_DOWN:
321                 case VK_LEFT:
322                 case VK_RIGHT:
323                     DIALOG_StatusBarUpdateCaretPos();
324                     break;
325                 default:
326                 {
327                     UpdateWindowCaption(FALSE);
328                     break;
329                 }
330             }
331         }
332         case WM_LBUTTONUP:
333         {
334             DIALOG_StatusBarUpdateCaretPos();
335             break;
336         }
337     }
338     return CallWindowProc( Globals.EditProc, hWnd, msg, wParam, lParam);
339 }
340 
341 /***********************************************************************
342  *
343  *           NOTEPAD_WndProc
344  */
345 static LRESULT
346 WINAPI
347 NOTEPAD_WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
348 {
349     switch (msg)
350     {
351 
352     case WM_CREATE:
353         Globals.hMenu = GetMenu(hWnd);
354 
355         // For now, the "Help" dialog is disabled due to the lack of HTML Help support
356         EnableMenuItem(Globals.hMenu, CMD_HELP_CONTENTS, MF_BYCOMMAND | MF_GRAYED);
357         break;
358 
359     case WM_COMMAND:
360         if (HIWORD(wParam) == EN_CHANGE || HIWORD(wParam) == EN_HSCROLL || HIWORD(wParam) == EN_VSCROLL)
361             DIALOG_StatusBarUpdateCaretPos();
362         if ((HIWORD(wParam) == EN_CHANGE))
363             NOTEPAD_EnableSearchMenu();
364         NOTEPAD_MenuCommand(LOWORD(wParam));
365         break;
366 
367     case WM_DESTROYCLIPBOARD:
368         /*MessageBox(Globals.hMainWnd, "Empty clipboard", "Debug", MB_ICONEXCLAMATION);*/
369         break;
370 
371     case WM_CLOSE:
372         if (DoCloseFile()) {
373             if (Globals.hFont)
374                 DeleteObject(Globals.hFont);
375             if (Globals.hDevMode)
376                 GlobalFree(Globals.hDevMode);
377             if (Globals.hDevNames)
378                 GlobalFree(Globals.hDevNames);
379             DestroyWindow(hWnd);
380         }
381         break;
382 
383     case WM_QUERYENDSESSION:
384         if (DoCloseFile()) {
385             return 1;
386         }
387         break;
388 
389     case WM_DESTROY:
390         SetWindowLongPtr(Globals.hEdit, GWLP_WNDPROC, (LONG_PTR)Globals.EditProc);
391         NOTEPAD_SaveSettingsToRegistry();
392         PostQuitMessage(0);
393         break;
394 
395     case WM_SIZE:
396     {
397         if ((Globals.bShowStatusBar != FALSE) && (Globals.bWrapLongLines == FALSE))
398         {
399             RECT rcStatusBar;
400             HDWP hdwp;
401 
402             if (!GetWindowRect(Globals.hStatusBar, &rcStatusBar))
403                 break;
404 
405             hdwp = BeginDeferWindowPos(2);
406             if (hdwp == NULL)
407                 break;
408 
409             hdwp = DeferWindowPos(hdwp,
410                                   Globals.hEdit,
411                                   NULL,
412                                   0,
413                                   0,
414                                   LOWORD(lParam),
415                                   HIWORD(lParam) - (rcStatusBar.bottom - rcStatusBar.top),
416                                   SWP_NOZORDER | SWP_NOMOVE);
417 
418             if (hdwp == NULL)
419                 break;
420 
421             hdwp = DeferWindowPos(hdwp,
422                                   Globals.hStatusBar,
423                                   NULL,
424                                   0,
425                                   0,
426                                   LOWORD(lParam),
427                                   LOWORD(wParam),
428                                   SWP_NOZORDER);
429 
430             if (hdwp != NULL)
431                 EndDeferWindowPos(hdwp);
432         }
433         else
434             SetWindowPos(Globals.hEdit,
435                          NULL,
436                          0,
437                          0,
438                          LOWORD(lParam),
439                          HIWORD(lParam),
440                          SWP_NOZORDER | SWP_NOMOVE);
441 
442         break;
443     }
444 
445     /* The entire client area is covered by edit control and by
446      * the status bar. So there is no need to erase main background.
447      * This resolves the horrible flicker effect during windows resizes. */
448     case WM_ERASEBKGND:
449         return 1;
450 
451     case WM_SETFOCUS:
452         SetFocus(Globals.hEdit);
453         break;
454 
455     case WM_DROPFILES:
456     {
457         TCHAR szFileName[MAX_PATH];
458         HDROP hDrop = (HDROP) wParam;
459 
460         DragQueryFile(hDrop, 0, szFileName, ARRAY_SIZE(szFileName));
461         DragFinish(hDrop);
462         DoOpenFile(szFileName);
463         break;
464     }
465     case WM_CHAR:
466     case WM_INITMENUPOPUP:
467         NOTEPAD_InitMenuPopup((HMENU)wParam, lParam);
468         break;
469     default:
470         if (msg == aFINDMSGSTRING)
471         {
472             FINDREPLACE *pFindReplace = (FINDREPLACE *) lParam;
473             Globals.find = *(FINDREPLACE *) lParam;
474 
475             if (pFindReplace->Flags & FR_FINDNEXT)
476                 NOTEPAD_FindNext(pFindReplace, FALSE, TRUE);
477             else if (pFindReplace->Flags & FR_REPLACE)
478                 NOTEPAD_FindNext(pFindReplace, TRUE, TRUE);
479             else if (pFindReplace->Flags & FR_REPLACEALL)
480                 NOTEPAD_ReplaceAll(pFindReplace);
481             else if (pFindReplace->Flags & FR_DIALOGTERM)
482                 NOTEPAD_FindTerm();
483             break;
484         }
485 
486         return DefWindowProc(hWnd, msg, wParam, lParam);
487     }
488     return 0;
489 }
490 
491 static int AlertFileDoesNotExist(LPCTSTR szFileName)
492 {
493     return DIALOG_StringMsgBox(Globals.hMainWnd, STRING_DOESNOTEXIST,
494                                szFileName,
495                                MB_ICONEXCLAMATION | MB_YESNO);
496 }
497 
498 static BOOL HandleCommandLine(LPTSTR cmdline)
499 {
500     BOOL opt_print = FALSE;
501 
502     while (*cmdline == _T(' ') || *cmdline == _T('-') || *cmdline == _T('/'))
503     {
504         TCHAR option;
505 
506         if (*cmdline++ == _T(' ')) continue;
507 
508         option = *cmdline;
509         if (option) cmdline++;
510         while (*cmdline == _T(' ')) cmdline++;
511 
512         switch(option)
513         {
514             case 'p':
515             case 'P':
516                 opt_print = TRUE;
517                 break;
518         }
519     }
520 
521     if (*cmdline)
522     {
523         /* file name is passed in the command line */
524         LPCTSTR file_name = NULL;
525         BOOL file_exists = FALSE;
526         TCHAR buf[MAX_PATH];
527 
528         if (cmdline[0] == _T('"'))
529         {
530             cmdline++;
531             cmdline[lstrlen(cmdline) - 1] = 0;
532         }
533 
534         file_name = cmdline;
535         if (FileExists(file_name))
536         {
537             file_exists = TRUE;
538         }
539         else if (!HasFileExtension(cmdline))
540         {
541             static const TCHAR txt[] = _T(".txt");
542 
543             /* try to find file with ".txt" extension */
544             if (!_tcscmp(txt, cmdline + _tcslen(cmdline) - _tcslen(txt)))
545             {
546                 file_exists = FALSE;
547             }
548             else
549             {
550                 _tcsncpy(buf, cmdline, MAX_PATH - _tcslen(txt) - 1);
551                 _tcscat(buf, txt);
552                 file_name = buf;
553                 file_exists = FileExists(file_name);
554             }
555         }
556 
557         if (file_exists)
558         {
559             DoOpenFile(file_name);
560             InvalidateRect(Globals.hMainWnd, NULL, FALSE);
561             if (opt_print)
562             {
563                 DIALOG_FilePrint();
564                 return FALSE;
565             }
566         }
567         else
568         {
569             switch (AlertFileDoesNotExist(file_name)) {
570             case IDYES:
571                 DoOpenFile(file_name);
572                 break;
573 
574             case IDNO:
575                 break;
576             }
577         }
578     }
579 
580     return TRUE;
581 }
582 
583 /***********************************************************************
584  *
585  *           WinMain
586  */
587 int WINAPI _tWinMain(HINSTANCE hInstance, HINSTANCE prev, LPTSTR cmdline, int show)
588 {
589     MSG msg;
590     HACCEL hAccel;
591     WNDCLASSEX wndclass;
592     HMONITOR monitor;
593     MONITORINFO info;
594     INT x, y;
595 
596     static const TCHAR className[] = _T("Notepad");
597     static const TCHAR winName[] = _T("Notepad");
598 
599     switch (GetUserDefaultUILanguage())
600     {
601     case MAKELANGID(LANG_HEBREW, SUBLANG_DEFAULT):
602         SetProcessDefaultLayout(LAYOUT_RTL);
603         break;
604 
605     default:
606         break;
607     }
608 
609     UNREFERENCED_PARAMETER(prev);
610 
611     aFINDMSGSTRING = (ATOM)RegisterWindowMessage(FINDMSGSTRING);
612 
613     ZeroMemory(&Globals, sizeof(Globals));
614     Globals.hInstance = hInstance;
615     NOTEPAD_LoadSettingsFromRegistry();
616 
617     ZeroMemory(&wndclass, sizeof(wndclass));
618     wndclass.cbSize = sizeof(wndclass);
619     wndclass.lpfnWndProc = NOTEPAD_WndProc;
620     wndclass.hInstance = Globals.hInstance;
621     wndclass.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_NPICON));
622     wndclass.hCursor = LoadCursor(0, IDC_ARROW);
623     wndclass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
624     wndclass.lpszMenuName = MAKEINTRESOURCE(MAIN_MENU);
625     wndclass.lpszClassName = className;
626     wndclass.hIconSm = (HICON)LoadImage(hInstance,
627                                         MAKEINTRESOURCE(IDI_NPICON),
628                                         IMAGE_ICON,
629                                         16,
630                                         16,
631                                         0);
632 
633     if (!RegisterClassEx(&wndclass)) return FALSE;
634 
635     /* Setup windows */
636 
637     monitor = MonitorFromRect(&Globals.main_rect, MONITOR_DEFAULTTOPRIMARY);
638     info.cbSize = sizeof(info);
639     GetMonitorInfoW(monitor, &info);
640 
641     x = Globals.main_rect.left;
642     y = Globals.main_rect.top;
643     if (Globals.main_rect.left >= info.rcWork.right ||
644         Globals.main_rect.top >= info.rcWork.bottom ||
645         Globals.main_rect.right < info.rcWork.left ||
646         Globals.main_rect.bottom < info.rcWork.top)
647         x = y = CW_USEDEFAULT;
648 
649     Globals.hMainWnd = CreateWindow(className,
650                                     winName,
651                                     WS_OVERLAPPEDWINDOW,
652                                     x,
653                                     y,
654                                     Globals.main_rect.right - Globals.main_rect.left,
655                                     Globals.main_rect.bottom - Globals.main_rect.top,
656                                     NULL,
657                                     NULL,
658                                     Globals.hInstance,
659                                     NULL);
660     if (!Globals.hMainWnd)
661     {
662         ShowLastError();
663         ExitProcess(1);
664     }
665 
666     DoCreateEditWindow();
667 
668     NOTEPAD_InitData();
669     DIALOG_FileNew();
670 
671     ShowWindow(Globals.hMainWnd, show);
672     UpdateWindow(Globals.hMainWnd);
673     DragAcceptFiles(Globals.hMainWnd, TRUE);
674 
675     DIALOG_ViewStatusBar();
676 
677     if (!HandleCommandLine(cmdline))
678     {
679         return 0;
680     }
681 
682     hAccel = LoadAccelerators(hInstance, MAKEINTRESOURCE(ID_ACCEL));
683 
684     while (GetMessage(&msg, 0, 0, 0))
685     {
686         if (!IsDialogMessage(Globals.hFindReplaceDlg, &msg) &&
687             !TranslateAccelerator(Globals.hMainWnd, hAccel, &msg))
688         {
689             TranslateMessage(&msg);
690             DispatchMessage(&msg);
691         }
692     }
693     return (int) msg.wParam;
694 }
695