main.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857
  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. *
  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.
  13. *
  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.
  18. *
  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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
  22. *
  23. */
  24. #include <stdio.h>
  25. #include <windows.h>
  26. #include <commctrl.h>
  27. #include <commdlg.h>
  28. #include <shellapi.h>
  29. #include <shlwapi.h>
  30. #include "main.h"
  31. #include "dialog.h"
  32. #include "notepad_res.h"
  33. NOTEPAD_GLOBALS Globals;
  34. static ATOM aFINDMSGSTRING;
  35. static RECT main_rect;
  36. /***********************************************************************
  37. *
  38. * SetFileNameAndEncoding
  39. *
  40. * Sets global file name and encoding (which is used to preselect original
  41. * encoding in Save As dialog, and when saving without using the Save As
  42. * dialog).
  43. */
  44. VOID SetFileNameAndEncoding(LPCWSTR szFileName, ENCODING enc)
  45. {
  46. lstrcpyW(Globals.szFileName, szFileName);
  47. Globals.szFileTitle[0] = 0;
  48. GetFileTitleW(szFileName, Globals.szFileTitle, ARRAY_SIZE(Globals.szFileTitle));
  49. Globals.encFile = enc;
  50. }
  51. void UpdateStatusBar(void)
  52. {
  53. int currentLine;
  54. int currentCol = -1;
  55. WCHAR statusTxt[256];
  56. int lineIndex;
  57. DWORD selStart;
  58. DWORD selEnd;
  59. SendMessageW(Globals.hEdit, EM_GETSEL, (WPARAM)&selStart, (LPARAM)&selEnd);
  60. if(selStart == selEnd)
  61. Globals.trackedSel = selStart;
  62. if(selStart < Globals.trackedSel)
  63. currentCol = selStart;
  64. else
  65. currentCol = selEnd;
  66. currentLine = SendMessageW(Globals.hEdit, EM_LINEFROMCHAR, currentCol, 0);
  67. lineIndex = SendMessageW(Globals.hEdit, EM_LINEINDEX, currentLine, 0);
  68. if(Globals.lastLn != currentLine || Globals.lastCol != currentCol)
  69. {
  70. swprintf(statusTxt, ARRAY_SIZE(statusTxt), Globals.szStatusString, currentLine + 1, (currentCol - lineIndex) + 1);
  71. SendMessageW(Globals.hStatusBar, SB_SETTEXTW, 0, (LPARAM)statusTxt);
  72. Globals.lastLn = currentLine;
  73. Globals.lastCol = currentCol;
  74. }
  75. }
  76. static void ToggleStatusBar(void)
  77. {
  78. RECT rc;
  79. Globals.bStatusBar = !Globals.bStatusBar;
  80. CheckMenuItem(GetMenu(Globals.hMainWnd), CMD_SBAR,
  81. MF_BYCOMMAND | (Globals.bStatusBar ? MF_CHECKED : MF_UNCHECKED));
  82. GetClientRect(Globals.hMainWnd, &rc);
  83. ShowWindow(Globals.hStatusBar, Globals.bStatusBar ? SW_SHOW : SW_HIDE);
  84. updateWindowSize(rc.right, rc.bottom);
  85. UpdateStatusBar();
  86. }
  87. void updateWindowSize(int width, int height)
  88. {
  89. int StatusBarHeight = 0;
  90. if(Globals.bStatusBar)
  91. {
  92. RECT SBarRect;
  93. SendMessageW(Globals.hStatusBar, WM_SIZE, 0, 0);
  94. GetWindowRect(Globals.hStatusBar, &SBarRect);
  95. StatusBarHeight = (SBarRect.bottom - SBarRect.top);
  96. }
  97. SetWindowPos(Globals.hEdit, NULL, 0, 0, width, height - StatusBarHeight,
  98. SWP_NOOWNERZORDER | SWP_NOZORDER);
  99. }
  100. /***********************************************************************
  101. *
  102. * NOTEPAD_SaveSettingToRegistry
  103. *
  104. * Save setting to registry HKCU\Software\Microsoft\Notepad.
  105. */
  106. static VOID NOTEPAD_SaveSettingToRegistry(void)
  107. {
  108. HKEY hkey;
  109. DWORD disp;
  110. if(RegCreateKeyExW(HKEY_CURRENT_USER, L"Software\\Microsoft\\Notepad", 0, NULL,
  111. REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, &disp) == ERROR_SUCCESS)
  112. {
  113. DWORD data;
  114. WINDOWPLACEMENT wndpl;
  115. wndpl.length = sizeof(WINDOWPLACEMENT);
  116. GetWindowPlacement(Globals.hMainWnd, &wndpl);
  117. main_rect = wndpl.rcNormalPosition;
  118. #define SET_NOTEPAD_REG(hkey, value_name, value_data) do { DWORD data = value_data; RegSetValueExW(hkey, value_name, 0, REG_DWORD, (LPBYTE)&data, sizeof(DWORD)); }while(0)
  119. SET_NOTEPAD_REG(hkey, L"fWrap", Globals.bWrapLongLines);
  120. SET_NOTEPAD_REG(hkey, L"iWindowPosX", main_rect.left);
  121. SET_NOTEPAD_REG(hkey, L"iWindowPosY", main_rect.top);
  122. SET_NOTEPAD_REG(hkey, L"iWindowPosDX", main_rect.right - main_rect.left);
  123. SET_NOTEPAD_REG(hkey, L"iWindowPosDY", main_rect.bottom - main_rect.top);
  124. SET_NOTEPAD_REG(hkey, L"lfCharSet", Globals.lfFont.lfCharSet);
  125. SET_NOTEPAD_REG(hkey, L"lfClipPrecision", Globals.lfFont.lfClipPrecision);
  126. SET_NOTEPAD_REG(hkey, L"lfEscapement", Globals.lfFont.lfEscapement);
  127. SET_NOTEPAD_REG(hkey, L"lfItalic", Globals.lfFont.lfItalic);
  128. SET_NOTEPAD_REG(hkey, L"lfOrientation", Globals.lfFont.lfOrientation);
  129. SET_NOTEPAD_REG(hkey, L"lfOutPrecision", Globals.lfFont.lfOutPrecision);
  130. SET_NOTEPAD_REG(hkey, L"lfPitchAndFamily", Globals.lfFont.lfPitchAndFamily);
  131. SET_NOTEPAD_REG(hkey, L"lfQuality", Globals.lfFont.lfQuality);
  132. SET_NOTEPAD_REG(hkey, L"lfStrikeOut", Globals.lfFont.lfStrikeOut);
  133. SET_NOTEPAD_REG(hkey, L"lfUnderline", Globals.lfFont.lfUnderline);
  134. SET_NOTEPAD_REG(hkey, L"lfWeight", Globals.lfFont.lfWeight);
  135. SET_NOTEPAD_REG(hkey, L"iMarginTop", Globals.iMarginTop);
  136. SET_NOTEPAD_REG(hkey, L"iMarginBottom", Globals.iMarginBottom);
  137. SET_NOTEPAD_REG(hkey, L"iMarginLeft", Globals.iMarginLeft);
  138. SET_NOTEPAD_REG(hkey, L"iMarginRight", Globals.iMarginRight);
  139. SET_NOTEPAD_REG(hkey, L"bStatusBar", Globals.bStatusBar);
  140. #undef SET_NOTEPAD_REG
  141. /* Store the current value as 10 * twips */
  142. data = MulDiv(abs(Globals.lfFont.lfHeight), 720, GetDpiForWindow(Globals.hMainWnd));
  143. RegSetValueExW(hkey, L"iPointSize", 0, REG_DWORD, (LPBYTE)&data, sizeof(DWORD));
  144. RegSetValueExW(hkey, L"lfFaceName", 0, REG_SZ, (LPBYTE)&Globals.lfFont.lfFaceName,
  145. lstrlenW(Globals.lfFont.lfFaceName) * sizeof(Globals.lfFont.lfFaceName[0]));
  146. RegSetValueExW(hkey, L"szHeader", 0, REG_SZ, (LPBYTE)&Globals.szHeader,
  147. lstrlenW(Globals.szHeader) * sizeof(Globals.szHeader[0]));
  148. RegSetValueExW(hkey, L"szTrailer", 0, REG_SZ, (LPBYTE)&Globals.szFooter,
  149. lstrlenW(Globals.szFooter) * sizeof(Globals.szFooter[0]));
  150. RegCloseKey(hkey);
  151. }
  152. }
  153. /***********************************************************************
  154. *
  155. * NOTEPAD_LoadSettingFromRegistry
  156. *
  157. * Load setting from registry HKCU\Software\Microsoft\Notepad.
  158. */
  159. static VOID NOTEPAD_LoadSettingFromRegistry(void)
  160. {
  161. HKEY hkey;
  162. INT base_length, dx, dy;
  163. base_length = (GetSystemMetrics(SM_CXSCREEN) > GetSystemMetrics(SM_CYSCREEN))?
  164. GetSystemMetrics(SM_CYSCREEN) : GetSystemMetrics(SM_CXSCREEN);
  165. dx = base_length * .95;
  166. dy = dx * 3 / 4;
  167. SetRect( &main_rect, 0, 0, dx, dy );
  168. Globals.bWrapLongLines = TRUE;
  169. Globals.iMarginTop = 2500;
  170. Globals.iMarginBottom = 2500;
  171. Globals.iMarginLeft = 2000;
  172. Globals.iMarginRight = 2000;
  173. Globals.bStatusBar = TRUE;
  174. Globals.lfFont.lfHeight = -12;
  175. Globals.lfFont.lfWidth = 0;
  176. Globals.lfFont.lfEscapement = 0;
  177. Globals.lfFont.lfOrientation = 0;
  178. Globals.lfFont.lfWeight = FW_REGULAR;
  179. Globals.lfFont.lfItalic = FALSE;
  180. Globals.lfFont.lfUnderline = FALSE;
  181. Globals.lfFont.lfStrikeOut = FALSE;
  182. Globals.lfFont.lfCharSet = DEFAULT_CHARSET;
  183. Globals.lfFont.lfOutPrecision = OUT_DEFAULT_PRECIS;
  184. Globals.lfFont.lfClipPrecision = CLIP_DEFAULT_PRECIS;
  185. Globals.lfFont.lfQuality = DEFAULT_QUALITY;
  186. Globals.lfFont.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
  187. lstrcpyW(Globals.lfFont.lfFaceName, L"System");
  188. LoadStringW(Globals.hInstance, STRING_PAGESETUP_HEADERVALUE,
  189. Globals.szHeader, ARRAY_SIZE(Globals.szHeader));
  190. LoadStringW(Globals.hInstance, STRING_PAGESETUP_FOOTERVALUE,
  191. Globals.szFooter, ARRAY_SIZE(Globals.szFooter));
  192. if(RegOpenKeyW(HKEY_CURRENT_USER, L"Software\\Microsoft\\Notepad", &hkey) == ERROR_SUCCESS)
  193. {
  194. WORD data_helper[MAX_PATH];
  195. DWORD type, size;
  196. int point_size;
  197. #define QUERY_NOTEPAD_REG(hkey, value_name, ret) do { DWORD type, data; DWORD size = sizeof(DWORD); if(RegQueryValueExW(hkey, value_name, 0, &type, (LPBYTE)&data, &size) == ERROR_SUCCESS) if(type == REG_DWORD) ret = data; } while(0)
  198. QUERY_NOTEPAD_REG(hkey, L"fWrap", Globals.bWrapLongLines);
  199. QUERY_NOTEPAD_REG(hkey, L"iWindowPosX", main_rect.left);
  200. QUERY_NOTEPAD_REG(hkey, L"iWindowPosY", main_rect.top);
  201. QUERY_NOTEPAD_REG(hkey, L"iWindowPosDX", dx);
  202. QUERY_NOTEPAD_REG(hkey, L"iWindowPosDY", dy);
  203. QUERY_NOTEPAD_REG(hkey, L"lfCharSet", Globals.lfFont.lfCharSet);
  204. QUERY_NOTEPAD_REG(hkey, L"lfClipPrecision", Globals.lfFont.lfClipPrecision);
  205. QUERY_NOTEPAD_REG(hkey, L"lfEscapement", Globals.lfFont.lfEscapement);
  206. QUERY_NOTEPAD_REG(hkey, L"lfItalic", Globals.lfFont.lfItalic);
  207. QUERY_NOTEPAD_REG(hkey, L"lfOrientation", Globals.lfFont.lfOrientation);
  208. QUERY_NOTEPAD_REG(hkey, L"lfOutPrecision", Globals.lfFont.lfOutPrecision);
  209. QUERY_NOTEPAD_REG(hkey, L"lfPitchAndFamily", Globals.lfFont.lfPitchAndFamily);
  210. QUERY_NOTEPAD_REG(hkey, L"lfQuality", Globals.lfFont.lfQuality);
  211. QUERY_NOTEPAD_REG(hkey, L"lfStrikeOut", Globals.lfFont.lfStrikeOut);
  212. QUERY_NOTEPAD_REG(hkey, L"lfUnderline", Globals.lfFont.lfUnderline);
  213. QUERY_NOTEPAD_REG(hkey, L"lfWeight", Globals.lfFont.lfWeight);
  214. QUERY_NOTEPAD_REG(hkey, L"iMarginTop", Globals.iMarginTop);
  215. QUERY_NOTEPAD_REG(hkey, L"iMarginBottom", Globals.iMarginBottom);
  216. QUERY_NOTEPAD_REG(hkey, L"iMarginLeft", Globals.iMarginLeft);
  217. QUERY_NOTEPAD_REG(hkey, L"iMarginRight", Globals.iMarginRight);
  218. QUERY_NOTEPAD_REG(hkey, L"bStatusBar", Globals.bStatusBar);
  219. #undef QUERY_NOTEPAD_REG
  220. main_rect.right = main_rect.left + dx;
  221. main_rect.bottom = main_rect.top + dy;
  222. size = sizeof(DWORD);
  223. if(RegQueryValueExW(hkey, L"iPointSize", 0, &type, (LPBYTE)&point_size, &size) == ERROR_SUCCESS)
  224. if(type == REG_DWORD && point_size)
  225. /* The value is stored as 10 * twips */
  226. Globals.lfFont.lfHeight = -MulDiv(abs(point_size), GetDpiForWindow(GetDesktopWindow()), 720);
  227. size = sizeof(Globals.lfFont.lfFaceName);
  228. if(RegQueryValueExW(hkey, L"lfFaceName", 0, &type, (LPBYTE)&data_helper, &size) == ERROR_SUCCESS)
  229. if(type == REG_SZ)
  230. lstrcpyW(Globals.lfFont.lfFaceName, data_helper);
  231. size = sizeof(Globals.szHeader);
  232. if(RegQueryValueExW(hkey, L"szHeader", 0, &type, (LPBYTE)&data_helper, &size) == ERROR_SUCCESS)
  233. if(type == REG_SZ)
  234. lstrcpyW(Globals.szHeader, data_helper);
  235. size = sizeof(Globals.szFooter);
  236. if(RegQueryValueExW(hkey, L"szTrailer", 0, &type, (LPBYTE)&data_helper, &size) == ERROR_SUCCESS)
  237. if(type == REG_SZ)
  238. lstrcpyW(Globals.szFooter, data_helper);
  239. RegCloseKey(hkey);
  240. }
  241. }
  242. /***********************************************************************
  243. *
  244. * NOTEPAD_MenuCommand
  245. *
  246. * All handling of main menu events
  247. */
  248. static int NOTEPAD_MenuCommand(WPARAM wParam)
  249. {
  250. switch (wParam)
  251. {
  252. case CMD_NEW: DIALOG_FileNew(); break;
  253. case CMD_OPEN: DIALOG_FileOpen(); break;
  254. case CMD_SAVE: DIALOG_FileSave(); break;
  255. case CMD_SAVE_AS: DIALOG_FileSaveAs(); break;
  256. case CMD_PRINT: DIALOG_FilePrint(); break;
  257. case CMD_PAGE_SETUP: DIALOG_FilePageSetup(); break;
  258. case CMD_PRINTER_SETUP: DIALOG_FilePrinterSetup();break;
  259. case CMD_EXIT: DIALOG_FileExit(); break;
  260. case CMD_UNDO: DIALOG_EditUndo(); break;
  261. case CMD_CUT: DIALOG_EditCut(); break;
  262. case CMD_COPY: DIALOG_EditCopy(); break;
  263. case CMD_PASTE: DIALOG_EditPaste(); break;
  264. case CMD_DELETE: DIALOG_EditDelete(); break;
  265. case CMD_SELECT_ALL: DIALOG_EditSelectAll(); break;
  266. case CMD_TIME_DATE: DIALOG_EditTimeDate();break;
  267. case CMD_SEARCH: DIALOG_Search(); break;
  268. case CMD_SEARCH_NEXT: DIALOG_SearchNext(); break;
  269. case CMD_REPLACE: DIALOG_Replace(); break;
  270. case CMD_GO_TO: DIALOG_EditGoTo(); break;
  271. case CMD_WRAP: DIALOG_EditWrap(); break;
  272. case CMD_FONT: DIALOG_SelectFont(); break;
  273. case CMD_SBAR: ToggleStatusBar(); break;
  274. case CMD_HELP_CONTENTS: DIALOG_HelpContents(); break;
  275. case CMD_HELP_ABOUT_NOTEPAD: DIALOG_HelpAboutNotepad(); break;
  276. default:
  277. break;
  278. }
  279. return 0;
  280. }
  281. /***********************************************************************
  282. * Data Initialization
  283. */
  284. static VOID NOTEPAD_InitData(VOID)
  285. {
  286. LPWSTR p = Globals.szFilter;
  287. LoadStringW(Globals.hInstance, STRING_TEXT_FILES_TXT, p, MAX_STRING_LEN);
  288. p += lstrlenW(p) + 1;
  289. lstrcpyW(p, L"*.txt");
  290. p += lstrlenW(p) + 1;
  291. LoadStringW(Globals.hInstance, STRING_ALL_FILES, p, MAX_STRING_LEN);
  292. p += lstrlenW(p) + 1;
  293. lstrcpyW(p, L"*.*");
  294. p += lstrlenW(p) + 1;
  295. *p = '\0';
  296. Globals.hDevMode = NULL;
  297. Globals.hDevNames = NULL;
  298. CheckMenuItem(GetMenu(Globals.hMainWnd), CMD_WRAP,
  299. MF_BYCOMMAND | (Globals.bWrapLongLines ? MF_CHECKED : MF_UNCHECKED));
  300. CheckMenuItem(GetMenu(Globals.hMainWnd), CMD_SBAR,
  301. MF_BYCOMMAND | (Globals.bStatusBar ? MF_CHECKED : MF_UNCHECKED));
  302. ShowWindow(Globals.hStatusBar, Globals.bStatusBar ? SW_SHOW : SW_HIDE);
  303. }
  304. /***********************************************************************
  305. * Enable/disable items on the menu based on control state
  306. */
  307. static VOID NOTEPAD_InitMenuPopup(HMENU menu, int index)
  308. {
  309. int enable;
  310. EnableMenuItem(menu, CMD_UNDO,
  311. SendMessageW(Globals.hEdit, EM_CANUNDO, 0, 0) ? MF_ENABLED : MF_GRAYED);
  312. EnableMenuItem(menu, CMD_PASTE,
  313. IsClipboardFormatAvailable(CF_TEXT) ? MF_ENABLED : MF_GRAYED);
  314. enable = SendMessageW(Globals.hEdit, EM_GETSEL, 0, 0);
  315. enable = (HIWORD(enable) == LOWORD(enable)) ? MF_GRAYED : MF_ENABLED;
  316. EnableMenuItem(menu, CMD_CUT, enable);
  317. EnableMenuItem(menu, CMD_COPY, enable);
  318. EnableMenuItem(menu, CMD_DELETE, enable);
  319. EnableMenuItem(menu, CMD_SELECT_ALL,
  320. GetWindowTextLengthW(Globals.hEdit) ? MF_ENABLED : MF_GRAYED);
  321. }
  322. static LPWSTR NOTEPAD_StrRStr(LPWSTR pszSource, LPWSTR pszLast, LPWSTR pszSrch)
  323. {
  324. int len = lstrlenW(pszSrch);
  325. pszLast--;
  326. while (pszLast >= pszSource)
  327. {
  328. if (StrCmpNW(pszLast, pszSrch, len) == 0)
  329. return pszLast;
  330. pszLast--;
  331. }
  332. return NULL;
  333. }
  334. /***********************************************************************
  335. * The user activated the Find dialog
  336. */
  337. void NOTEPAD_DoFind(FINDREPLACEW *fr)
  338. {
  339. LPWSTR content, found;
  340. int len = lstrlenW(fr->lpstrFindWhat);
  341. int fileLen;
  342. DWORD pos;
  343. fileLen = GetWindowTextLengthW(Globals.hEdit) + 1;
  344. content = HeapAlloc(GetProcessHeap(), 0, fileLen * sizeof(WCHAR));
  345. if (!content) return;
  346. GetWindowTextW(Globals.hEdit, content, fileLen);
  347. SendMessageW(Globals.hEdit, EM_GETSEL, 0, (LPARAM)&pos);
  348. switch (fr->Flags & (FR_DOWN|FR_MATCHCASE))
  349. {
  350. case 0:
  351. found = StrRStrIW(content, content+pos-len, fr->lpstrFindWhat);
  352. break;
  353. case FR_DOWN:
  354. found = StrStrIW(content+pos, fr->lpstrFindWhat);
  355. break;
  356. case FR_MATCHCASE:
  357. found = NOTEPAD_StrRStr(content, content+pos-len, fr->lpstrFindWhat);
  358. break;
  359. case FR_DOWN|FR_MATCHCASE:
  360. found = StrStrW(content+pos, fr->lpstrFindWhat);
  361. break;
  362. default: /* shouldn't happen */
  363. return;
  364. }
  365. pos = found - content;
  366. HeapFree(GetProcessHeap(), 0, content);
  367. if (!found)
  368. {
  369. DIALOG_StringMsgBox(Globals.hFindReplaceDlg, STRING_NOTFOUND, fr->lpstrFindWhat,
  370. MB_ICONINFORMATION|MB_OK);
  371. return;
  372. }
  373. SendMessageW(Globals.hEdit, EM_SETSEL, pos, pos + len);
  374. }
  375. static void NOTEPAD_DoReplace(FINDREPLACEW *fr)
  376. {
  377. LPWSTR content;
  378. int len = lstrlenW(fr->lpstrFindWhat);
  379. int fileLen;
  380. DWORD pos;
  381. DWORD pos_start;
  382. fileLen = GetWindowTextLengthW(Globals.hEdit) + 1;
  383. content = HeapAlloc(GetProcessHeap(), 0, fileLen * sizeof(WCHAR));
  384. if (!content) return;
  385. GetWindowTextW(Globals.hEdit, content, fileLen);
  386. SendMessageW(Globals.hEdit, EM_GETSEL, (WPARAM)&pos_start, (LPARAM)&pos);
  387. switch (fr->Flags & (FR_DOWN|FR_MATCHCASE))
  388. {
  389. case FR_DOWN:
  390. if ( pos-pos_start == len && StrCmpNIW(fr->lpstrFindWhat, content+pos_start, len) == 0)
  391. SendMessageW(Globals.hEdit, EM_REPLACESEL, TRUE, (LPARAM)fr->lpstrReplaceWith);
  392. break;
  393. case FR_DOWN|FR_MATCHCASE:
  394. if ( pos-pos_start == len && StrCmpNW(fr->lpstrFindWhat, content+pos_start, len) == 0)
  395. SendMessageW(Globals.hEdit, EM_REPLACESEL, TRUE, (LPARAM)fr->lpstrReplaceWith);
  396. break;
  397. default: /* shouldn't happen */
  398. return;
  399. }
  400. HeapFree(GetProcessHeap(), 0, content);
  401. NOTEPAD_DoFind(fr);
  402. }
  403. static void NOTEPAD_DoReplaceAll(FINDREPLACEW *fr)
  404. {
  405. LPWSTR content;
  406. int len = lstrlenW(fr->lpstrFindWhat);
  407. int fileLen;
  408. SIZE_T pos;
  409. SendMessageW(Globals.hEdit, EM_SETSEL, 0, 0);
  410. while(TRUE){
  411. fileLen = GetWindowTextLengthW(Globals.hEdit) + 1;
  412. content = HeapAlloc(GetProcessHeap(), 0, fileLen * sizeof(WCHAR));
  413. if (!content) return;
  414. GetWindowTextW(Globals.hEdit, content, fileLen);
  415. SendMessageW(Globals.hEdit, EM_GETSEL, 0, (LPARAM)&pos);
  416. switch (fr->Flags & (FR_DOWN|FR_MATCHCASE))
  417. {
  418. case FR_DOWN:
  419. pos = StrStrIW(content+pos, fr->lpstrFindWhat) - content;
  420. if (pos == -(SIZE_T)content) pos = ~(SIZE_T)0;
  421. break;
  422. case FR_DOWN|FR_MATCHCASE:
  423. pos = StrStrW(content+pos, fr->lpstrFindWhat) - content;
  424. if (pos == -(SIZE_T)content) pos = ~(SIZE_T)0;
  425. break;
  426. default: /* shouldn't happen */
  427. return;
  428. }
  429. HeapFree(GetProcessHeap(), 0, content);
  430. if(pos == ~(SIZE_T)0)
  431. {
  432. SendMessageW(Globals.hEdit, EM_SETSEL, 0, 0);
  433. return;
  434. }
  435. SendMessageW(Globals.hEdit, EM_SETSEL, pos, pos + len);
  436. SendMessageW(Globals.hEdit, EM_REPLACESEL, TRUE, (LPARAM)fr->lpstrReplaceWith);
  437. }
  438. }
  439. LRESULT CALLBACK EDIT_CallBackProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam,
  440. UINT_PTR uIdSubclass, DWORD_PTR dwRefData)
  441. {
  442. switch (msg)
  443. {
  444. case WM_KEYDOWN:
  445. case WM_KEYUP:
  446. case WM_MBUTTONDOWN:
  447. case WM_MBUTTONUP:
  448. case WM_LBUTTONDOWN:
  449. case WM_LBUTTONUP:
  450. UpdateStatusBar();
  451. break;
  452. case WM_MOUSEMOVE:
  453. if(wParam == MK_LBUTTON)
  454. UpdateStatusBar();
  455. break;
  456. default:
  457. break;
  458. }
  459. return DefSubclassProc(hWnd, msg, wParam, lParam);
  460. }
  461. /***********************************************************************
  462. *
  463. * NOTEPAD_WndProc
  464. */
  465. static LRESULT WINAPI NOTEPAD_WndProc(HWND hWnd, UINT msg, WPARAM wParam,
  466. LPARAM lParam)
  467. {
  468. if (msg == aFINDMSGSTRING) /* not a constant so can't be used in switch */
  469. {
  470. FINDREPLACEW *fr = (FINDREPLACEW *)lParam;
  471. if (fr->Flags & FR_DIALOGTERM)
  472. Globals.hFindReplaceDlg = NULL;
  473. if (fr->Flags & FR_FINDNEXT)
  474. {
  475. Globals.lastFind = *fr;
  476. NOTEPAD_DoFind(fr);
  477. }
  478. if (fr->Flags & FR_REPLACE)
  479. {
  480. Globals.lastFind = *fr;
  481. NOTEPAD_DoReplace(fr);
  482. }
  483. if (fr->Flags & FR_REPLACEALL)
  484. {
  485. Globals.lastFind = *fr;
  486. NOTEPAD_DoReplaceAll(fr);
  487. }
  488. return 0;
  489. }
  490. switch (msg) {
  491. case WM_CREATE:
  492. {
  493. DWORD dwStyle = WS_CHILD | WS_VISIBLE | WS_BORDER | WS_VSCROLL |
  494. ES_AUTOVSCROLL | ES_MULTILINE | ES_NOHIDESEL;
  495. RECT rc;
  496. GetClientRect(hWnd, &rc);
  497. if (!Globals.bWrapLongLines) dwStyle |= WS_HSCROLL | ES_AUTOHSCROLL;
  498. Globals.hEdit = CreateWindowExW(WS_EX_CLIENTEDGE, L"edit", NULL,
  499. dwStyle, 0, 0, rc.right, rc.bottom, hWnd,
  500. NULL, Globals.hInstance, NULL);
  501. SetWindowSubclass(Globals.hEdit, EDIT_CallBackProc, 0, 0);
  502. Globals.hFont = CreateFontIndirectW(&Globals.lfFont);
  503. SendMessageW(Globals.hEdit, WM_SETFONT, (WPARAM)Globals.hFont, FALSE);
  504. SendMessageW(Globals.hEdit, EM_LIMITTEXT, 0, 0);
  505. Globals.hStatusBar = CreateWindowExW(0, STATUSCLASSNAMEW, NULL,
  506. WS_VISIBLE | WS_CHILD, 0, 0, 0, 0, hWnd, NULL,
  507. Globals.hInstance, NULL);
  508. LoadStringW(Globals.hInstance, STRING_STATUSBAR, (LPWSTR)&Globals.szStatusString, 0);
  509. Globals.lastLn = -1;
  510. Globals.lastCol = -1;
  511. UpdateStatusBar();
  512. break;
  513. }
  514. case WM_COMMAND:
  515. NOTEPAD_MenuCommand(LOWORD(wParam));
  516. break;
  517. case WM_DESTROYCLIPBOARD:
  518. /*MessageBoxW(Globals.hMainWnd, "Empty clipboard", "Debug", MB_ICONEXCLAMATION);*/
  519. break;
  520. case WM_CLOSE:
  521. if (DoCloseFile()) {
  522. DestroyWindow(hWnd);
  523. }
  524. break;
  525. case WM_QUERYENDSESSION:
  526. if (DoCloseFile()) {
  527. return 1;
  528. }
  529. break;
  530. case WM_DESTROY:
  531. NOTEPAD_SaveSettingToRegistry();
  532. PostQuitMessage(0);
  533. break;
  534. case WM_SIZE:
  535. updateWindowSize(LOWORD(lParam), HIWORD(lParam));
  536. break;
  537. case WM_SETFOCUS:
  538. SetFocus(Globals.hEdit);
  539. break;
  540. case WM_DROPFILES:
  541. {
  542. WCHAR szFileName[MAX_PATH];
  543. HANDLE hDrop = (HANDLE) wParam;
  544. DragQueryFileW(hDrop, 0, szFileName, ARRAY_SIZE(szFileName));
  545. DragFinish(hDrop);
  546. DoOpenFile(szFileName, ENCODING_AUTO);
  547. break;
  548. }
  549. case WM_INITMENUPOPUP:
  550. NOTEPAD_InitMenuPopup((HMENU)wParam, lParam);
  551. break;
  552. default:
  553. return DefWindowProcW(hWnd, msg, wParam, lParam);
  554. }
  555. return 0;
  556. }
  557. static int AlertFileDoesNotExist(LPCWSTR szFileName)
  558. {
  559. int nResult;
  560. WCHAR szMessage[MAX_STRING_LEN];
  561. WCHAR szResource[MAX_STRING_LEN];
  562. LoadStringW(Globals.hInstance, STRING_DOESNOTEXIST, szResource, ARRAY_SIZE(szResource));
  563. wsprintfW(szMessage, szResource, szFileName);
  564. LoadStringW(Globals.hInstance, STRING_ERROR, szResource, ARRAY_SIZE(szResource));
  565. nResult = MessageBoxW(Globals.hMainWnd, szMessage, szResource,
  566. MB_ICONEXCLAMATION | MB_YESNOCANCEL);
  567. return(nResult);
  568. }
  569. static void HandleCommandLine(LPWSTR cmdline)
  570. {
  571. WCHAR delimiter, *ptr;
  572. BOOL opt_print = FALSE;
  573. /* skip white space */
  574. while (*cmdline == ' ') cmdline++;
  575. /* skip executable name */
  576. delimiter = (*cmdline == '"' ? '"' : ' ');
  577. if (*cmdline == delimiter) cmdline++;
  578. while (*cmdline && *cmdline != delimiter) cmdline++;
  579. if (*cmdline == delimiter) cmdline++;
  580. while (*cmdline == ' ') cmdline++;
  581. ptr = cmdline;
  582. while (*ptr == ' ' || *ptr == '-' || *ptr == '/')
  583. {
  584. WCHAR option;
  585. if (*ptr++ == ' ') continue;
  586. option = *ptr;
  587. if (option) ptr++;
  588. while (*ptr == ' ') ptr++;
  589. switch(option)
  590. {
  591. case 'p':
  592. case 'P':
  593. {
  594. if (!opt_print)
  595. {
  596. opt_print = TRUE;
  597. cmdline = ptr;
  598. }
  599. break;
  600. }
  601. }
  602. }
  603. if (*cmdline)
  604. {
  605. /* file name is passed in the command line */
  606. LPCWSTR file_name;
  607. BOOL file_exists;
  608. WCHAR buf[MAX_PATH];
  609. if (cmdline[0] == '"')
  610. {
  611. WCHAR* wc;
  612. cmdline++;
  613. wc=cmdline;
  614. /* Note: Double-quotes are not allowed in Windows filenames */
  615. while (*wc && *wc != '"') wc++;
  616. /* On Windows notepad ignores further arguments too */
  617. *wc = 0;
  618. }
  619. if (FileExists(cmdline))
  620. {
  621. file_exists = TRUE;
  622. file_name = cmdline;
  623. }
  624. else
  625. {
  626. /* try to find file with ".txt" extension */
  627. if (wcschr(PathFindFileNameW(cmdline), '.'))
  628. {
  629. file_exists = FALSE;
  630. file_name = cmdline;
  631. }
  632. else
  633. {
  634. lstrcpynW(buf, cmdline, MAX_PATH - lstrlenW(L".txt") - 1);
  635. lstrcatW(buf, L".txt");
  636. file_name = buf;
  637. file_exists = FileExists(buf);
  638. }
  639. }
  640. if (file_exists)
  641. {
  642. DoOpenFile(file_name, ENCODING_AUTO);
  643. InvalidateRect(Globals.hMainWnd, NULL, FALSE);
  644. if (opt_print)
  645. DIALOG_FilePrint();
  646. }
  647. else
  648. {
  649. switch (AlertFileDoesNotExist(file_name)) {
  650. case IDYES:
  651. {
  652. HANDLE file;
  653. SetFileNameAndEncoding(file_name, ENCODING_ANSI);
  654. file = CreateFileW(file_name, GENERIC_WRITE, FILE_SHARE_WRITE,
  655. NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  656. if (file != INVALID_HANDLE_VALUE) CloseHandle(file);
  657. UpdateWindowCaption();
  658. break;
  659. }
  660. case IDNO:
  661. break;
  662. case IDCANCEL:
  663. DestroyWindow(Globals.hMainWnd);
  664. break;
  665. }
  666. }
  667. }
  668. }
  669. /***********************************************************************
  670. *
  671. * WinMain
  672. */
  673. int PASCAL WinMain(HINSTANCE hInstance, HINSTANCE prev, LPSTR cmdline, int show)
  674. {
  675. MSG msg;
  676. HACCEL hAccel;
  677. WNDCLASSEXW class;
  678. HMONITOR monitor;
  679. MONITORINFO info;
  680. INT x, y;
  681. InitCommonControls();
  682. aFINDMSGSTRING = RegisterWindowMessageW(FINDMSGSTRINGW);
  683. ZeroMemory(&Globals, sizeof(Globals));
  684. Globals.hInstance = hInstance;
  685. NOTEPAD_LoadSettingFromRegistry();
  686. ZeroMemory(&class, sizeof(class));
  687. class.cbSize = sizeof(class);
  688. class.lpfnWndProc = NOTEPAD_WndProc;
  689. class.hInstance = Globals.hInstance;
  690. class.hIcon = LoadIconW(Globals.hInstance, MAKEINTRESOURCEW(IDI_NOTEPAD));
  691. class.hIconSm = LoadImageW(Globals.hInstance, MAKEINTRESOURCEW(IDI_NOTEPAD), IMAGE_ICON,
  692. GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON),
  693. LR_SHARED);
  694. class.hCursor = LoadCursorW(0, (LPCWSTR)IDC_ARROW);
  695. class.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
  696. class.lpszMenuName = MAKEINTRESOURCEW(MAIN_MENU);
  697. class.lpszClassName = L"Notepad";
  698. if (!RegisterClassExW(&class)) return FALSE;
  699. /* Setup windows */
  700. monitor = MonitorFromRect( &main_rect, MONITOR_DEFAULTTOPRIMARY );
  701. info.cbSize = sizeof(info);
  702. GetMonitorInfoW( monitor, &info );
  703. x = main_rect.left;
  704. y = main_rect.top;
  705. if (main_rect.left >= info.rcWork.right ||
  706. main_rect.top >= info.rcWork.bottom ||
  707. main_rect.right < info.rcWork.left ||
  708. main_rect.bottom < info.rcWork.top)
  709. x = y = CW_USEDEFAULT;
  710. Globals.hMainWnd =
  711. CreateWindowW(L"Notepad", L"Notepad", WS_OVERLAPPEDWINDOW, x, y,
  712. main_rect.right - main_rect.left, main_rect.bottom - main_rect.top,
  713. NULL, NULL, Globals.hInstance, NULL);
  714. if (!Globals.hMainWnd)
  715. {
  716. ShowLastError();
  717. ExitProcess(1);
  718. }
  719. NOTEPAD_InitData();
  720. DIALOG_FileNew();
  721. ShowWindow(Globals.hMainWnd, show);
  722. UpdateWindow(Globals.hMainWnd);
  723. DragAcceptFiles(Globals.hMainWnd, TRUE);
  724. HandleCommandLine(GetCommandLineW());
  725. hAccel = LoadAcceleratorsW(hInstance, MAKEINTRESOURCEW(ID_ACCEL));
  726. while (GetMessageW(&msg, 0, 0, 0))
  727. {
  728. if (!IsDialogMessageW(Globals.hFindReplaceDlg, &msg) && !TranslateAcceleratorW(Globals.hMainWnd, hAccel, &msg))
  729. {
  730. TranslateMessage(&msg);
  731. DispatchMessageW(&msg);
  732. }
  733. }
  734. return msg.wParam;
  735. }