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
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
28 #define _CRT_SECURE_NO_DEPRECATE
36 #include "notepad_res.h"
38 NOTEPAD_GLOBALS Globals
;
39 static ATOM aFINDMSGSTRING
;
41 /***********************************************************************
45 * Sets Global File Name.
47 VOID
SetFileName(LPCWSTR szFileName
)
49 lstrcpy(Globals
.szFileName
, szFileName
);
50 Globals
.szFileTitle
[0] = 0;
51 GetFileTitle(szFileName
, Globals
.szFileTitle
, sizeof(Globals
.szFileTitle
));
54 /***********************************************************************
58 * All handling of main menu events
60 static int NOTEPAD_MenuCommand(WPARAM wParam
)
64 case CMD_NEW
: DIALOG_FileNew(); break;
65 case CMD_OPEN
: DIALOG_FileOpen(); break;
66 case CMD_SAVE
: DIALOG_FileSave(); break;
67 case CMD_SAVE_AS
: DIALOG_FileSaveAs(); break;
68 case CMD_PRINT
: DIALOG_FilePrint(); break;
69 case CMD_PAGE_SETUP
: DIALOG_FilePageSetup(); break;
70 case CMD_PRINTER_SETUP
: DIALOG_FilePrinterSetup();break;
71 case CMD_EXIT
: DIALOG_FileExit(); break;
73 case CMD_UNDO
: DIALOG_EditUndo(); break;
74 case CMD_CUT
: DIALOG_EditCut(); break;
75 case CMD_COPY
: DIALOG_EditCopy(); break;
76 case CMD_PASTE
: DIALOG_EditPaste(); break;
77 case CMD_DELETE
: DIALOG_EditDelete(); break;
78 case CMD_SELECT_ALL
: DIALOG_EditSelectAll(); break;
79 case CMD_TIME_DATE
: DIALOG_EditTimeDate();break;
81 case CMD_SEARCH
: DIALOG_Search(); break;
82 case CMD_SEARCH_NEXT
: DIALOG_SearchNext(); break;
83 case CMD_REPLACE
: DIALOG_Replace(); break;
84 case CMD_GOTO
: DIALOG_GoTo(); break;
86 case CMD_WRAP
: DIALOG_EditWrap(); break;
87 case CMD_FONT
: DIALOG_SelectFont(); break;
89 case CMD_HELP_CONTENTS
: DIALOG_HelpContents(); break;
90 case CMD_HELP_SEARCH
: DIALOG_HelpSearch(); break;
91 case CMD_HELP_ON_HELP
: DIALOG_HelpHelp(); break;
92 case CMD_LICENSE
: DIALOG_HelpLicense(); break;
93 case CMD_NO_WARRANTY
: DIALOG_HelpNoWarranty(); break;
94 case CMD_ABOUT_WINE
: DIALOG_HelpAboutWine(); break;
102 /***********************************************************************
107 static BOOL
NOTEPAD_FindTextAt(FINDREPLACE
*pFindReplace
, LPCTSTR pszText
, int iTextLength
, DWORD dwPosition
)
112 iTargetLength
= _tcslen(pFindReplace
->lpstrFindWhat
);
114 /* Make proper comparison */
115 if (pFindReplace
->Flags
& FR_MATCHCASE
)
116 bMatches
= !_tcsncmp(&pszText
[dwPosition
], pFindReplace
->lpstrFindWhat
, iTargetLength
);
118 bMatches
= !_tcsnicmp(&pszText
[dwPosition
], pFindReplace
->lpstrFindWhat
, iTargetLength
);
120 if (bMatches
&& pFindReplace
->Flags
& FR_WHOLEWORD
)
122 if ((dwPosition
> 0) && !_istspace(pszText
[dwPosition
-1]))
124 if ((dwPosition
< iTextLength
- 1) && !_istspace(pszText
[dwPosition
+1]))
131 /***********************************************************************
136 static BOOL
NOTEPAD_FindNext(FINDREPLACE
*pFindReplace
, BOOL bReplace
, BOOL bShowAlert
)
138 int iTextLength
, iTargetLength
;
140 LPTSTR pszText
= NULL
;
141 DWORD dwPosition
, dwBegin
, dwEnd
;
142 BOOL bMatches
= FALSE
;
143 TCHAR szResource
[128], szText
[128];
146 iTargetLength
= _tcslen(pFindReplace
->lpstrFindWhat
);
148 iTextLength
= GetWindowTextLength(Globals
.hEdit
);
152 pszText
= (LPTSTR
) HeapAlloc(GetProcessHeap(), 0, (iTextLength
+ 1) * sizeof(TCHAR
));
156 GetWindowText(Globals
.hEdit
, pszText
, iTextLength
+ 1);
159 SendMessage(Globals
.hEdit
, EM_GETSEL
, (WPARAM
) &dwBegin
, (LPARAM
) &dwEnd
);
160 if (bReplace
&& ((dwEnd
- dwBegin
) == iTargetLength
))
162 if (NOTEPAD_FindTextAt(pFindReplace
, pszText
, iTextLength
, dwBegin
))
164 SendMessage(Globals
.hEdit
, EM_REPLACESEL
, TRUE
, (LPARAM
) pFindReplace
->lpstrReplaceWith
);
165 iAdjustment
= _tcslen(pFindReplace
->lpstrReplaceWith
) - (dwEnd
- dwBegin
);
170 while(dwPosition
< iTextLength
)
172 bMatches
= NOTEPAD_FindTextAt(pFindReplace
, pszText
, iTextLength
, dwPosition
);
176 if (pFindReplace
->Flags
& FR_DOWN
)
185 if (dwPosition
> dwBegin
)
186 dwPosition
+= iAdjustment
;
187 SendMessage(Globals
.hEdit
, EM_SETSEL
, dwPosition
, dwPosition
+ iTargetLength
);
188 SendMessage(Globals
.hEdit
, EM_SCROLLCARET
, 0, 0);
193 /* Can't find target */
196 LoadString(Globals
.hInstance
, STRING_CANNOTFIND
, szResource
, SIZEOF(szResource
));
197 _sntprintf(szText
, SIZEOF(szText
), szResource
, pFindReplace
->lpstrFindWhat
);
198 LoadString(Globals
.hInstance
, STRING_NOTEPAD
, szResource
, SIZEOF(szResource
));
199 MessageBox(Globals
.hFindReplaceDlg
, szText
, szResource
, MB_OK
);
205 HeapFree(GetProcessHeap(), 0, pszText
);
209 /***********************************************************************
214 static VOID
NOTEPAD_ReplaceAll(FINDREPLACE
*pFindReplace
)
216 BOOL bShowAlert
= TRUE
;
218 SendMessage(Globals
.hEdit
, EM_SETSEL
, 0, 0);
220 while (NOTEPAD_FindNext(pFindReplace
, TRUE
, bShowAlert
))
226 /***********************************************************************
231 static VOID
NOTEPAD_FindTerm(VOID
)
233 Globals
.hFindReplaceDlg
= NULL
;
236 /***********************************************************************
237 * Data Initialization
239 static VOID
NOTEPAD_InitData(VOID
)
241 LPWSTR p
= Globals
.szFilter
;
242 static const WCHAR txt_files
[] = { '*','.','t','x','t',0 };
243 static const WCHAR all_files
[] = { '*','.','*',0 };
245 LoadString(Globals
.hInstance
, STRING_TEXT_FILES_TXT
, p
, MAX_STRING_LEN
);
247 lstrcpy(p
, txt_files
);
249 LoadString(Globals
.hInstance
, STRING_ALL_FILES
, p
, MAX_STRING_LEN
);
251 lstrcpy(p
, all_files
);
256 /***********************************************************************
257 * Enable/disable items on the menu based on control state
259 static VOID
NOTEPAD_InitMenuPopup(HMENU menu
, int index
)
263 CheckMenuItem(GetMenu(Globals
.hMainWnd
), CMD_WRAP
,
264 MF_BYCOMMAND
| (Globals
.bWrapLongLines
? MF_CHECKED
: MF_UNCHECKED
));
266 EnableMenuItem(menu
, CMD_UNDO
,
267 SendMessage(Globals
.hEdit
, EM_CANUNDO
, 0, 0) ? MF_ENABLED
: MF_GRAYED
);
268 EnableMenuItem(menu
, CMD_PASTE
,
269 IsClipboardFormatAvailable(CF_TEXT
) ? MF_ENABLED
: MF_GRAYED
);
270 enable
= SendMessage(Globals
.hEdit
, EM_GETSEL
, 0, 0);
271 enable
= (HIWORD(enable
) == LOWORD(enable
)) ? MF_GRAYED
: MF_ENABLED
;
272 EnableMenuItem(menu
, CMD_CUT
, enable
);
273 EnableMenuItem(menu
, CMD_COPY
, enable
);
274 EnableMenuItem(menu
, CMD_DELETE
, enable
);
276 EnableMenuItem(menu
, CMD_SELECT_ALL
,
277 GetWindowTextLength(Globals
.hEdit
) ? MF_ENABLED
: MF_GRAYED
);
280 /***********************************************************************
284 static LRESULT WINAPI
NOTEPAD_WndProc(HWND hWnd
, UINT msg
, WPARAM wParam
,
291 static const WCHAR editW
[] = { 'e','d','i','t',0 };
293 GetClientRect(hWnd
, &rc
);
294 Globals
.hEdit
= CreateWindowEx(EDIT_EXSTYLE
, editW
, NULL
, Globals
.bWrapLongLines
? EDIT_STYLE_WRAP
: EDIT_STYLE
,
295 0, 0, rc
.right
, rc
.bottom
, hWnd
,
296 NULL
, Globals
.hInstance
, NULL
);
299 SendMessage(Globals
.hEdit
, EM_LIMITTEXT
, 0, 0);
301 SendMessage(Globals
.hEdit
, WM_SETFONT
, (WPARAM
)Globals
.hFont
, (LPARAM
)TRUE
);
306 NOTEPAD_MenuCommand(LOWORD(wParam
));
309 case WM_DESTROYCLIPBOARD
:
310 /*MessageBox(Globals.hMainWnd, "Empty clipboard", "Debug", MB_ICONEXCLAMATION);*/
319 case WM_QUERYENDSESSION
:
330 SetWindowPos(Globals
.hEdit
, NULL
, 0, 0, LOWORD(lParam
), HIWORD(lParam
),
331 SWP_NOOWNERZORDER
| SWP_NOZORDER
);
335 SetFocus(Globals
.hEdit
);
340 WCHAR szFileName
[MAX_PATH
];
341 HANDLE hDrop
= (HANDLE
) wParam
;
343 DragQueryFile(hDrop
, 0, szFileName
, SIZEOF(szFileName
));
345 DoOpenFile(szFileName
);
349 case WM_INITMENUPOPUP
:
350 NOTEPAD_InitMenuPopup((HMENU
)wParam
, lParam
);
354 if (msg
== aFINDMSGSTRING
)
356 FINDREPLACE
*pFindReplace
= (FINDREPLACE
*) lParam
;
358 if (pFindReplace
->Flags
& FR_FINDNEXT
)
359 NOTEPAD_FindNext(pFindReplace
, FALSE
, TRUE
);
360 else if (pFindReplace
->Flags
& FR_REPLACE
)
361 NOTEPAD_FindNext(pFindReplace
, TRUE
, TRUE
);
362 else if (pFindReplace
->Flags
& FR_REPLACEALL
)
363 NOTEPAD_ReplaceAll(pFindReplace
);
364 else if (pFindReplace
->Flags
& FR_DIALOGTERM
)
369 return DefWindowProc(hWnd
, msg
, wParam
, lParam
);
374 static int AlertFileDoesNotExist(LPCWSTR szFileName
)
377 WCHAR szMessage
[MAX_STRING_LEN
];
378 WCHAR szResource
[MAX_STRING_LEN
];
380 LoadString(Globals
.hInstance
, STRING_DOESNOTEXIST
, szResource
, SIZEOF(szResource
));
381 wsprintf(szMessage
, szResource
, szFileName
);
383 LoadString(Globals
.hInstance
, STRING_NOTEPAD
, szResource
, SIZEOF(szResource
));
385 nResult
= MessageBox(Globals
.hMainWnd
, szMessage
, szResource
,
386 MB_ICONEXCLAMATION
| MB_YESNO
);
391 static void HandleCommandLine(LPWSTR cmdline
)
396 /* skip white space */
397 while (*cmdline
== ' ') cmdline
++;
399 /* skip executable name */
400 delimiter
= (*cmdline
== '"' ? '"' : ' ');
406 while (*cmdline
&& *cmdline
!= delimiter
);
407 if (*cmdline
== delimiter
) cmdline
++;
409 while (*cmdline
== ' ' || *cmdline
== '-' || *cmdline
== '/')
413 if (*cmdline
++ == ' ') continue;
416 if (option
) cmdline
++;
417 while (*cmdline
== ' ') cmdline
++;
430 /* file name is passed in the command line */
435 if (cmdline
[0] == '"')
438 cmdline
[lstrlen(cmdline
) - 1] = 0;
441 if (FileExists(cmdline
))
448 static const WCHAR txtW
[] = { '.','t','x','t',0 };
450 /* try to find file with ".txt" extension */
451 if (!lstrcmp(txtW
, cmdline
+ lstrlen(cmdline
) - lstrlen(txtW
)))
458 lstrcpyn(buf
, cmdline
, MAX_PATH
- lstrlen(txtW
) - 1);
461 file_exists
= FileExists(buf
);
467 DoOpenFile(file_name
);
468 InvalidateRect(Globals
.hMainWnd
, NULL
, FALSE
);
474 switch (AlertFileDoesNotExist(file_name
)) {
476 DoOpenFile(file_name
);
486 /***********************************************************************
490 int PASCAL
WinMain(HINSTANCE hInstance
, HINSTANCE prev
, LPSTR cmdline
, int show
)
495 static const WCHAR className
[] = {'N','P','C','l','a','s','s',0};
496 static const WCHAR winName
[] = {'N','o','t','e','p','a','d',0};
498 aFINDMSGSTRING
= RegisterWindowMessage(FINDMSGSTRING
);
500 ZeroMemory(&Globals
, sizeof(Globals
));
501 Globals
.hInstance
= hInstance
;
504 ZeroMemory(&class, sizeof(class));
505 class.cbSize
= sizeof(class);
506 class.lpfnWndProc
= NOTEPAD_WndProc
;
507 class.hInstance
= Globals
.hInstance
;
508 class.hIcon
= LoadIcon(0, IDI_APPLICATION
);
509 class.hCursor
= LoadCursor(0, IDC_ARROW
);
510 class.hbrBackground
= (HBRUSH
)(COLOR_WINDOW
+ 1);
511 class.lpszMenuName
= MAKEINTRESOURCE(MAIN_MENU
);
512 class.lpszClassName
= className
;
514 if (!RegisterClassEx(&class)) return FALSE
;
519 CreateWindow(className
, winName
, WS_OVERLAPPEDWINDOW
,
520 CW_USEDEFAULT
, 0, CW_USEDEFAULT
, 0,
521 NULL
, NULL
, Globals
.hInstance
, NULL
);
522 if (!Globals
.hMainWnd
)
531 ShowWindow(Globals
.hMainWnd
, show
);
532 UpdateWindow(Globals
.hMainWnd
);
533 DragAcceptFiles(Globals
.hMainWnd
, TRUE
);
535 HandleCommandLine(GetCommandLine());
537 hAccel
= LoadAccelerators( hInstance
, MAKEINTRESOURCE(ID_ACCEL
) );
539 while (GetMessage(&msg
, 0, 0, 0))
541 if (!IsDialogMessage(Globals
.hFindReplaceDlg
, &msg
) &&
542 !TranslateAccelerator(Globals
.hMainWnd
, hAccel
, &msg
))
544 TranslateMessage(&msg
);
545 DispatchMessage(&msg
);