4e47566b59eaa8bc5a31c4a830084caf3330d7b9
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
34 #include "notepad_res.h"
36 NOTEPAD_GLOBALS Globals
;
37 static ATOM aFINDMSGSTRING
;
39 /***********************************************************************
43 * Sets Global File Name.
45 VOID
SetFileName(LPCWSTR szFileName
)
47 lstrcpy(Globals
.szFileName
, szFileName
);
48 Globals
.szFileTitle
[0] = 0;
49 GetFileTitle(szFileName
, Globals
.szFileTitle
, sizeof(Globals
.szFileTitle
));
52 /***********************************************************************
56 * All handling of main menu events
58 static int NOTEPAD_MenuCommand(WPARAM wParam
)
62 case CMD_NEW
: DIALOG_FileNew(); break;
63 case CMD_OPEN
: DIALOG_FileOpen(); break;
64 case CMD_SAVE
: DIALOG_FileSave(); break;
65 case CMD_SAVE_AS
: DIALOG_FileSaveAs(); break;
66 case CMD_PRINT
: DIALOG_FilePrint(); break;
67 case CMD_PAGE_SETUP
: DIALOG_FilePageSetup(); break;
68 case CMD_PRINTER_SETUP
: DIALOG_FilePrinterSetup();break;
69 case CMD_EXIT
: DIALOG_FileExit(); break;
71 case CMD_UNDO
: DIALOG_EditUndo(); break;
72 case CMD_CUT
: DIALOG_EditCut(); break;
73 case CMD_COPY
: DIALOG_EditCopy(); break;
74 case CMD_PASTE
: DIALOG_EditPaste(); break;
75 case CMD_DELETE
: DIALOG_EditDelete(); break;
76 case CMD_SELECT_ALL
: DIALOG_EditSelectAll(); break;
77 case CMD_TIME_DATE
: DIALOG_EditTimeDate();break;
79 case CMD_SEARCH
: DIALOG_Search(); break;
80 case CMD_SEARCH_NEXT
: DIALOG_SearchNext(); break;
81 case CMD_REPLACE
: DIALOG_Replace(); break;
82 case CMD_GOTO
: DIALOG_GoTo(); break;
84 case CMD_WRAP
: DIALOG_EditWrap(); break;
85 case CMD_FONT
: DIALOG_SelectFont(); break;
87 case CMD_HELP_CONTENTS
: DIALOG_HelpContents(); break;
88 case CMD_HELP_SEARCH
: DIALOG_HelpSearch(); break;
89 case CMD_HELP_ON_HELP
: DIALOG_HelpHelp(); break;
90 case CMD_LICENSE
: DIALOG_HelpLicense(); break;
91 case CMD_NO_WARRANTY
: DIALOG_HelpNoWarranty(); break;
92 case CMD_ABOUT_WINE
: DIALOG_HelpAboutWine(); break;
100 /***********************************************************************
105 static BOOL
NOTEPAD_FindTextAt(FINDREPLACE
*pFindReplace
, LPCTSTR pszText
, int iTextLength
, DWORD dwPosition
)
110 iTargetLength
= _tcslen(pFindReplace
->lpstrFindWhat
);
112 /* Make proper comparison */
113 if (pFindReplace
->Flags
& FR_MATCHCASE
)
114 bMatches
= !_tcsncmp(&pszText
[dwPosition
], pFindReplace
->lpstrFindWhat
, iTargetLength
);
116 bMatches
= !_tcsnicmp(&pszText
[dwPosition
], pFindReplace
->lpstrFindWhat
, iTargetLength
);
118 if (bMatches
&& pFindReplace
->Flags
& FR_WHOLEWORD
)
120 if ((dwPosition
> 0) && !_istspace(pszText
[dwPosition
-1]))
122 if ((dwPosition
< iTextLength
- 1) && !_istspace(pszText
[dwPosition
+1]))
129 /***********************************************************************
134 static BOOL
NOTEPAD_FindNext(FINDREPLACE
*pFindReplace
, BOOL bReplace
, BOOL bShowAlert
)
136 int iTextLength
, iTargetLength
;
138 LPTSTR pszText
= NULL
;
139 DWORD dwPosition
, dwBegin
, dwEnd
;
140 BOOL bMatches
= FALSE
;
141 TCHAR szResource
[128], szText
[128];
144 iTargetLength
= _tcslen(pFindReplace
->lpstrFindWhat
);
146 iTextLength
= GetWindowTextLength(Globals
.hEdit
);
150 pszText
= (LPTSTR
) HeapAlloc(GetProcessHeap(), 0, (iTextLength
+ 1) * sizeof(TCHAR
));
154 GetWindowText(Globals
.hEdit
, pszText
, iTextLength
+ 1);
157 SendMessage(Globals
.hEdit
, EM_GETSEL
, (WPARAM
) &dwBegin
, (LPARAM
) &dwEnd
);
158 if (bReplace
&& ((dwEnd
- dwBegin
) == iTargetLength
))
160 if (NOTEPAD_FindTextAt(pFindReplace
, pszText
, iTextLength
, dwBegin
))
162 SendMessage(Globals
.hEdit
, EM_REPLACESEL
, TRUE
, (LPARAM
) pFindReplace
->lpstrReplaceWith
);
163 iAdjustment
= _tcslen(pFindReplace
->lpstrReplaceWith
) - (dwEnd
- dwBegin
);
168 while(dwPosition
< iTextLength
)
170 bMatches
= NOTEPAD_FindTextAt(pFindReplace
, pszText
, iTextLength
, dwPosition
);
174 if (pFindReplace
->Flags
& FR_DOWN
)
183 if (dwPosition
> dwBegin
)
184 dwPosition
+= iAdjustment
;
185 SendMessage(Globals
.hEdit
, EM_SETSEL
, dwPosition
, dwPosition
+ iTargetLength
);
186 SendMessage(Globals
.hEdit
, EM_SCROLLCARET
, 0, 0);
191 /* Can't find target */
194 LoadString(Globals
.hInstance
, STRING_CANNOTFIND
, szResource
, SIZEOF(szResource
));
195 _sntprintf(szText
, SIZEOF(szText
), szResource
, pFindReplace
->lpstrFindWhat
);
196 LoadString(Globals
.hInstance
, STRING_NOTEPAD
, szResource
, SIZEOF(szResource
));
197 MessageBox(Globals
.hFindReplaceDlg
, szText
, szResource
, MB_OK
);
203 HeapFree(GetProcessHeap(), 0, pszText
);
207 /***********************************************************************
212 static VOID
NOTEPAD_ReplaceAll(FINDREPLACE
*pFindReplace
)
214 BOOL bShowAlert
= TRUE
;
216 SendMessage(Globals
.hEdit
, EM_SETSEL
, 0, 0);
218 while (NOTEPAD_FindNext(pFindReplace
, TRUE
, bShowAlert
))
224 /***********************************************************************
229 static VOID
NOTEPAD_FindTerm(VOID
)
231 Globals
.hFindReplaceDlg
= NULL
;
234 /***********************************************************************
235 * Data Initialization
237 static VOID
NOTEPAD_InitData(VOID
)
239 LPWSTR p
= Globals
.szFilter
;
240 static const WCHAR txt_files
[] = { '*','.','t','x','t',0 };
241 static const WCHAR all_files
[] = { '*','.','*',0 };
243 LoadString(Globals
.hInstance
, STRING_TEXT_FILES_TXT
, p
, MAX_STRING_LEN
);
245 lstrcpy(p
, txt_files
);
247 LoadString(Globals
.hInstance
, STRING_ALL_FILES
, p
, MAX_STRING_LEN
);
249 lstrcpy(p
, all_files
);
254 /***********************************************************************
255 * Enable/disable items on the menu based on control state
257 static VOID
NOTEPAD_InitMenuPopup(HMENU menu
, int index
)
261 CheckMenuItem(GetMenu(Globals
.hMainWnd
), CMD_WRAP
,
262 MF_BYCOMMAND
| (Globals
.bWrapLongLines
? MF_CHECKED
: MF_UNCHECKED
));
264 EnableMenuItem(menu
, CMD_UNDO
,
265 SendMessage(Globals
.hEdit
, EM_CANUNDO
, 0, 0) ? MF_ENABLED
: MF_GRAYED
);
266 EnableMenuItem(menu
, CMD_PASTE
,
267 IsClipboardFormatAvailable(CF_TEXT
) ? MF_ENABLED
: MF_GRAYED
);
268 enable
= SendMessage(Globals
.hEdit
, EM_GETSEL
, 0, 0);
269 enable
= (HIWORD(enable
) == LOWORD(enable
)) ? MF_GRAYED
: MF_ENABLED
;
270 EnableMenuItem(menu
, CMD_CUT
, enable
);
271 EnableMenuItem(menu
, CMD_COPY
, enable
);
272 EnableMenuItem(menu
, CMD_DELETE
, enable
);
274 EnableMenuItem(menu
, CMD_SELECT_ALL
,
275 GetWindowTextLength(Globals
.hEdit
) ? MF_ENABLED
: MF_GRAYED
);
278 /***********************************************************************
282 static LRESULT WINAPI
NOTEPAD_WndProc(HWND hWnd
, UINT msg
, WPARAM wParam
,
289 static const WCHAR editW
[] = { 'e','d','i','t',0 };
291 GetClientRect(hWnd
, &rc
);
292 Globals
.hEdit
= CreateWindowEx(EDIT_EXSTYLE
, editW
, NULL
, Globals
.bWrapLongLines
? EDIT_STYLE_WRAP
: EDIT_STYLE
,
293 0, 0, rc
.right
, rc
.bottom
, hWnd
,
294 NULL
, Globals
.hInstance
, NULL
);
297 SendMessage(Globals
.hEdit
, EM_LIMITTEXT
, 0, 0);
299 SendMessage(Globals
.hEdit
, WM_SETFONT
, (WPARAM
)Globals
.hFont
, (LPARAM
)TRUE
);
304 NOTEPAD_MenuCommand(LOWORD(wParam
));
307 case WM_DESTROYCLIPBOARD
:
308 /*MessageBox(Globals.hMainWnd, "Empty clipboard", "Debug", MB_ICONEXCLAMATION);*/
317 case WM_QUERYENDSESSION
:
328 SetWindowPos(Globals
.hEdit
, NULL
, 0, 0, LOWORD(lParam
), HIWORD(lParam
),
329 SWP_NOOWNERZORDER
| SWP_NOZORDER
);
333 SetFocus(Globals
.hEdit
);
338 WCHAR szFileName
[MAX_PATH
];
339 HANDLE hDrop
= (HANDLE
) wParam
;
341 DragQueryFile(hDrop
, 0, szFileName
, SIZEOF(szFileName
));
343 DoOpenFile(szFileName
);
347 case WM_INITMENUPOPUP
:
348 NOTEPAD_InitMenuPopup((HMENU
)wParam
, lParam
);
352 if (msg
== aFINDMSGSTRING
)
354 FINDREPLACE
*pFindReplace
= (FINDREPLACE
*) lParam
;
356 if (pFindReplace
->Flags
& FR_FINDNEXT
)
357 NOTEPAD_FindNext(pFindReplace
, FALSE
, TRUE
);
358 else if (pFindReplace
->Flags
& FR_REPLACE
)
359 NOTEPAD_FindNext(pFindReplace
, TRUE
, TRUE
);
360 else if (pFindReplace
->Flags
& FR_REPLACEALL
)
361 NOTEPAD_ReplaceAll(pFindReplace
);
362 else if (pFindReplace
->Flags
& FR_DIALOGTERM
)
367 return DefWindowProc(hWnd
, msg
, wParam
, lParam
);
372 static int AlertFileDoesNotExist(LPCWSTR szFileName
)
375 WCHAR szMessage
[MAX_STRING_LEN
];
376 WCHAR szResource
[MAX_STRING_LEN
];
378 LoadString(Globals
.hInstance
, STRING_DOESNOTEXIST
, szResource
, SIZEOF(szResource
));
379 wsprintf(szMessage
, szResource
, szFileName
);
381 LoadString(Globals
.hInstance
, STRING_NOTEPAD
, szResource
, SIZEOF(szResource
));
383 nResult
= MessageBox(Globals
.hMainWnd
, szMessage
, szResource
,
384 MB_ICONEXCLAMATION
| MB_YESNO
);
389 static void HandleCommandLine(LPWSTR cmdline
)
394 /* skip white space */
395 while (*cmdline
== ' ') cmdline
++;
397 /* skip executable name */
398 delimiter
= (*cmdline
== '"' ? '"' : ' ');
404 while (*cmdline
&& *cmdline
!= delimiter
);
405 if (*cmdline
== delimiter
) cmdline
++;
407 while (*cmdline
== ' ' || *cmdline
== '-' || *cmdline
== '/')
411 if (*cmdline
++ == ' ') continue;
414 if (option
) cmdline
++;
415 while (*cmdline
== ' ') cmdline
++;
428 /* file name is passed in the command line */
433 if (cmdline
[0] == '"')
436 cmdline
[lstrlen(cmdline
) - 1] = 0;
439 if (FileExists(cmdline
))
446 static const WCHAR txtW
[] = { '.','t','x','t',0 };
448 /* try to find file with ".txt" extension */
449 if (!lstrcmp(txtW
, cmdline
+ lstrlen(cmdline
) - lstrlen(txtW
)))
456 lstrcpyn(buf
, cmdline
, MAX_PATH
- lstrlen(txtW
) - 1);
459 file_exists
= FileExists(buf
);
465 DoOpenFile(file_name
);
466 InvalidateRect(Globals
.hMainWnd
, NULL
, FALSE
);
472 switch (AlertFileDoesNotExist(file_name
)) {
474 DoOpenFile(file_name
);
484 /***********************************************************************
488 int PASCAL
WinMain(HINSTANCE hInstance
, HINSTANCE prev
, LPSTR cmdline
, int show
)
493 static const WCHAR className
[] = {'N','P','C','l','a','s','s',0};
494 static const WCHAR winName
[] = {'N','o','t','e','p','a','d',0};
496 aFINDMSGSTRING
= RegisterWindowMessage(FINDMSGSTRING
);
498 ZeroMemory(&Globals
, sizeof(Globals
));
499 Globals
.hInstance
= hInstance
;
502 ZeroMemory(&class, sizeof(class));
503 class.cbSize
= sizeof(class);
504 class.lpfnWndProc
= NOTEPAD_WndProc
;
505 class.hInstance
= Globals
.hInstance
;
506 class.hIcon
= LoadIcon(0, IDI_APPLICATION
);
507 class.hCursor
= LoadCursor(0, IDC_ARROW
);
508 class.hbrBackground
= (HBRUSH
)(COLOR_WINDOW
+ 1);
509 class.lpszMenuName
= MAKEINTRESOURCE(MAIN_MENU
);
510 class.lpszClassName
= className
;
512 if (!RegisterClassEx(&class)) return FALSE
;
517 CreateWindow(className
, winName
, WS_OVERLAPPEDWINDOW
,
518 CW_USEDEFAULT
, 0, CW_USEDEFAULT
, 0,
519 NULL
, NULL
, Globals
.hInstance
, NULL
);
520 if (!Globals
.hMainWnd
)
529 ShowWindow(Globals
.hMainWnd
, show
);
530 UpdateWindow(Globals
.hMainWnd
);
531 DragAcceptFiles(Globals
.hMainWnd
, TRUE
);
533 HandleCommandLine(GetCommandLine());
535 hAccel
= LoadAccelerators( hInstance
, MAKEINTRESOURCE(ID_ACCEL
) );
537 while (GetMessage(&msg
, 0, 0, 0))
539 if (!IsDialogMessage(Globals
.hFindReplaceDlg
, &msg
) &&
540 !TranslateAccelerator(Globals
.hMainWnd
, hAccel
, &msg
))
542 TranslateMessage(&msg
);
543 DispatchMessage(&msg
);