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