xref: /reactos/base/applications/wordpad/wordpad.c (revision 84ccccab)
1 /*
2  * Wordpad implementation
3  *
4  * Copyright 2004 by Krzysztof Foltman
5  * Copyright 2007-2008 by Alexander N. Sørnes <alex@thehandofagony.com>
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21 
22 #define WIN32_LEAN_AND_MEAN
23 #define _WIN32_IE 0x0400
24 
25 #include <stdio.h>
26 #include <assert.h>
27 #include <windef.h>
28 #include <winbase.h>
29 #include <wingdi.h>
30 #include <winuser.h>
31 #include <richedit.h>
32 #include <commctrl.h>
33 #include <commdlg.h>
34 #include <shellapi.h>
35 #include <wine/unicode.h>
36 
37 #include "wordpad.h"
38 
39 #ifdef NONAMELESSUNION
40 # define U(x)  (x).u
41 # define U2(x) (x).u2
42 # define U3(x) (x).u3
43 #else
44 # define U(x)  (x)
45 # define U2(x) (x)
46 # define U3(x) (x)
47 #endif
48 
49 /* use LoadString */
50 static const WCHAR wszAppTitle[] = {'W','o','r','d','p','a','d',0};
51 
52 static const WCHAR wszMainWndClass[] = {'W','O','R','D','P','A','D','T','O','P',0};
53 
54 static const WCHAR stringFormat[] = {'%','2','d','\0'};
55 
56 const WCHAR wszPreviewWndClass[] = {'P','r','t','P','r','e','v','i','e','w',0};
57 LRESULT CALLBACK preview_proc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam);
58 
59 static HWND hMainWnd;
60 static HWND hEditorWnd;
61 static HWND hFindWnd;
62 static HMENU hColorPopupMenu;
63 
64 static UINT ID_FINDMSGSTRING;
65 
66 static DWORD wordWrap[2];
67 static DWORD barState[2];
68 static WPARAM fileFormat = SF_RTF;
69 
70 static WCHAR wszFileName[MAX_PATH];
71 static WCHAR wszFilter[MAX_STRING_LEN*4+6*3+5];
72 static WCHAR wszDefaultFileName[MAX_STRING_LEN];
73 static WCHAR wszSaveChanges[MAX_STRING_LEN];
74 static WCHAR units_cmW[MAX_STRING_LEN];
75 static WCHAR units_inW[MAX_STRING_LEN];
76 static WCHAR units_inchW[MAX_STRING_LEN];
77 static WCHAR units_ptW[MAX_STRING_LEN];
78 
79 static LRESULT OnSize( HWND hWnd, WPARAM wParam, LPARAM lParam );
80 
81 typedef enum
82 {
83     UNIT_CM,
84     UNIT_INCH,
85     UNIT_PT
86 } UNIT;
87 
88 typedef struct
89 {
90     int endPos;
91     BOOL wrapped;
92     WCHAR findBuffer[128];
93 } FINDREPLACE_custom;
94 
95 /* Load string resources */
96 static void DoLoadStrings(void)
97 {
98     LPWSTR p = wszFilter;
99     static const WCHAR files_rtf[] = {'*','.','r','t','f','\0'};
100     static const WCHAR files_txt[] = {'*','.','t','x','t','\0'};
101     static const WCHAR files_all[] = {'*','.','*','\0'};
102 
103     HINSTANCE hInstance = GetModuleHandleW(0);
104 
105     LoadStringW(hInstance, STRING_RICHTEXT_FILES_RTF, p, MAX_STRING_LEN);
106     p += lstrlenW(p) + 1;
107     lstrcpyW(p, files_rtf);
108     p += lstrlenW(p) + 1;
109     LoadStringW(hInstance, STRING_TEXT_FILES_TXT, p, MAX_STRING_LEN);
110     p += lstrlenW(p) + 1;
111     lstrcpyW(p, files_txt);
112     p += lstrlenW(p) + 1;
113     LoadStringW(hInstance, STRING_TEXT_FILES_UNICODE_TXT, p, MAX_STRING_LEN);
114     p += lstrlenW(p) + 1;
115     lstrcpyW(p, files_txt);
116     p += lstrlenW(p) + 1;
117     LoadStringW(hInstance, STRING_ALL_FILES, p, MAX_STRING_LEN);
118     p += lstrlenW(p) + 1;
119     lstrcpyW(p, files_all);
120     p += lstrlenW(p) + 1;
121     *p = '\0';
122 
123     p = wszDefaultFileName;
124     LoadStringW(hInstance, STRING_DEFAULT_FILENAME, p, MAX_STRING_LEN);
125 
126     p = wszSaveChanges;
127     LoadStringW(hInstance, STRING_PROMPT_SAVE_CHANGES, p, MAX_STRING_LEN);
128 
129     LoadStringW(hInstance, STRING_UNITS_CM, units_cmW, MAX_STRING_LEN);
130     LoadStringW(hInstance, STRING_UNITS_IN, units_inW, MAX_STRING_LEN);
131     LoadStringW(hInstance, STRING_UNITS_INCH, units_inchW, MAX_STRING_LEN);
132     LoadStringW(hInstance, STRING_UNITS_PT, units_ptW, MAX_STRING_LEN);
133 }
134 
135 /* Show a message box with resource strings */
136 static int MessageBoxWithResStringW(HWND hWnd, LPCWSTR lpText, LPCWSTR lpCaption, UINT uType)
137 {
138     MSGBOXPARAMSW params;
139 
140     params.cbSize             = sizeof(params);
141     params.hwndOwner          = hWnd;
142     params.hInstance          = GetModuleHandleW(0);
143     params.lpszText           = lpText;
144     params.lpszCaption        = lpCaption;
145     params.dwStyle            = uType;
146     params.lpszIcon           = NULL;
147     params.dwContextHelpId    = 0;
148     params.lpfnMsgBoxCallback = NULL;
149     params.dwLanguageId       = 0;
150     return MessageBoxIndirectW(&params);
151 }
152 
153 
154 static void AddButton(HWND hwndToolBar, int nImage, int nCommand)
155 {
156     TBBUTTON button;
157 
158     ZeroMemory(&button, sizeof(button));
159     button.iBitmap = nImage;
160     button.idCommand = nCommand;
161     button.fsState = TBSTATE_ENABLED;
162     button.fsStyle = BTNS_BUTTON;
163     button.dwData = 0;
164     button.iString = -1;
165     SendMessageW(hwndToolBar, TB_ADDBUTTONSW, 1, (LPARAM)&button);
166 }
167 
168 static void AddSeparator(HWND hwndToolBar)
169 {
170     TBBUTTON button;
171 
172     ZeroMemory(&button, sizeof(button));
173     button.iBitmap = -1;
174     button.idCommand = 0;
175     button.fsState = 0;
176     button.fsStyle = BTNS_SEP;
177     button.dwData = 0;
178     button.iString = -1;
179     SendMessageW(hwndToolBar, TB_ADDBUTTONSW, 1, (LPARAM)&button);
180 }
181 
182 static DWORD CALLBACK stream_in(DWORD_PTR cookie, LPBYTE buffer, LONG cb, LONG *pcb)
183 {
184     HANDLE hFile = (HANDLE)cookie;
185     DWORD read;
186 
187     if(!ReadFile(hFile, buffer, cb, &read, 0))
188         return 1;
189 
190     *pcb = read;
191 
192     return 0;
193 }
194 
195 static DWORD CALLBACK stream_out(DWORD_PTR cookie, LPBYTE buffer, LONG cb, LONG *pcb)
196 {
197     DWORD written;
198     int ret;
199     HANDLE hFile = (HANDLE)cookie;
200 
201     ret = WriteFile(hFile, buffer, cb, &written, 0);
202 
203     if(!ret || (cb != written))
204         return 1;
205 
206     *pcb = cb;
207 
208     return 0;
209 }
210 
211 LPWSTR file_basename(LPWSTR path)
212 {
213     LPWSTR pos = path + lstrlenW(path);
214 
215     while(pos > path)
216     {
217         if(*pos == '\\' || *pos == '/')
218         {
219             pos++;
220             break;
221         }
222         pos--;
223     }
224     return pos;
225 }
226 
227 static void set_caption(LPCWSTR wszNewFileName)
228 {
229     static const WCHAR wszSeparator[] = {' ','-',' '};
230     WCHAR *wszCaption;
231     SIZE_T length = 0;
232 
233     if(!wszNewFileName)
234         wszNewFileName = wszDefaultFileName;
235     else
236         wszNewFileName = file_basename((LPWSTR)wszNewFileName);
237 
238     wszCaption = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
239                 lstrlenW(wszNewFileName)*sizeof(WCHAR)+sizeof(wszSeparator)+sizeof(wszAppTitle));
240 
241     if(!wszCaption)
242         return;
243 
244     memcpy(wszCaption, wszNewFileName, lstrlenW(wszNewFileName)*sizeof(WCHAR));
245     length += lstrlenW(wszNewFileName);
246     memcpy(wszCaption + length, wszSeparator, sizeof(wszSeparator));
247     length += sizeof(wszSeparator) / sizeof(WCHAR);
248     memcpy(wszCaption + length, wszAppTitle, sizeof(wszAppTitle));
249 
250     SetWindowTextW(hMainWnd, wszCaption);
251 
252     HeapFree(GetProcessHeap(), 0, wszCaption);
253 }
254 
255 static BOOL validate_endptr(LPCWSTR endptr, UNIT *punit)
256 {
257     if(punit != NULL)
258         *punit = UNIT_CM;
259     if(!endptr)
260         return FALSE;
261     if(!*endptr)
262         return TRUE;
263 
264     while(*endptr == ' ')
265         endptr++;
266 
267     if(punit == NULL)
268         return *endptr == '\0';
269 
270     if(!lstrcmpW(endptr, units_cmW))
271     {
272         *punit = UNIT_CM;
273         endptr += lstrlenW(units_cmW);
274     }
275     else if (!lstrcmpW(endptr, units_inW))
276     {
277         *punit = UNIT_INCH;
278         endptr += lstrlenW(units_inW);
279     }
280     else if (!lstrcmpW(endptr, units_inchW))
281     {
282         *punit = UNIT_INCH;
283         endptr += lstrlenW(units_inchW);
284     }
285     else if (!lstrcmpW(endptr, units_ptW))
286     {
287         *punit = UNIT_PT;
288         endptr += lstrlenW(units_ptW);
289     }
290 
291     return *endptr == '\0';
292 }
293 
294 static BOOL number_from_string(LPCWSTR string, float *num, UNIT *punit)
295 {
296     double ret;
297     WCHAR *endptr;
298 
299     *num = 0;
300     errno = 0;
301     ret = wcstod(string, &endptr);
302 
303     if (punit != NULL)
304         *punit = UNIT_CM;
305     if((ret == 0 && errno != 0) || endptr == string || !validate_endptr(endptr, punit))
306     {
307         return FALSE;
308     } else
309     {
310         *num = (float)ret;
311         return TRUE;
312     }
313 }
314 
315 static void set_size(float size)
316 {
317     CHARFORMAT2W fmt;
318 
319     ZeroMemory(&fmt, sizeof(fmt));
320     fmt.cbSize = sizeof(fmt);
321     fmt.dwMask = CFM_SIZE;
322     fmt.yHeight = (int)(size * 20.0);
323     SendMessageW(hEditorWnd, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
324 }
325 
326 static void on_sizelist_modified(HWND hwndSizeList, LPWSTR wszNewFontSize)
327 {
328     WCHAR sizeBuffer[MAX_STRING_LEN];
329     CHARFORMAT2W format;
330 
331     ZeroMemory(&format, sizeof(format));
332     format.cbSize = sizeof(format);
333     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&format);
334 
335     wsprintfW(sizeBuffer, stringFormat, format.yHeight / 20);
336     if(lstrcmpW(sizeBuffer, wszNewFontSize))
337     {
338         float size = 0;
339         if(number_from_string(wszNewFontSize, &size, NULL)
340            && size > 0)
341         {
342             set_size(size);
343         } else
344         {
345             SetWindowTextW(hwndSizeList, sizeBuffer);
346             MessageBoxWithResStringW(hMainWnd, MAKEINTRESOURCEW(STRING_INVALID_NUMBER),
347                         wszAppTitle, MB_OK | MB_ICONINFORMATION);
348         }
349     }
350 }
351 
352 static void add_size(HWND hSizeListWnd, unsigned size)
353 {
354     WCHAR buffer[3];
355     COMBOBOXEXITEMW cbItem;
356     cbItem.mask = CBEIF_TEXT;
357     cbItem.iItem = -1;
358 
359     wsprintfW(buffer, stringFormat, size);
360     cbItem.pszText = buffer;
361     SendMessageW(hSizeListWnd, CBEM_INSERTITEMW, 0, (LPARAM)&cbItem);
362 }
363 
364 static void populate_size_list(HWND hSizeListWnd)
365 {
366     HWND hReBarWnd = GetDlgItem(hMainWnd, IDC_REBAR);
367     HWND hFontListWnd = GetDlgItem(hReBarWnd, IDC_FONTLIST);
368     COMBOBOXEXITEMW cbFontItem;
369     CHARFORMAT2W fmt;
370     HWND hListEditWnd = (HWND)SendMessageW(hSizeListWnd, CBEM_GETEDITCONTROL, 0, 0);
371     HDC hdc = GetDC(hMainWnd);
372     static const unsigned choices[] = {8,9,10,11,12,14,16,18,20,22,24,26,28,36,48,72};
373     WCHAR buffer[3];
374     size_t i;
375     DWORD fontStyle;
376 
377     ZeroMemory(&fmt, sizeof(fmt));
378     fmt.cbSize = sizeof(fmt);
379     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
380 
381     cbFontItem.mask = CBEIF_LPARAM;
382     cbFontItem.iItem = SendMessageW(hFontListWnd, CB_FINDSTRINGEXACT, -1, (LPARAM)fmt.szFaceName);
383     SendMessageW(hFontListWnd, CBEM_GETITEMW, 0, (LPARAM)&cbFontItem);
384 
385     fontStyle = (DWORD)LOWORD(cbFontItem.lParam);
386 
387     SendMessageW(hSizeListWnd, CB_RESETCONTENT, 0, 0);
388 
389     if((fontStyle & RASTER_FONTTYPE) && cbFontItem.iItem)
390     {
391         add_size(hSizeListWnd, (BYTE)MulDiv(HIWORD(cbFontItem.lParam), 72,
392                                GetDeviceCaps(hdc, LOGPIXELSY)));
393     } else
394     {
395         for(i = 0; i < sizeof(choices)/sizeof(choices[0]); i++)
396             add_size(hSizeListWnd, choices[i]);
397     }
398 
399     wsprintfW(buffer, stringFormat, fmt.yHeight / 20);
400     SendMessageW(hListEditWnd, WM_SETTEXT, 0, (LPARAM)buffer);
401 }
402 
403 static void update_size_list(void)
404 {
405     HWND hReBar = GetDlgItem(hMainWnd, IDC_REBAR);
406     HWND hwndSizeList = GetDlgItem(hReBar, IDC_SIZELIST);
407     HWND hwndSizeListEdit = (HWND)SendMessageW(hwndSizeList, CBEM_GETEDITCONTROL, 0, 0);
408     WCHAR fontSize[MAX_STRING_LEN], sizeBuffer[MAX_STRING_LEN];
409     CHARFORMAT2W fmt;
410 
411     ZeroMemory(&fmt, sizeof(fmt));
412     fmt.cbSize = sizeof(fmt);
413 
414     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
415 
416     SendMessageW(hwndSizeListEdit, WM_GETTEXT, MAX_PATH, (LPARAM)fontSize);
417     wsprintfW(sizeBuffer, stringFormat, fmt.yHeight / 20);
418 
419     if(lstrcmpW(fontSize, sizeBuffer))
420         SendMessageW(hwndSizeListEdit, WM_SETTEXT, 0, (LPARAM)sizeBuffer);
421 }
422 
423 static void update_font_list(void)
424 {
425     HWND hReBar = GetDlgItem(hMainWnd, IDC_REBAR);
426     HWND hFontList = GetDlgItem(hReBar, IDC_FONTLIST);
427     HWND hFontListEdit = (HWND)SendMessageW(hFontList, CBEM_GETEDITCONTROL, 0, 0);
428     WCHAR fontName[MAX_STRING_LEN];
429     CHARFORMAT2W fmt;
430 
431     ZeroMemory(&fmt, sizeof(fmt));
432     fmt.cbSize = sizeof(fmt);
433 
434     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
435     if (!SendMessageW(hFontListEdit, WM_GETTEXT, MAX_PATH, (LPARAM)fontName)) return;
436 
437     if(lstrcmpW(fontName, fmt.szFaceName))
438     {
439         SendMessageW(hFontListEdit, WM_SETTEXT, 0, (LPARAM)fmt.szFaceName);
440         populate_size_list(GetDlgItem(hReBar, IDC_SIZELIST));
441     } else
442     {
443         update_size_list();
444     }
445 }
446 
447 static void clear_formatting(void)
448 {
449     PARAFORMAT2 pf;
450 
451     pf.cbSize = sizeof(pf);
452     pf.dwMask = PFM_ALIGNMENT;
453     pf.wAlignment = PFA_LEFT;
454     SendMessageW(hEditorWnd, EM_SETPARAFORMAT, 0, (LPARAM)&pf);
455 }
456 
457 static int fileformat_number(WPARAM format)
458 {
459     int number = 0;
460 
461     if(format == SF_TEXT)
462     {
463         number = 1;
464     } else if (format == (SF_TEXT | SF_UNICODE))
465     {
466         number = 2;
467     }
468     return number;
469 }
470 
471 static WPARAM fileformat_flags(int format)
472 {
473     WPARAM flags[] = { SF_RTF , SF_TEXT , SF_TEXT | SF_UNICODE };
474 
475     return flags[format];
476 }
477 
478 static void set_font(LPCWSTR wszFaceName)
479 {
480     HWND hReBarWnd = GetDlgItem(hMainWnd, IDC_REBAR);
481     HWND hSizeListWnd = GetDlgItem(hReBarWnd, IDC_SIZELIST);
482     HWND hFontListWnd = GetDlgItem(hReBarWnd, IDC_FONTLIST);
483     HWND hFontListEditWnd = (HWND)SendMessageW(hFontListWnd, CBEM_GETEDITCONTROL, 0, 0);
484     CHARFORMAT2W fmt;
485 
486     ZeroMemory(&fmt, sizeof(fmt));
487 
488     fmt.cbSize = sizeof(fmt);
489     fmt.dwMask = CFM_FACE;
490 
491     lstrcpyW(fmt.szFaceName, wszFaceName);
492 
493     SendMessageW(hEditorWnd, EM_SETCHARFORMAT,  SCF_SELECTION, (LPARAM)&fmt);
494 
495     populate_size_list(hSizeListWnd);
496 
497     SendMessageW(hFontListEditWnd, WM_SETTEXT, 0, (LPARAM)wszFaceName);
498 }
499 
500 static void set_default_font(void)
501 {
502     static const WCHAR richTextFont[] = {'T','i','m','e','s',' ','N','e','w',' ',
503                                          'R','o','m','a','n',0};
504     static const WCHAR plainTextFont[] = {'C','o','u','r','i','e','r',' ','N','e','w',0};
505     CHARFORMAT2W fmt;
506     LPCWSTR font;
507 
508     ZeroMemory(&fmt, sizeof(fmt));
509 
510     fmt.cbSize = sizeof(fmt);
511     fmt.dwMask = CFM_FACE | CFM_BOLD | CFM_ITALIC | CFM_UNDERLINE;
512     fmt.dwEffects = 0;
513 
514     if(fileFormat & SF_RTF)
515         font = richTextFont;
516     else
517         font = plainTextFont;
518 
519     lstrcpyW(fmt.szFaceName, font);
520 
521     SendMessageW(hEditorWnd, EM_SETCHARFORMAT,  SCF_DEFAULT, (LPARAM)&fmt);
522 }
523 
524 static void on_fontlist_modified(LPWSTR wszNewFaceName)
525 {
526     CHARFORMAT2W format;
527     ZeroMemory(&format, sizeof(format));
528     format.cbSize = sizeof(format);
529     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&format);
530 
531     if(lstrcmpW(format.szFaceName, wszNewFaceName))
532         set_font(wszNewFaceName);
533 }
534 
535 static void add_font(LPCWSTR fontName, DWORD fontType, HWND hListWnd, const NEWTEXTMETRICEXW *ntmc)
536 {
537     COMBOBOXEXITEMW cbItem;
538     WCHAR buffer[MAX_PATH];
539     int fontHeight = 0;
540 
541     cbItem.mask = CBEIF_TEXT;
542     cbItem.pszText = buffer;
543     cbItem.cchTextMax = MAX_STRING_LEN;
544     cbItem.iItem = 0;
545 
546     while(SendMessageW(hListWnd, CBEM_GETITEMW, 0, (LPARAM)&cbItem))
547     {
548         if(lstrcmpiW(cbItem.pszText, fontName) <= 0)
549             cbItem.iItem++;
550         else
551             break;
552     }
553     cbItem.pszText = HeapAlloc( GetProcessHeap(), 0, (lstrlenW(fontName) + 1)*sizeof(WCHAR) );
554     lstrcpyW( cbItem.pszText, fontName );
555 
556     cbItem.mask |= CBEIF_LPARAM;
557     if(fontType & RASTER_FONTTYPE)
558         fontHeight = ntmc->ntmTm.tmHeight - ntmc->ntmTm.tmInternalLeading;
559 
560     cbItem.lParam = MAKELONG(fontType,fontHeight);
561     SendMessageW(hListWnd, CBEM_INSERTITEMW, 0, (LPARAM)&cbItem);
562     HeapFree( GetProcessHeap(), 0, cbItem.pszText );
563 }
564 
565 static void dialog_choose_font(void)
566 {
567     CHOOSEFONTW cf;
568     LOGFONTW lf;
569     CHARFORMAT2W fmt;
570     HDC hDC = GetDC(hMainWnd);
571 
572     ZeroMemory(&cf, sizeof(cf));
573     cf.lStructSize = sizeof(cf);
574     cf.hwndOwner = hMainWnd;
575     cf.lpLogFont = &lf;
576     cf.Flags = CF_SCREENFONTS | CF_NOSCRIPTSEL | CF_INITTOLOGFONTSTRUCT | CF_EFFECTS | CF_NOVERTFONTS;
577 
578     ZeroMemory(&fmt, sizeof(fmt));
579     fmt.cbSize = sizeof(fmt);
580 
581     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
582     lstrcpyW(cf.lpLogFont->lfFaceName, fmt.szFaceName);
583     cf.lpLogFont->lfItalic = (fmt.dwEffects & CFE_ITALIC) != 0;
584     cf.lpLogFont->lfWeight = (fmt.dwEffects & CFE_BOLD) ? FW_BOLD : FW_NORMAL;
585     cf.lpLogFont->lfUnderline = (fmt.dwEffects & CFE_UNDERLINE) != 0;
586     cf.lpLogFont->lfStrikeOut = (fmt.dwEffects & CFE_STRIKEOUT) != 0;
587     cf.lpLogFont->lfHeight = -MulDiv(fmt.yHeight / 20, GetDeviceCaps(hDC, LOGPIXELSY), 72);
588     cf.rgbColors = fmt.crTextColor;
589 
590     if(ChooseFontW(&cf))
591     {
592         ZeroMemory(&fmt, sizeof(fmt));
593         fmt.cbSize = sizeof(fmt);
594         fmt.dwMask = CFM_BOLD | CFM_ITALIC | CFM_SIZE | CFM_UNDERLINE | CFM_STRIKEOUT | CFM_COLOR;
595         fmt.yHeight = cf.iPointSize * 2;
596 
597         if(cf.nFontType & BOLD_FONTTYPE)
598             fmt.dwEffects |= CFE_BOLD;
599         if(cf.nFontType & ITALIC_FONTTYPE)
600             fmt.dwEffects |= CFE_ITALIC;
601         if(cf.lpLogFont->lfUnderline)
602             fmt.dwEffects |= CFE_UNDERLINE;
603         if(cf.lpLogFont->lfStrikeOut)
604             fmt.dwEffects |= CFE_STRIKEOUT;
605 
606         fmt.crTextColor = cf.rgbColors;
607 
608         SendMessageW(hEditorWnd, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
609         set_font(cf.lpLogFont->lfFaceName);
610     }
611 }
612 
613 
614 static int CALLBACK enum_font_proc(const LOGFONTW *lpelfe, const TEXTMETRICW *lpntme,
615                             DWORD FontType, LPARAM lParam)
616 {
617     HWND hListWnd = (HWND) lParam;
618 
619     if (lpelfe->lfFaceName[0] == '@') return 1;  /* ignore vertical fonts */
620 
621     if(SendMessageW(hListWnd, CB_FINDSTRINGEXACT, -1, (LPARAM)lpelfe->lfFaceName) == CB_ERR)
622     {
623 
624         add_font(lpelfe->lfFaceName, FontType, hListWnd, (const NEWTEXTMETRICEXW*)lpntme);
625     }
626 
627     return 1;
628 }
629 
630 static void populate_font_list(HWND hListWnd)
631 {
632     HDC hdc = GetDC(hMainWnd);
633     LOGFONTW fontinfo;
634     HWND hListEditWnd = (HWND)SendMessageW(hListWnd, CBEM_GETEDITCONTROL, 0, 0);
635     CHARFORMAT2W fmt;
636 
637     fontinfo.lfCharSet = DEFAULT_CHARSET;
638     *fontinfo.lfFaceName = '\0';
639     fontinfo.lfPitchAndFamily = 0;
640 
641     EnumFontFamiliesExW(hdc, &fontinfo, enum_font_proc,
642                         (LPARAM)hListWnd, 0);
643 
644     ZeroMemory(&fmt, sizeof(fmt));
645     fmt.cbSize = sizeof(fmt);
646     SendMessageW(hEditorWnd, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM)&fmt);
647     SendMessageW(hListEditWnd, WM_SETTEXT, 0, (LPARAM)fmt.szFaceName);
648 }
649 
650 static void update_window(void)
651 {
652     RECT rect;
653 
654     GetClientRect(hMainWnd, &rect);
655 
656     OnSize(hMainWnd, SIZE_RESTORED, MAKELPARAM(rect.right, rect.bottom));
657 }
658 
659 static BOOL is_bar_visible(int bandId)
660 {
661     return barState[reg_formatindex(fileFormat)] & (1 << bandId);
662 }
663 
664 static void store_bar_state(int bandId, BOOL show)
665 {
666     int formatIndex = reg_formatindex(fileFormat);
667 
668     if(show)
669         barState[formatIndex] |= (1 << bandId);
670     else
671         barState[formatIndex] &= ~(1 << bandId);
672 }
673 
674 static void set_toolbar_state(int bandId, BOOL show)
675 {
676     HWND hwndReBar = GetDlgItem(hMainWnd, IDC_REBAR);
677 
678     SendMessageW(hwndReBar, RB_SHOWBAND, SendMessageW(hwndReBar, RB_IDTOINDEX, bandId, 0), show);
679 
680     if(bandId == BANDID_TOOLBAR)
681     {
682         REBARBANDINFOW rbbinfo;
683         int index = SendMessageW(hwndReBar, RB_IDTOINDEX, BANDID_FONTLIST, 0);
684 
685         rbbinfo.cbSize = REBARBANDINFOW_V6_SIZE;
686         rbbinfo.fMask = RBBIM_STYLE;
687 
688         SendMessageW(hwndReBar, RB_GETBANDINFOW, index, (LPARAM)&rbbinfo);
689 
690         if(!show)
691             rbbinfo.fStyle &= ~RBBS_BREAK;
692         else
693             rbbinfo.fStyle |= RBBS_BREAK;
694 
695         SendMessageW(hwndReBar, RB_SETBANDINFOW, index, (LPARAM)&rbbinfo);
696     }
697 
698     if(bandId == BANDID_TOOLBAR || bandId == BANDID_FORMATBAR || bandId == BANDID_RULER)
699         store_bar_state(bandId, show);
700 }
701 
702 static void set_statusbar_state(BOOL show)
703 {
704     HWND hStatusWnd = GetDlgItem(hMainWnd, IDC_STATUSBAR);
705 
706     ShowWindow(hStatusWnd, show ? SW_SHOW : SW_HIDE);
707     store_bar_state(BANDID_STATUSBAR, show);
708 }
709 
710 static void set_bar_states(void)
711 {
712     set_toolbar_state(BANDID_TOOLBAR, is_bar_visible(BANDID_TOOLBAR));
713     set_toolbar_state(BANDID_FONTLIST, is_bar_visible(BANDID_FORMATBAR));
714     set_toolbar_state(BANDID_SIZELIST, is_bar_visible(BANDID_FORMATBAR));
715     set_toolbar_state(BANDID_FORMATBAR, is_bar_visible(BANDID_FORMATBAR));
716     set_toolbar_state(BANDID_RULER, is_bar_visible(BANDID_RULER));
717     set_statusbar_state(is_bar_visible(BANDID_STATUSBAR));
718 
719     update_window();
720 }
721 
722 static void preview_exit(HWND hMainWnd)
723 {
724     HMENU hMenu = LoadMenuW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDM_MAINMENU));
725     HWND hEditorWnd = GetDlgItem(hMainWnd, IDC_EDITOR);
726 
727     set_bar_states();
728     ShowWindow(hEditorWnd, TRUE);
729 
730     close_preview(hMainWnd);
731 
732     SetMenu(hMainWnd, hMenu);
733     registry_read_filelist(hMainWnd);
734 
735     update_window();
736 }
737 
738 static void set_fileformat(WPARAM format)
739 {
740     fileFormat = format;
741 
742     set_bar_states();
743     set_default_font();
744     target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
745 }
746 
747 static void ShowOpenError(DWORD Code)
748 {
749     LPWSTR Message;
750 
751     switch(Code)
752     {
753         case ERROR_ACCESS_DENIED:
754             Message = MAKEINTRESOURCEW(STRING_OPEN_ACCESS_DENIED);
755             break;
756 
757         default:
758             Message = MAKEINTRESOURCEW(STRING_OPEN_FAILED);
759     }
760     MessageBoxW(hMainWnd, Message, wszAppTitle, MB_ICONEXCLAMATION | MB_OK);
761 }
762 
763 static void DoOpenFile(LPCWSTR szOpenFileName)
764 {
765     HANDLE hFile;
766     EDITSTREAM es;
767     char fileStart[5];
768     DWORD readOut;
769     WPARAM format = SF_TEXT;
770 
771     hFile = CreateFileW(szOpenFileName, GENERIC_READ, FILE_SHARE_READ, NULL,
772                         OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
773     if (hFile == INVALID_HANDLE_VALUE)
774     {
775         ShowOpenError(GetLastError());
776         return;
777     }
778 
779     ReadFile(hFile, fileStart, 5, &readOut, NULL);
780     SetFilePointer(hFile, 0, NULL, FILE_BEGIN);
781 
782     if(readOut >= 2 && (BYTE)fileStart[0] == 0xff && (BYTE)fileStart[1] == 0xfe)
783     {
784         format = SF_TEXT | SF_UNICODE;
785         SetFilePointer(hFile, 2, NULL, FILE_BEGIN);
786     } else if(readOut >= 5)
787     {
788         static const char header[] = "{\\rtf";
789         static const BYTE STG_magic[] = { 0xd0,0xcf,0x11,0xe0 };
790 
791         if(!memcmp(header, fileStart, 5))
792             format = SF_RTF;
793         else if (!memcmp(STG_magic, fileStart, sizeof(STG_magic)))
794         {
795             CloseHandle(hFile);
796             MessageBoxWithResStringW(hMainWnd, MAKEINTRESOURCEW(STRING_OLE_STORAGE_NOT_SUPPORTED),
797                     wszAppTitle, MB_OK | MB_ICONEXCLAMATION);
798             return;
799         }
800     }
801 
802     es.dwCookie = (DWORD_PTR)hFile;
803     es.pfnCallback = stream_in;
804 
805     clear_formatting();
806     set_fileformat(format);
807     SendMessageW(hEditorWnd, EM_STREAMIN, format, (LPARAM)&es);
808 
809     CloseHandle(hFile);
810 
811     SetFocus(hEditorWnd);
812 
813     set_caption(szOpenFileName);
814 
815     lstrcpyW(wszFileName, szOpenFileName);
816     SendMessageW(hEditorWnd, EM_SETMODIFY, FALSE, 0);
817     registry_set_filelist(szOpenFileName, hMainWnd);
818     update_font_list();
819 }
820 
821 static void ShowWriteError(DWORD Code)
822 {
823     LPWSTR Message;
824 
825     switch(Code)
826     {
827         case ERROR_ACCESS_DENIED:
828             Message = MAKEINTRESOURCEW(STRING_WRITE_ACCESS_DENIED);
829             break;
830 
831         default:
832             Message = MAKEINTRESOURCEW(STRING_WRITE_FAILED);
833     }
834     MessageBoxW(hMainWnd, Message, wszAppTitle, MB_ICONEXCLAMATION | MB_OK);
835 }
836 
837 static BOOL DoSaveFile(LPCWSTR wszSaveFileName, WPARAM format)
838 {
839     HANDLE hFile;
840     EDITSTREAM stream;
841     LRESULT ret;
842 
843     hFile = CreateFileW(wszSaveFileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
844         FILE_ATTRIBUTE_NORMAL, NULL);
845 
846     if(hFile == INVALID_HANDLE_VALUE)
847     {
848         ShowWriteError(GetLastError());
849         return FALSE;
850     }
851 
852     if(format == (SF_TEXT | SF_UNICODE))
853     {
854         static const BYTE unicode[] = {0xff,0xfe};
855         DWORD writeOut;
856         WriteFile(hFile, &unicode, sizeof(unicode), &writeOut, 0);
857 
858         if(writeOut != sizeof(unicode))
859         {
860             CloseHandle(hFile);
861             return FALSE;
862         }
863     }
864 
865     stream.dwCookie = (DWORD_PTR)hFile;
866     stream.pfnCallback = stream_out;
867 
868     ret = SendMessageW(hEditorWnd, EM_STREAMOUT, format, (LPARAM)&stream);
869 
870     CloseHandle(hFile);
871 
872     SetFocus(hEditorWnd);
873 
874     if(!ret)
875     {
876         GETTEXTLENGTHEX gt;
877         gt.flags = GTL_DEFAULT;
878         gt.codepage = 1200;
879 
880         if(SendMessageW(hEditorWnd, EM_GETTEXTLENGTHEX, (WPARAM)&gt, 0))
881             return FALSE;
882     }
883 
884     lstrcpyW(wszFileName, wszSaveFileName);
885     set_caption(wszFileName);
886     SendMessageW(hEditorWnd, EM_SETMODIFY, FALSE, 0);
887     set_fileformat(format);
888 
889     return TRUE;
890 }
891 
892 static BOOL DialogSaveFile(void)
893 {
894     OPENFILENAMEW sfn;
895 
896     WCHAR wszFile[MAX_PATH] = {'\0'};
897     static const WCHAR wszDefExt[] = {'r','t','f','\0'};
898 
899     ZeroMemory(&sfn, sizeof(sfn));
900 
901     sfn.lStructSize = sizeof(sfn);
902     sfn.Flags = OFN_HIDEREADONLY | OFN_PATHMUSTEXIST | OFN_OVERWRITEPROMPT | OFN_ENABLESIZING;
903     sfn.hwndOwner = hMainWnd;
904     sfn.lpstrFilter = wszFilter;
905     sfn.lpstrFile = wszFile;
906     sfn.nMaxFile = MAX_PATH;
907     sfn.lpstrDefExt = wszDefExt;
908     sfn.nFilterIndex = fileformat_number(fileFormat)+1;
909 
910     while(GetSaveFileNameW(&sfn))
911     {
912         if(fileformat_flags(sfn.nFilterIndex-1) != SF_RTF)
913         {
914             if(MessageBoxWithResStringW(hMainWnd, MAKEINTRESOURCEW(STRING_SAVE_LOSEFORMATTING),
915                            wszAppTitle, MB_YESNO | MB_ICONEXCLAMATION) != IDYES)
916                 continue;
917         }
918         return DoSaveFile(sfn.lpstrFile, fileformat_flags(sfn.nFilterIndex-1));
919     }
920     return FALSE;
921 }
922 
923 static BOOL prompt_save_changes(void)
924 {
925     if(!wszFileName[0])
926     {
927         GETTEXTLENGTHEX gt;
928         gt.flags = GTL_NUMCHARS;
929         gt.codepage = 1200;
930         if(!SendMessageW(hEditorWnd, EM_GETTEXTLENGTHEX, (WPARAM)&gt, 0))
931             return TRUE;
932     }
933 
934     if(!SendMessageW(hEditorWnd, EM_GETMODIFY, 0, 0))
935     {
936         return TRUE;
937     } else
938     {
939         LPWSTR displayFileName;
940         WCHAR *text;
941         int ret;
942 
943         if(!wszFileName[0])
944             displayFileName = wszDefaultFileName;
945         else
946             displayFileName = file_basename(wszFileName);
947 
948         text = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
949                          (lstrlenW(displayFileName)+lstrlenW(wszSaveChanges))*sizeof(WCHAR));
950 
951         if(!text)
952             return FALSE;
953 
954         wsprintfW(text, wszSaveChanges, displayFileName);
955 
956         ret = MessageBoxW(hMainWnd, text, wszAppTitle, MB_YESNOCANCEL | MB_ICONEXCLAMATION);
957 
958         HeapFree(GetProcessHeap(), 0, text);
959 
960         switch(ret)
961         {
962             case IDNO:
963                 return TRUE;
964 
965             case IDYES:
966                 if(wszFileName[0])
967                     return DoSaveFile(wszFileName, fileFormat);
968                 return DialogSaveFile();
969 
970             default:
971                 return FALSE;
972         }
973     }
974 }
975 
976 static void DialogOpenFile(void)
977 {
978     OPENFILENAMEW ofn;
979 
980     WCHAR wszFile[MAX_PATH] = {'\0'};
981     static const WCHAR wszDefExt[] = {'r','t','f','\0'};
982 
983     ZeroMemory(&ofn, sizeof(ofn));
984 
985     ofn.lStructSize = sizeof(ofn);
986     ofn.Flags = OFN_HIDEREADONLY | OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST | OFN_ENABLESIZING;
987     ofn.hwndOwner = hMainWnd;
988     ofn.lpstrFilter = wszFilter;
989     ofn.lpstrFile = wszFile;
990     ofn.nMaxFile = MAX_PATH;
991     ofn.lpstrDefExt = wszDefExt;
992     ofn.nFilterIndex = fileformat_number(fileFormat)+1;
993 
994     if(GetOpenFileNameW(&ofn))
995     {
996         if(prompt_save_changes())
997             DoOpenFile(ofn.lpstrFile);
998     }
999 }
1000 
1001 static void dialog_about(void)
1002 {
1003     HICON icon = LoadImageW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDI_WORDPAD), IMAGE_ICON, 48, 48, LR_SHARED);
1004     ShellAboutW(hMainWnd, wszAppTitle, 0, icon);
1005 }
1006 
1007 static INT_PTR CALLBACK formatopts_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
1008 {
1009     switch(message)
1010     {
1011         case WM_INITDIALOG:
1012             {
1013                 LPPROPSHEETPAGEW ps = (LPPROPSHEETPAGEW)lParam;
1014                 int wrap = -1;
1015                 char id[4];
1016                 HWND hIdWnd = GetDlgItem(hWnd, IDC_PAGEFMT_ID);
1017 
1018                 sprintf(id, "%d\n", (int)ps->lParam);
1019                 SetWindowTextA(hIdWnd, id);
1020                 if(wordWrap[ps->lParam] == ID_WORDWRAP_NONE)
1021                     wrap = IDC_PAGEFMT_WN;
1022                 else if(wordWrap[ps->lParam] == ID_WORDWRAP_WINDOW)
1023                     wrap = IDC_PAGEFMT_WW;
1024                 else if(wordWrap[ps->lParam] == ID_WORDWRAP_MARGIN)
1025                     wrap = IDC_PAGEFMT_WM;
1026 
1027                 if(wrap != -1)
1028                     CheckRadioButton(hWnd, IDC_PAGEFMT_WN,
1029                                      IDC_PAGEFMT_WM, wrap);
1030 
1031                 if(barState[ps->lParam] & (1 << BANDID_TOOLBAR))
1032                     CheckDlgButton(hWnd, IDC_PAGEFMT_TB, TRUE);
1033                 if(barState[ps->lParam] & (1 << BANDID_FORMATBAR))
1034                     CheckDlgButton(hWnd, IDC_PAGEFMT_FB, TRUE);
1035                 if(barState[ps->lParam] & (1 << BANDID_RULER))
1036                     CheckDlgButton(hWnd, IDC_PAGEFMT_RU, TRUE);
1037                 if(barState[ps->lParam] & (1 << BANDID_STATUSBAR))
1038                     CheckDlgButton(hWnd, IDC_PAGEFMT_SB, TRUE);
1039             }
1040             break;
1041 
1042         case WM_COMMAND:
1043             switch(LOWORD(wParam))
1044             {
1045                 case IDC_PAGEFMT_WN:
1046                 case IDC_PAGEFMT_WW:
1047                 case IDC_PAGEFMT_WM:
1048                     CheckRadioButton(hWnd, IDC_PAGEFMT_WN, IDC_PAGEFMT_WM,
1049                                      LOWORD(wParam));
1050                     break;
1051 
1052                 case IDC_PAGEFMT_TB:
1053                 case IDC_PAGEFMT_FB:
1054                 case IDC_PAGEFMT_RU:
1055                 case IDC_PAGEFMT_SB:
1056                     CheckDlgButton(hWnd, LOWORD(wParam),
1057                                    !IsDlgButtonChecked(hWnd, LOWORD(wParam)));
1058                     break;
1059             }
1060             break;
1061         case WM_NOTIFY:
1062             {
1063                 LPNMHDR header = (LPNMHDR)lParam;
1064                 if(header->code == PSN_APPLY)
1065                 {
1066                     HWND hIdWnd = GetDlgItem(hWnd, IDC_PAGEFMT_ID);
1067                     char sid[4];
1068                     int id;
1069 
1070                     GetWindowTextA(hIdWnd, sid, 4);
1071                     id = atoi(sid);
1072                     if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_WN))
1073                         wordWrap[id] = ID_WORDWRAP_NONE;
1074                     else if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_WW))
1075                         wordWrap[id] = ID_WORDWRAP_WINDOW;
1076                     else if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_WM))
1077                         wordWrap[id] = ID_WORDWRAP_MARGIN;
1078 
1079                     if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_TB))
1080                         barState[id] |= (1 << BANDID_TOOLBAR);
1081                     else
1082                         barState[id] &= ~(1 << BANDID_TOOLBAR);
1083 
1084                     if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_FB))
1085                         barState[id] |= (1 << BANDID_FORMATBAR);
1086                     else
1087                         barState[id] &= ~(1 << BANDID_FORMATBAR);
1088 
1089                     if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_RU))
1090                         barState[id] |= (1 << BANDID_RULER);
1091                     else
1092                         barState[id] &= ~(1 << BANDID_RULER);
1093 
1094                     if(IsDlgButtonChecked(hWnd, IDC_PAGEFMT_SB))
1095                         barState[id] |= (1 << BANDID_STATUSBAR);
1096                     else
1097                         barState[id] &= ~(1 << BANDID_STATUSBAR);
1098                 }
1099             }
1100             break;
1101     }
1102     return FALSE;
1103 }
1104 
1105 static void dialog_viewproperties(void)
1106 {
1107     PROPSHEETPAGEW psp[2];
1108     PROPSHEETHEADERW psh;
1109     size_t i;
1110     HINSTANCE hInstance = GetModuleHandleW(0);
1111     LPCPROPSHEETPAGEW ppsp = (LPCPROPSHEETPAGEW)&psp;
1112 
1113     psp[0].dwSize = sizeof(PROPSHEETPAGEW);
1114     psp[0].dwFlags = PSP_USETITLE;
1115     U(psp[0]).pszTemplate = MAKEINTRESOURCEW(IDD_FORMATOPTS);
1116     psp[0].pfnDlgProc = formatopts_proc;
1117     psp[0].hInstance = hInstance;
1118     psp[0].lParam = reg_formatindex(SF_TEXT);
1119     psp[0].pfnCallback = NULL;
1120     psp[0].pszTitle = MAKEINTRESOURCEW(STRING_VIEWPROPS_TEXT);
1121     for(i = 1; i < sizeof(psp)/sizeof(psp[0]); i++)
1122     {
1123         psp[i].dwSize = psp[0].dwSize;
1124         psp[i].dwFlags = psp[0].dwFlags;
1125         U(psp[i]).pszTemplate = U(psp[0]).pszTemplate;
1126         psp[i].pfnDlgProc = psp[0].pfnDlgProc;
1127         psp[i].hInstance = psp[0].hInstance;
1128         psp[i].lParam = reg_formatindex(SF_RTF);
1129         psp[i].pfnCallback = psp[0].pfnCallback;
1130         psp[i].pszTitle = MAKEINTRESOURCEW(STRING_VIEWPROPS_RICHTEXT);
1131     }
1132 
1133     psh.dwSize = sizeof(psh);
1134     psh.dwFlags = PSH_USEICONID | PSH_PROPSHEETPAGE | PSH_NOAPPLYNOW;
1135     psh.hwndParent = hMainWnd;
1136     psh.hInstance = hInstance;
1137     psh.pszCaption = MAKEINTRESOURCEW(STRING_VIEWPROPS_TITLE);
1138     psh.nPages = sizeof(psp)/sizeof(psp[0]);
1139     U3(psh).ppsp = ppsp;
1140     U(psh).pszIcon = MAKEINTRESOURCEW(IDI_WORDPAD);
1141 
1142     if(fileFormat & SF_RTF)
1143         U2(psh).nStartPage = 1;
1144     else
1145         U2(psh).nStartPage = 0;
1146     PropertySheetW(&psh);
1147     set_bar_states();
1148     target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
1149 }
1150 
1151 static void HandleCommandLine(LPWSTR cmdline)
1152 {
1153     WCHAR delimiter;
1154     BOOL opt_print = FALSE;
1155 
1156     /* skip white space */
1157     while (*cmdline == ' ') cmdline++;
1158 
1159     /* skip executable name */
1160     delimiter = (*cmdline == '"' ? '"' : ' ');
1161 
1162     if (*cmdline == delimiter) cmdline++;
1163     while (*cmdline && *cmdline != delimiter) cmdline++;
1164     if (*cmdline == delimiter) cmdline++;
1165 
1166     while (*cmdline)
1167     {
1168         while (isspace(*cmdline)) cmdline++;
1169 
1170         if (*cmdline == '-' || *cmdline == '/')
1171         {
1172             if (!cmdline[2] || isspace(cmdline[2]))
1173             {
1174                 switch (cmdline[1])
1175                 {
1176                 case 'P':
1177                 case 'p':
1178                     opt_print = TRUE;
1179                     cmdline += 2;
1180                     continue;
1181                 }
1182             }
1183             /* a filename starting by / */
1184         }
1185         break;
1186     }
1187 
1188     if (*cmdline)
1189     {
1190         /* file name is passed on the command line */
1191         if (cmdline[0] == '"')
1192         {
1193             cmdline++;
1194             cmdline[lstrlenW(cmdline) - 1] = 0;
1195         }
1196         DoOpenFile(cmdline);
1197         InvalidateRect(hMainWnd, NULL, FALSE);
1198     }
1199 
1200     if (opt_print)
1201         MessageBoxWithResStringW(hMainWnd, MAKEINTRESOURCEW(STRING_PRINTING_NOT_IMPLEMENTED), wszAppTitle, MB_OK);
1202 }
1203 
1204 static LRESULT handle_findmsg(LPFINDREPLACEW pFr)
1205 {
1206     if(pFr->Flags & FR_DIALOGTERM)
1207     {
1208         hFindWnd = 0;
1209         pFr->Flags = FR_FINDNEXT;
1210         return 0;
1211     }
1212 
1213     if(pFr->Flags & FR_FINDNEXT || pFr->Flags & FR_REPLACE || pFr->Flags & FR_REPLACEALL)
1214     {
1215         FINDREPLACE_custom *custom_data = (FINDREPLACE_custom*)pFr->lCustData;
1216         DWORD flags;
1217         FINDTEXTEXW ft;
1218         CHARRANGE sel;
1219         LRESULT ret = -1;
1220         HMENU hMenu = GetMenu(hMainWnd);
1221         MENUITEMINFOW mi;
1222 
1223         mi.cbSize = sizeof(mi);
1224         mi.fMask = MIIM_DATA;
1225         mi.dwItemData = 1;
1226         SetMenuItemInfoW(hMenu, ID_FIND_NEXT, FALSE, &mi);
1227 
1228         /* Make sure find field is saved. */
1229         if (pFr->lpstrFindWhat != custom_data->findBuffer)
1230         {
1231             lstrcpynW(custom_data->findBuffer, pFr->lpstrFindWhat,
1232                       sizeof(custom_data->findBuffer) / sizeof(WCHAR));
1233             pFr->lpstrFindWhat = custom_data->findBuffer;
1234         }
1235 
1236         SendMessageW(hEditorWnd, EM_GETSEL, (WPARAM)&sel.cpMin, (LPARAM)&sel.cpMax);
1237         if(custom_data->endPos == -1) {
1238             custom_data->endPos = sel.cpMin;
1239             custom_data->wrapped = FALSE;
1240         }
1241 
1242         flags = FR_DOWN | (pFr->Flags & (FR_MATCHCASE | FR_WHOLEWORD));
1243         ft.lpstrText = pFr->lpstrFindWhat;
1244 
1245         /* Only replace the existing selection if it is an exact match. */
1246         if (sel.cpMin != sel.cpMax &&
1247             (pFr->Flags & FR_REPLACE || pFr->Flags & FR_REPLACEALL))
1248         {
1249             ft.chrg = sel;
1250             SendMessageW(hEditorWnd, EM_FINDTEXTEXW, flags, (LPARAM)&ft);
1251             if (ft.chrgText.cpMin == sel.cpMin && ft.chrgText.cpMax == sel.cpMax) {
1252                 SendMessageW(hEditorWnd, EM_REPLACESEL, TRUE, (LPARAM)pFr->lpstrReplaceWith);
1253                 SendMessageW(hEditorWnd, EM_GETSEL, (WPARAM)&sel.cpMin, (LPARAM)&sel.cpMax);
1254             }
1255         }
1256 
1257         /* Search from the start of the selection, but exclude the first character
1258          * from search if there is a selection. */
1259         ft.chrg.cpMin = sel.cpMin;
1260         if (sel.cpMin != sel.cpMax)
1261             ft.chrg.cpMin++;
1262 
1263         /* Search to the end, then wrap around and search from the start. */
1264         if (!custom_data->wrapped) {
1265             ft.chrg.cpMax = -1;
1266             ret = SendMessageW(hEditorWnd, EM_FINDTEXTEXW, flags, (LPARAM)&ft);
1267             if (ret == -1) {
1268                 custom_data->wrapped = TRUE;
1269                 ft.chrg.cpMin = 0;
1270             }
1271         }
1272 
1273         if (ret == -1) {
1274             ft.chrg.cpMax = custom_data->endPos + lstrlenW(pFr->lpstrFindWhat) - 1;
1275             if (ft.chrg.cpMax > ft.chrg.cpMin)
1276                 ret = SendMessageW(hEditorWnd, EM_FINDTEXTEXW, flags, (LPARAM)&ft);
1277         }
1278 
1279         if (ret == -1) {
1280             custom_data->endPos = -1;
1281             EnableWindow(hMainWnd, FALSE);
1282             MessageBoxWithResStringW(hFindWnd, MAKEINTRESOURCEW(STRING_SEARCH_FINISHED),
1283                                      wszAppTitle, MB_OK | MB_ICONASTERISK | MB_TASKMODAL);
1284             EnableWindow(hMainWnd, TRUE);
1285         } else {
1286             SendMessageW(hEditorWnd, EM_SETSEL, ft.chrgText.cpMin, ft.chrgText.cpMax);
1287             SendMessageW(hEditorWnd, EM_SCROLLCARET, 0, 0);
1288 
1289             if (pFr->Flags & FR_REPLACEALL)
1290                 return handle_findmsg(pFr);
1291         }
1292     }
1293 
1294     return 0;
1295 }
1296 
1297 static void dialog_find(LPFINDREPLACEW fr, BOOL replace)
1298 {
1299     static WCHAR selBuffer[128];
1300     static WCHAR replaceBuffer[128];
1301     static FINDREPLACE_custom custom_data;
1302     static const WCHAR endl = '\r';
1303     FINDTEXTW ft;
1304 
1305     /* Allow only one search/replace dialog to open */
1306     if(hFindWnd != NULL)
1307     {
1308         SetActiveWindow(hFindWnd);
1309         return;
1310     }
1311 
1312     ZeroMemory(fr, sizeof(FINDREPLACEW));
1313     fr->lStructSize = sizeof(FINDREPLACEW);
1314     fr->hwndOwner = hMainWnd;
1315     fr->Flags = FR_HIDEUPDOWN;
1316     /* Find field is filled with the selected text if it is non-empty
1317      * and stays within the same paragraph, otherwise the previous
1318      * find field is used. */
1319     SendMessageW(hEditorWnd, EM_GETSEL, (WPARAM)&ft.chrg.cpMin,
1320                  (LPARAM)&ft.chrg.cpMax);
1321     ft.lpstrText = &endl;
1322     if (ft.chrg.cpMin != ft.chrg.cpMax &&
1323         SendMessageW(hEditorWnd, EM_FINDTEXTW, FR_DOWN, (LPARAM)&ft) == -1)
1324     {
1325         /* Use a temporary buffer for the selected text so that the saved
1326          * find field is only overwritten when a find/replace is clicked. */
1327         GETTEXTEX gt = {sizeof(selBuffer), GT_SELECTION, 1200, NULL, NULL};
1328         SendMessageW(hEditorWnd, EM_GETTEXTEX, (WPARAM)&gt, (LPARAM)selBuffer);
1329         fr->lpstrFindWhat = selBuffer;
1330     } else {
1331         fr->lpstrFindWhat = custom_data.findBuffer;
1332     }
1333     fr->lpstrReplaceWith = replaceBuffer;
1334     custom_data.endPos = -1;
1335     custom_data.wrapped = FALSE;
1336     fr->lCustData = (LPARAM)&custom_data;
1337     fr->wFindWhatLen = sizeof(custom_data.findBuffer);
1338     fr->wReplaceWithLen = sizeof(replaceBuffer);
1339 
1340     if(replace)
1341         hFindWnd = ReplaceTextW(fr);
1342     else
1343         hFindWnd = FindTextW(fr);
1344 }
1345 
1346 static int units_to_twips(UNIT unit, float number)
1347 {
1348     int twips = 0;
1349 
1350     switch(unit)
1351     {
1352     case UNIT_CM:
1353         twips = (int)(number * 1000.0 / (float)CENTMM_PER_INCH * (float)TWIPS_PER_INCH);
1354         break;
1355 
1356     case UNIT_INCH:
1357         twips = (int)(number * (float)TWIPS_PER_INCH);
1358         break;
1359 
1360     case UNIT_PT:
1361         twips = (int)(number * (0.0138 * (float)TWIPS_PER_INCH));
1362         break;
1363     }
1364 
1365     return twips;
1366 }
1367 
1368 static void append_current_units(LPWSTR buffer)
1369 {
1370     static const WCHAR space[] = {' ', 0};
1371     lstrcatW(buffer, space);
1372     lstrcatW(buffer, units_cmW);
1373 }
1374 
1375 static void number_with_units(LPWSTR buffer, int number)
1376 {
1377     static const WCHAR fmt[] = {'%','.','2','f',' ','%','s','\0'};
1378     float converted = (float)number / (float)TWIPS_PER_INCH *(float)CENTMM_PER_INCH / 1000.0;
1379 
1380     sprintfW(buffer, fmt, converted, units_cmW);
1381 }
1382 
1383 static BOOL get_comboexlist_selection(HWND hComboEx, LPWSTR wszBuffer, UINT bufferLength)
1384 {
1385     COMBOBOXEXITEMW cbItem;
1386     COMBOBOXINFO cbInfo;
1387     HWND hCombo, hList;
1388     int idx, result;
1389 
1390     hCombo = (HWND)SendMessageW(hComboEx, CBEM_GETCOMBOCONTROL, 0, 0);
1391     if (!hCombo)
1392         return FALSE;
1393     cbInfo.cbSize = sizeof(COMBOBOXINFO);
1394     result = SendMessageW(hCombo, CB_GETCOMBOBOXINFO, 0, (LPARAM)&cbInfo);
1395     if (!result)
1396         return FALSE;
1397     hList = cbInfo.hwndList;
1398     idx = SendMessageW(hList, LB_GETCURSEL, 0, 0);
1399     if (idx < 0)
1400         return FALSE;
1401 
1402     ZeroMemory(&cbItem, sizeof(cbItem));
1403     cbItem.mask = CBEIF_TEXT;
1404     cbItem.iItem = idx;
1405     cbItem.pszText = wszBuffer;
1406     cbItem.cchTextMax = bufferLength-1;
1407     result = SendMessageW(hComboEx, CBEM_GETITEMW, 0, (LPARAM)&cbItem);
1408 
1409     return result != 0;
1410 }
1411 
1412 static INT_PTR CALLBACK datetime_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
1413 {
1414     switch(message)
1415     {
1416         case WM_INITDIALOG:
1417             {
1418                 WCHAR buffer[MAX_STRING_LEN];
1419                 SYSTEMTIME st;
1420                 HWND hListWnd = GetDlgItem(hWnd, IDC_DATETIME);
1421                 GetLocalTime(&st);
1422 
1423                 GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &st, 0, (LPWSTR)&buffer,
1424                                MAX_STRING_LEN);
1425                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1426                 GetDateFormatW(LOCALE_USER_DEFAULT, DATE_LONGDATE, &st, 0, (LPWSTR)&buffer,
1427                                MAX_STRING_LEN);
1428                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1429                 GetTimeFormatW(LOCALE_USER_DEFAULT, 0, &st, 0, (LPWSTR)&buffer, MAX_STRING_LEN);
1430                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1431 
1432                 SendMessageW(hListWnd, LB_SETSEL, TRUE, 0);
1433             }
1434             break;
1435 
1436         case WM_COMMAND:
1437             switch(LOWORD(wParam))
1438             {
1439                 case IDC_DATETIME:
1440                     if (HIWORD(wParam) != LBN_DBLCLK)
1441                         break;
1442                     /* Fall through */
1443 
1444                 case IDOK:
1445                     {
1446                         LRESULT index;
1447                         HWND hListWnd = GetDlgItem(hWnd, IDC_DATETIME);
1448 
1449                         index = SendMessageW(hListWnd, LB_GETCURSEL, 0, 0);
1450 
1451                         if(index != LB_ERR)
1452                         {
1453                             WCHAR buffer[MAX_STRING_LEN];
1454                             SendMessageW(hListWnd, LB_GETTEXT, index, (LPARAM)&buffer);
1455                             SendMessageW(hEditorWnd, EM_REPLACESEL, TRUE, (LPARAM)&buffer);
1456                         }
1457                     }
1458                     /* Fall through */
1459 
1460                 case IDCANCEL:
1461                     EndDialog(hWnd, wParam);
1462                     return TRUE;
1463             }
1464     }
1465     return FALSE;
1466 }
1467 
1468 static INT_PTR CALLBACK newfile_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
1469 {
1470     switch(message)
1471     {
1472         case WM_INITDIALOG:
1473             {
1474                 HINSTANCE hInstance = GetModuleHandleW(0);
1475                 WCHAR buffer[MAX_STRING_LEN];
1476                 HWND hListWnd = GetDlgItem(hWnd, IDC_NEWFILE);
1477 
1478                 LoadStringW(hInstance, STRING_NEWFILE_RICHTEXT, buffer, MAX_STRING_LEN);
1479                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1480                 LoadStringW(hInstance, STRING_NEWFILE_TXT, buffer, MAX_STRING_LEN);
1481                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1482                 LoadStringW(hInstance, STRING_NEWFILE_TXT_UNICODE, buffer, MAX_STRING_LEN);
1483                 SendMessageW(hListWnd, LB_ADDSTRING, 0, (LPARAM)&buffer);
1484 
1485                 SendMessageW(hListWnd, LB_SETSEL, TRUE, 0);
1486             }
1487             break;
1488 
1489         case WM_COMMAND:
1490             switch(LOWORD(wParam))
1491             {
1492                 case IDOK:
1493                     {
1494                         LRESULT index;
1495                         HWND hListWnd = GetDlgItem(hWnd, IDC_NEWFILE);
1496                         index = SendMessageW(hListWnd, LB_GETCURSEL, 0, 0);
1497 
1498                         if(index != LB_ERR)
1499                             EndDialog(hWnd, MAKELONG(fileformat_flags(index),0));
1500                     }
1501                     return TRUE;
1502 
1503                 case IDCANCEL:
1504                     EndDialog(hWnd, MAKELONG(ID_NEWFILE_ABORT,0));
1505                     return TRUE;
1506             }
1507     }
1508     return FALSE;
1509 }
1510 
1511 static INT_PTR CALLBACK paraformat_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
1512 {
1513     static const WORD ALIGNMENT_VALUES[] = {PFA_LEFT, PFA_RIGHT, PFA_CENTER};
1514 
1515     switch(message)
1516     {
1517         case WM_INITDIALOG:
1518             {
1519                 HINSTANCE hInstance = GetModuleHandleW(0);
1520                 WCHAR buffer[MAX_STRING_LEN];
1521                 HWND hListWnd = GetDlgItem(hWnd, IDC_PARA_ALIGN);
1522                 HWND hLeftWnd = GetDlgItem(hWnd, IDC_PARA_LEFT);
1523                 HWND hRightWnd = GetDlgItem(hWnd, IDC_PARA_RIGHT);
1524                 HWND hFirstWnd = GetDlgItem(hWnd, IDC_PARA_FIRST);
1525                 PARAFORMAT2 pf;
1526                 int index = 0;
1527 
1528                 LoadStringW(hInstance, STRING_ALIGN_LEFT, buffer,
1529                             MAX_STRING_LEN);
1530                 SendMessageW(hListWnd, CB_ADDSTRING, 0, (LPARAM)buffer);
1531                 LoadStringW(hInstance, STRING_ALIGN_RIGHT, buffer,
1532                             MAX_STRING_LEN);
1533                 SendMessageW(hListWnd, CB_ADDSTRING, 0, (LPARAM)buffer);
1534                 LoadStringW(hInstance, STRING_ALIGN_CENTER, buffer,
1535                             MAX_STRING_LEN);
1536                 SendMessageW(hListWnd, CB_ADDSTRING, 0, (LPARAM)buffer);
1537 
1538                 pf.cbSize = sizeof(pf);
1539                 pf.dwMask = PFM_ALIGNMENT | PFM_OFFSET | PFM_RIGHTINDENT |
1540                             PFM_STARTINDENT;
1541                 SendMessageW(hEditorWnd, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
1542 
1543                 if(pf.wAlignment == PFA_RIGHT)
1544                     index ++;
1545                 else if(pf.wAlignment == PFA_CENTER)
1546                     index += 2;
1547 
1548                 SendMessageW(hListWnd, CB_SETCURSEL, index, 0);
1549 
1550                 number_with_units(buffer, pf.dxStartIndent + pf.dxOffset);
1551                 SetWindowTextW(hLeftWnd, buffer);
1552                 number_with_units(buffer, pf.dxRightIndent);
1553                 SetWindowTextW(hRightWnd, buffer);
1554                 number_with_units(buffer, -pf.dxOffset);
1555                 SetWindowTextW(hFirstWnd, buffer);
1556             }
1557             break;
1558 
1559         case WM_COMMAND:
1560             switch(LOWORD(wParam))
1561             {
1562                 case IDOK:
1563                     {
1564                         HWND hListWnd = GetDlgItem(hWnd, IDC_PARA_ALIGN);
1565                         HWND hLeftWnd = GetDlgItem(hWnd, IDC_PARA_LEFT);
1566                         HWND hRightWnd = GetDlgItem(hWnd, IDC_PARA_RIGHT);
1567                         HWND hFirstWnd = GetDlgItem(hWnd, IDC_PARA_FIRST);
1568                         WCHAR buffer[MAX_STRING_LEN];
1569                         int index;
1570                         float num;
1571                         int ret = 0;
1572                         PARAFORMAT pf;
1573                         UNIT unit;
1574 
1575                         index = SendMessageW(hListWnd, CB_GETCURSEL, 0, 0);
1576                         pf.wAlignment = ALIGNMENT_VALUES[index];
1577 
1578                         GetWindowTextW(hLeftWnd, buffer, MAX_STRING_LEN);
1579                         if(number_from_string(buffer, &num, &unit))
1580                             ret++;
1581                         pf.dxOffset = units_to_twips(unit, num);
1582                         GetWindowTextW(hRightWnd, buffer, MAX_STRING_LEN);
1583                         if(number_from_string(buffer, &num, &unit))
1584                             ret++;
1585                         pf.dxRightIndent = units_to_twips(unit, num);
1586                         GetWindowTextW(hFirstWnd, buffer, MAX_STRING_LEN);
1587                         if(number_from_string(buffer, &num, &unit))
1588                             ret++;
1589                         pf.dxStartIndent = units_to_twips(unit, num);
1590 
1591                         if(ret != 3)
1592                         {
1593                             MessageBoxWithResStringW(hMainWnd, MAKEINTRESOURCEW(STRING_INVALID_NUMBER),
1594                                         wszAppTitle, MB_OK | MB_ICONASTERISK);
1595                             return FALSE;
1596                         } else
1597                         {
1598                             if (pf.dxOffset + pf.dxStartIndent < 0
1599                                 && pf.dxStartIndent < 0)
1600                             {
1601                                 /* The first line is before the left edge, so
1602                                  * make sure it is at the left edge. */
1603                                 pf.dxOffset = -pf.dxStartIndent;
1604                             } else if (pf.dxOffset < 0) {
1605                                 /* The second and following lines are before
1606                                  * the left edge, so set it to be at the left
1607                                  * edge, and adjust the first line since it
1608                                  * is relative to it. */
1609                                 pf.dxStartIndent = max(pf.dxStartIndent + pf.dxOffset, 0);
1610                                 pf.dxOffset = 0;
1611                             }
1612                             /* Internally the dxStartIndent is the absolute
1613                              * offset for the first line and dxOffset is
1614                              * to it value as opposed how it is displayed with
1615                              * the first line being the relative value.
1616                              * These two lines make the adjustments. */
1617                             pf.dxStartIndent = pf.dxStartIndent + pf.dxOffset;
1618                             pf.dxOffset = pf.dxOffset - pf.dxStartIndent;
1619 
1620                             pf.cbSize = sizeof(pf);
1621                             pf.dwMask = PFM_ALIGNMENT | PFM_OFFSET | PFM_RIGHTINDENT |
1622                                         PFM_STARTINDENT;
1623                             SendMessageW(hEditorWnd, EM_SETPARAFORMAT, 0, (LPARAM)&pf);
1624                         }
1625                     }
1626                     /* Fall through */
1627 
1628                 case IDCANCEL:
1629                     EndDialog(hWnd, wParam);
1630                     return TRUE;
1631             }
1632     }
1633     return FALSE;
1634 }
1635 
1636 static INT_PTR CALLBACK tabstops_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
1637 {
1638     switch(message)
1639     {
1640         case WM_INITDIALOG:
1641             {
1642                 HWND hTabWnd = GetDlgItem(hWnd, IDC_TABSTOPS);
1643                 PARAFORMAT pf;
1644                 WCHAR buffer[MAX_STRING_LEN];
1645                 int i;
1646 
1647                 pf.cbSize = sizeof(pf);
1648                 pf.dwMask = PFM_TABSTOPS;
1649                 SendMessageW(hEditorWnd, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
1650                 SendMessageW(hTabWnd, CB_LIMITTEXT, MAX_STRING_LEN-1, 0);
1651 
1652                 for(i = 0; i < pf.cTabCount; i++)
1653                 {
1654                     number_with_units(buffer, pf.rgxTabs[i]);
1655                     SendMessageW(hTabWnd, CB_ADDSTRING, 0, (LPARAM)&buffer);
1656                 }
1657                 SetFocus(hTabWnd);
1658             }
1659             break;
1660 
1661         case WM_COMMAND:
1662             switch(LOWORD(wParam))
1663             {
1664                 case IDC_TABSTOPS:
1665                     {
1666                         HWND hTabWnd = (HWND)lParam;
1667                         HWND hAddWnd = GetDlgItem(hWnd, ID_TAB_ADD);
1668                         HWND hDelWnd = GetDlgItem(hWnd, ID_TAB_DEL);
1669                         HWND hEmptyWnd = GetDlgItem(hWnd, ID_TAB_EMPTY);
1670 
1671                         if(GetWindowTextLengthW(hTabWnd))
1672                             EnableWindow(hAddWnd, TRUE);
1673                         else
1674                             EnableWindow(hAddWnd, FALSE);
1675 
1676                         if(SendMessageW(hTabWnd, CB_GETCOUNT, 0, 0))
1677                         {
1678                             EnableWindow(hEmptyWnd, TRUE);
1679 
1680                             if(SendMessageW(hTabWnd, CB_GETCURSEL, 0, 0) == CB_ERR)
1681                                 EnableWindow(hDelWnd, FALSE);
1682                             else
1683                                 EnableWindow(hDelWnd, TRUE);
1684                         } else
1685                         {
1686                             EnableWindow(hEmptyWnd, FALSE);
1687                         }
1688                     }
1689                     break;
1690 
1691                 case ID_TAB_ADD:
1692                     {
1693                         HWND hTabWnd = GetDlgItem(hWnd, IDC_TABSTOPS);
1694                         WCHAR buffer[MAX_STRING_LEN];
1695                         UNIT unit;
1696 
1697                         GetWindowTextW(hTabWnd, buffer, MAX_STRING_LEN);
1698                         append_current_units(buffer);
1699 
1700                         if(SendMessageW(hTabWnd, CB_FINDSTRINGEXACT, -1, (LPARAM)&buffer) == CB_ERR)
1701                         {
1702                             float number = 0;
1703                             int item_count = SendMessageW(hTabWnd, CB_GETCOUNT, 0, 0);
1704 
1705                             if(!number_from_string(buffer, &number, &unit))
1706                             {
1707                                 MessageBoxWithResStringW(hWnd, MAKEINTRESOURCEW(STRING_INVALID_NUMBER),
1708                                              wszAppTitle, MB_OK | MB_ICONINFORMATION);
1709                             } else if (item_count >= MAX_TAB_STOPS) {
1710                                 MessageBoxWithResStringW(hWnd, MAKEINTRESOURCEW(STRING_MAX_TAB_STOPS),
1711                                              wszAppTitle, MB_OK | MB_ICONINFORMATION);
1712                             } else {
1713                                 int i;
1714                                 float next_number = -1;
1715                                 int next_number_in_twips = -1;
1716                                 int insert_number = units_to_twips(unit, number);
1717 
1718                                 /* linear search for position to insert the string */
1719                                 for(i = 0; i < item_count; i++)
1720                                 {
1721                                     SendMessageW(hTabWnd, CB_GETLBTEXT, i, (LPARAM)&buffer);
1722                                     number_from_string(buffer, &next_number, &unit);
1723                                     next_number_in_twips = units_to_twips(unit, next_number);
1724                                     if (insert_number <= next_number_in_twips)
1725                                         break;
1726                                 }
1727                                 if (insert_number != next_number_in_twips)
1728                                 {
1729                                     number_with_units(buffer, insert_number);
1730                                     SendMessageW(hTabWnd, CB_INSERTSTRING, i, (LPARAM)&buffer);
1731                                     SetWindowTextW(hTabWnd, 0);
1732                                 }
1733                             }
1734                         }
1735                         SetFocus(hTabWnd);
1736                     }
1737                     break;
1738 
1739                 case ID_TAB_DEL:
1740                     {
1741                         HWND hTabWnd = GetDlgItem(hWnd, IDC_TABSTOPS);
1742                         LRESULT ret;
1743                         ret = SendMessageW(hTabWnd, CB_GETCURSEL, 0, 0);
1744                         if(ret != CB_ERR)
1745                             SendMessageW(hTabWnd, CB_DELETESTRING, ret, 0);
1746                     }
1747                     break;
1748 
1749                 case ID_TAB_EMPTY:
1750                     {
1751                         HWND hTabWnd = GetDlgItem(hWnd, IDC_TABSTOPS);
1752                         SendMessageW(hTabWnd, CB_RESETCONTENT, 0, 0);
1753                         SetFocus(hTabWnd);
1754                     }
1755                     break;
1756 
1757                 case IDOK:
1758                     {
1759                         HWND hTabWnd = GetDlgItem(hWnd, IDC_TABSTOPS);
1760                         int i;
1761                         WCHAR buffer[MAX_STRING_LEN];
1762                         PARAFORMAT pf;
1763                         float number;
1764                         UNIT unit;
1765 
1766                         pf.cbSize = sizeof(pf);
1767                         pf.dwMask = PFM_TABSTOPS;
1768 
1769                         for(i = 0; SendMessageW(hTabWnd, CB_GETLBTEXT, i,
1770                                                 (LPARAM)&buffer) != CB_ERR &&
1771                                                         i < MAX_TAB_STOPS; i++)
1772                         {
1773                             number_from_string(buffer, &number, &unit);
1774                             pf.rgxTabs[i] = units_to_twips(unit, number);
1775                         }
1776                         pf.cTabCount = i;
1777                         SendMessageW(hEditorWnd, EM_SETPARAFORMAT, 0, (LPARAM)&pf);
1778                     }
1779                     /* Fall through */
1780                 case IDCANCEL:
1781                     EndDialog(hWnd, wParam);
1782                     return TRUE;
1783             }
1784     }
1785     return FALSE;
1786 }
1787 
1788 static LRESULT OnCreate( HWND hWnd )
1789 {
1790     HWND hToolBarWnd, hFormatBarWnd,  hReBarWnd, hFontListWnd, hSizeListWnd, hRulerWnd;
1791     HINSTANCE hInstance = GetModuleHandleW(0);
1792     HANDLE hDLL;
1793     TBADDBITMAP ab;
1794     int nStdBitmaps = 0;
1795     REBARINFO rbi;
1796     REBARBANDINFOW rbb;
1797     static const WCHAR wszRichEditDll[] = {'R','I','C','H','E','D','2','0','.','D','L','L','\0'};
1798     static const WCHAR wszRichEditText[] = {'R','i','c','h','E','d','i','t',' ','t','e','x','t','\0'};
1799 
1800     CreateStatusWindowW(CCS_NODIVIDER|WS_CHILD|WS_VISIBLE, wszRichEditText, hWnd, IDC_STATUSBAR);
1801 
1802     hReBarWnd = CreateWindowExW(WS_EX_TOOLWINDOW, REBARCLASSNAMEW, NULL,
1803       CCS_NODIVIDER|WS_CHILD|WS_VISIBLE|WS_CLIPSIBLINGS|WS_CLIPCHILDREN|RBS_VARHEIGHT|CCS_TOP,
1804       CW_USEDEFAULT, CW_USEDEFAULT, 0, 0, hWnd, (HMENU)IDC_REBAR, hInstance, NULL);
1805 
1806     rbi.cbSize = sizeof(rbi);
1807     rbi.fMask = 0;
1808     rbi.himl = NULL;
1809     if(!SendMessageW(hReBarWnd, RB_SETBARINFO, 0, (LPARAM)&rbi))
1810         return -1;
1811 
1812     hToolBarWnd = CreateToolbarEx(hReBarWnd, CCS_NOPARENTALIGN|CCS_NOMOVEY|WS_VISIBLE|WS_CHILD|TBSTYLE_TOOLTIPS|BTNS_BUTTON,
1813       IDC_TOOLBAR,
1814       1, hInstance, IDB_TOOLBAR,
1815       NULL, 0,
1816       24, 24, 16, 16, sizeof(TBBUTTON));
1817 
1818     ab.hInst = HINST_COMMCTRL;
1819     ab.nID = IDB_STD_SMALL_COLOR;
1820     nStdBitmaps = SendMessageW(hToolBarWnd, TB_ADDBITMAP, 0, (LPARAM)&ab);
1821 
1822     AddButton(hToolBarWnd, nStdBitmaps+STD_FILENEW, ID_FILE_NEW);
1823     AddButton(hToolBarWnd, nStdBitmaps+STD_FILEOPEN, ID_FILE_OPEN);
1824     AddButton(hToolBarWnd, nStdBitmaps+STD_FILESAVE, ID_FILE_SAVE);
1825     AddSeparator(hToolBarWnd);
1826     AddButton(hToolBarWnd, nStdBitmaps+STD_PRINT, ID_PRINT_QUICK);
1827     AddButton(hToolBarWnd, nStdBitmaps+STD_PRINTPRE, ID_PREVIEW);
1828     AddSeparator(hToolBarWnd);
1829     AddButton(hToolBarWnd, nStdBitmaps+STD_FIND, ID_FIND);
1830     AddSeparator(hToolBarWnd);
1831     AddButton(hToolBarWnd, nStdBitmaps+STD_CUT, ID_EDIT_CUT);
1832     AddButton(hToolBarWnd, nStdBitmaps+STD_COPY, ID_EDIT_COPY);
1833     AddButton(hToolBarWnd, nStdBitmaps+STD_PASTE, ID_EDIT_PASTE);
1834     AddButton(hToolBarWnd, nStdBitmaps+STD_UNDO, ID_EDIT_UNDO);
1835     AddButton(hToolBarWnd, nStdBitmaps+STD_REDOW, ID_EDIT_REDO);
1836     AddSeparator(hToolBarWnd);
1837     AddButton(hToolBarWnd, 0, ID_DATETIME);
1838 
1839     SendMessageW(hToolBarWnd, TB_AUTOSIZE, 0, 0);
1840 
1841     rbb.cbSize = REBARBANDINFOW_V6_SIZE;
1842     rbb.fMask = RBBIM_SIZE | RBBIM_CHILDSIZE | RBBIM_CHILD | RBBIM_STYLE | RBBIM_ID;
1843     rbb.fStyle = RBBS_CHILDEDGE | RBBS_BREAK | RBBS_NOGRIPPER;
1844     rbb.cx = 0;
1845     rbb.hwndChild = hToolBarWnd;
1846     rbb.cxMinChild = 0;
1847     rbb.cyChild = rbb.cyMinChild = HIWORD(SendMessageW(hToolBarWnd, TB_GETBUTTONSIZE, 0, 0));
1848     rbb.wID = BANDID_TOOLBAR;
1849 
1850     SendMessageW(hReBarWnd, RB_INSERTBANDW, -1, (LPARAM)&rbb);
1851 
1852     hFontListWnd = CreateWindowExW(0, WC_COMBOBOXEXW, NULL,
1853                       WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN | CBS_SORT,
1854                       0, 0, 200, 150, hReBarWnd, (HMENU)IDC_FONTLIST, hInstance, NULL);
1855 
1856     rbb.hwndChild = hFontListWnd;
1857     rbb.cx = 200;
1858     rbb.wID = BANDID_FONTLIST;
1859 
1860     SendMessageW(hReBarWnd, RB_INSERTBANDW, -1, (LPARAM)&rbb);
1861 
1862     hSizeListWnd = CreateWindowExW(0, WC_COMBOBOXEXW, NULL,
1863                       WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN,
1864                       0, 0, 50, 150, hReBarWnd, (HMENU)IDC_SIZELIST, hInstance, NULL);
1865 
1866     rbb.hwndChild = hSizeListWnd;
1867     rbb.cx = 50;
1868     rbb.fStyle ^= RBBS_BREAK;
1869     rbb.wID = BANDID_SIZELIST;
1870 
1871     SendMessageW(hReBarWnd, RB_INSERTBANDW, -1, (LPARAM)&rbb);
1872 
1873     hFormatBarWnd = CreateToolbarEx(hReBarWnd,
1874          CCS_NOPARENTALIGN | CCS_NOMOVEY | WS_VISIBLE | TBSTYLE_TOOLTIPS | BTNS_BUTTON,
1875          IDC_FORMATBAR, 8, hInstance, IDB_FORMATBAR, NULL, 0, 16, 16, 16, 16, sizeof(TBBUTTON));
1876 
1877     AddButton(hFormatBarWnd, 0, ID_FORMAT_BOLD);
1878     AddButton(hFormatBarWnd, 1, ID_FORMAT_ITALIC);
1879     AddButton(hFormatBarWnd, 2, ID_FORMAT_UNDERLINE);
1880     AddButton(hFormatBarWnd, 3, ID_FORMAT_COLOR);
1881     AddSeparator(hFormatBarWnd);
1882     AddButton(hFormatBarWnd, 4, ID_ALIGN_LEFT);
1883     AddButton(hFormatBarWnd, 5, ID_ALIGN_CENTER);
1884     AddButton(hFormatBarWnd, 6, ID_ALIGN_RIGHT);
1885     AddSeparator(hFormatBarWnd);
1886     AddButton(hFormatBarWnd, 7, ID_BULLET);
1887 
1888     SendMessageW(hFormatBarWnd, TB_AUTOSIZE, 0, 0);
1889 
1890     rbb.hwndChild = hFormatBarWnd;
1891     rbb.wID = BANDID_FORMATBAR;
1892 
1893     SendMessageW(hReBarWnd, RB_INSERTBANDW, -1, (LPARAM)&rbb);
1894 
1895     hRulerWnd = CreateWindowExW(0, WC_STATICW, NULL, WS_VISIBLE | WS_CHILD,
1896                                 0, 0, 200, 10, hReBarWnd,  (HMENU)IDC_RULER, hInstance, NULL);
1897 
1898 
1899     rbb.hwndChild = hRulerWnd;
1900     rbb.wID = BANDID_RULER;
1901     rbb.fStyle |= RBBS_BREAK;
1902 
1903     SendMessageW(hReBarWnd, RB_INSERTBANDW, -1, (LPARAM)&rbb);
1904 
1905     hDLL = LoadLibraryW(wszRichEditDll);
1906     if(!hDLL)
1907     {
1908         MessageBoxWithResStringW(hWnd, MAKEINTRESOURCEW(STRING_LOAD_RICHED_FAILED), wszAppTitle,
1909                     MB_OK | MB_ICONEXCLAMATION);
1910         PostQuitMessage(1);
1911     }
1912 
1913     hEditorWnd = CreateWindowExW(WS_EX_CLIENTEDGE, RICHEDIT_CLASS20W, NULL,
1914       WS_CHILD|WS_VISIBLE|ES_SELECTIONBAR|ES_MULTILINE|ES_AUTOVSCROLL
1915       |ES_WANTRETURN|WS_VSCROLL|ES_NOHIDESEL|WS_HSCROLL,
1916       0, 0, 1000, 100, hWnd, (HMENU)IDC_EDITOR, hInstance, NULL);
1917 
1918     if (!hEditorWnd)
1919     {
1920         fprintf(stderr, "Error code %u\n", GetLastError());
1921         return -1;
1922     }
1923     assert(hEditorWnd);
1924 
1925     setup_richedit_olecallback(hEditorWnd);
1926     SetFocus(hEditorWnd);
1927     SendMessageW(hEditorWnd, EM_SETEVENTMASK, 0, ENM_SELCHANGE);
1928 
1929     set_default_font();
1930 
1931     populate_font_list(hFontListWnd);
1932     populate_size_list(hSizeListWnd);
1933     DoLoadStrings();
1934     SendMessageW(hEditorWnd, EM_SETMODIFY, FALSE, 0);
1935 
1936     ID_FINDMSGSTRING = RegisterWindowMessageW(FINDMSGSTRINGW);
1937 
1938     registry_read_filelist(hWnd);
1939     registry_read_formatopts_all(barState, wordWrap);
1940     registry_read_options();
1941     DragAcceptFiles(hWnd, TRUE);
1942 
1943     return 0;
1944 }
1945 
1946 static LRESULT OnUser( HWND hWnd )
1947 {
1948     HWND hwndEditor = GetDlgItem(hWnd, IDC_EDITOR);
1949     HWND hwndReBar = GetDlgItem(hWnd, IDC_REBAR);
1950     HWND hwndToolBar = GetDlgItem(hwndReBar, IDC_TOOLBAR);
1951     HWND hwndFormatBar = GetDlgItem(hwndReBar, IDC_FORMATBAR);
1952     int from, to;
1953     CHARFORMAT2W fmt;
1954     PARAFORMAT2 pf;
1955     GETTEXTLENGTHEX gt;
1956 
1957     ZeroMemory(&fmt, sizeof(fmt));
1958     fmt.cbSize = sizeof(fmt);
1959 
1960     ZeroMemory(&pf, sizeof(pf));
1961     pf.cbSize = sizeof(pf);
1962 
1963     gt.flags = GTL_NUMCHARS;
1964     gt.codepage = 1200;
1965 
1966     SendMessageW(hwndToolBar, TB_ENABLEBUTTON, ID_FIND,
1967                  SendMessageW(hwndEditor, EM_GETTEXTLENGTHEX, (WPARAM)&gt, 0) ? 1 : 0);
1968 
1969     SendMessageW(hwndEditor, EM_GETCHARFORMAT, TRUE, (LPARAM)&fmt);
1970 
1971     SendMessageW(hwndEditor, EM_GETSEL, (WPARAM)&from, (LPARAM)&to);
1972     SendMessageW(hwndToolBar, TB_ENABLEBUTTON, ID_EDIT_UNDO,
1973       SendMessageW(hwndEditor, EM_CANUNDO, 0, 0));
1974     SendMessageW(hwndToolBar, TB_ENABLEBUTTON, ID_EDIT_REDO,
1975       SendMessageW(hwndEditor, EM_CANREDO, 0, 0));
1976     SendMessageW(hwndToolBar, TB_ENABLEBUTTON, ID_EDIT_CUT, from == to ? 0 : 1);
1977     SendMessageW(hwndToolBar, TB_ENABLEBUTTON, ID_EDIT_COPY, from == to ? 0 : 1);
1978 
1979     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_FORMAT_BOLD, (fmt.dwMask & CFM_BOLD) &&
1980             (fmt.dwEffects & CFE_BOLD));
1981     SendMessageW(hwndFormatBar, TB_INDETERMINATE, ID_FORMAT_BOLD, !(fmt.dwMask & CFM_BOLD));
1982     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_FORMAT_ITALIC, (fmt.dwMask & CFM_ITALIC) &&
1983             (fmt.dwEffects & CFE_ITALIC));
1984     SendMessageW(hwndFormatBar, TB_INDETERMINATE, ID_FORMAT_ITALIC, !(fmt.dwMask & CFM_ITALIC));
1985     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_FORMAT_UNDERLINE, (fmt.dwMask & CFM_UNDERLINE) &&
1986             (fmt.dwEffects & CFE_UNDERLINE));
1987     SendMessageW(hwndFormatBar, TB_INDETERMINATE, ID_FORMAT_UNDERLINE, !(fmt.dwMask & CFM_UNDERLINE));
1988 
1989     SendMessageW(hwndEditor, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
1990     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_ALIGN_LEFT, (pf.wAlignment == PFA_LEFT));
1991     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_ALIGN_CENTER, (pf.wAlignment == PFA_CENTER));
1992     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_ALIGN_RIGHT, (pf.wAlignment == PFA_RIGHT));
1993 
1994     SendMessageW(hwndFormatBar, TB_CHECKBUTTON, ID_BULLET, (pf.wNumbering & PFN_BULLET));
1995     return 0;
1996 }
1997 
1998 static LRESULT OnNotify( HWND hWnd, LPARAM lParam)
1999 {
2000     HWND hwndEditor = GetDlgItem(hWnd, IDC_EDITOR);
2001     HWND hwndReBar = GetDlgItem(hWnd, IDC_REBAR);
2002     NMHDR *pHdr = (NMHDR *)lParam;
2003     HWND hwndFontList = GetDlgItem(hwndReBar, IDC_FONTLIST);
2004     HWND hwndSizeList = GetDlgItem(hwndReBar, IDC_SIZELIST);
2005 
2006     if (pHdr->hwndFrom == hwndFontList || pHdr->hwndFrom == hwndSizeList)
2007     {
2008         if (pHdr->code == CBEN_ENDEDITW)
2009         {
2010             NMCBEENDEDITW *endEdit = (NMCBEENDEDITW *)lParam;
2011             if(pHdr->hwndFrom == hwndFontList)
2012             {
2013                 on_fontlist_modified(endEdit->szText);
2014             } else if (pHdr->hwndFrom == hwndSizeList)
2015             {
2016                 on_sizelist_modified(hwndSizeList,endEdit->szText);
2017             }
2018         }
2019         return 0;
2020     }
2021 
2022     if (pHdr->hwndFrom != hwndEditor)
2023         return 0;
2024 
2025     if (pHdr->code == EN_SELCHANGE)
2026     {
2027         SELCHANGE *pSC = (SELCHANGE *)lParam;
2028         char buf[128];
2029 
2030         update_font_list();
2031 
2032         sprintf( buf,"selection = %d..%d, line count=%ld",
2033                  pSC->chrg.cpMin, pSC->chrg.cpMax,
2034                 SendMessageW(hwndEditor, EM_GETLINECOUNT, 0, 0));
2035         SetWindowTextA(GetDlgItem(hWnd, IDC_STATUSBAR), buf);
2036         SendMessageW(hWnd, WM_USER, 0, 0);
2037         return 1;
2038     }
2039     return 0;
2040 }
2041 
2042 /* Copied from dlls/comdlg32/fontdlg.c */
2043 static const COLORREF textcolors[]=
2044 {
2045     0x00000000L,0x00000080L,0x00008000L,0x00008080L,
2046     0x00800000L,0x00800080L,0x00808000L,0x00808080L,
2047     0x00c0c0c0L,0x000000ffL,0x0000ff00L,0x0000ffffL,
2048     0x00ff0000L,0x00ff00ffL,0x00ffff00L,0x00FFFFFFL
2049 };
2050 
2051 static LRESULT OnCommand( HWND hWnd, WPARAM wParam, LPARAM lParam)
2052 {
2053     HWND hwndEditor = GetDlgItem(hWnd, IDC_EDITOR);
2054     static FINDREPLACEW findreplace;
2055 
2056     if ((HWND)lParam == hwndEditor)
2057         return 0;
2058 
2059     switch(LOWORD(wParam))
2060     {
2061 
2062     case ID_FILE_EXIT:
2063         PostMessageW(hWnd, WM_CLOSE, 0, 0);
2064         break;
2065 
2066     case ID_FILE_NEW:
2067         {
2068             HINSTANCE hInstance = GetModuleHandleW(0);
2069             int ret = DialogBoxW(hInstance, MAKEINTRESOURCEW(IDD_NEWFILE), hWnd, newfile_proc);
2070 
2071             if(ret != ID_NEWFILE_ABORT)
2072             {
2073                 if(prompt_save_changes())
2074                 {
2075                     SETTEXTEX st;
2076 
2077                     set_caption(NULL);
2078                     wszFileName[0] = '\0';
2079 
2080                     clear_formatting();
2081 
2082                     st.flags = ST_DEFAULT;
2083                     st.codepage = 1200;
2084                     SendMessageW(hEditorWnd, EM_SETTEXTEX, (WPARAM)&st, 0);
2085 
2086                     SendMessageW(hEditorWnd, EM_SETMODIFY, FALSE, 0);
2087                     set_fileformat(ret);
2088                     update_font_list();
2089                 }
2090             }
2091         }
2092         break;
2093 
2094     case ID_FILE_OPEN:
2095         DialogOpenFile();
2096         break;
2097 
2098     case ID_FILE_SAVE:
2099         if(wszFileName[0])
2100         {
2101             DoSaveFile(wszFileName, fileFormat);
2102             break;
2103         }
2104         /* Fall through */
2105 
2106     case ID_FILE_SAVEAS:
2107         DialogSaveFile();
2108         break;
2109 
2110     case ID_FILE_RECENT1:
2111     case ID_FILE_RECENT2:
2112     case ID_FILE_RECENT3:
2113     case ID_FILE_RECENT4:
2114         {
2115             HMENU hMenu = GetMenu(hWnd);
2116             MENUITEMINFOW mi;
2117 
2118             mi.cbSize = sizeof(MENUITEMINFOW);
2119             mi.fMask = MIIM_DATA;
2120             if(GetMenuItemInfoW(hMenu, LOWORD(wParam), FALSE, &mi))
2121                 DoOpenFile((LPWSTR)mi.dwItemData);
2122         }
2123         break;
2124 
2125     case ID_FIND:
2126         dialog_find(&findreplace, FALSE);
2127         break;
2128 
2129     case ID_FIND_NEXT:
2130         handle_findmsg(&findreplace);
2131         break;
2132 
2133     case ID_REPLACE:
2134         dialog_find(&findreplace, TRUE);
2135         break;
2136 
2137     case ID_FONTSETTINGS:
2138         dialog_choose_font();
2139         break;
2140 
2141     case ID_PRINT:
2142         dialog_print(hWnd, wszFileName);
2143         target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
2144         break;
2145 
2146     case ID_PRINT_QUICK:
2147         print_quick(hMainWnd, wszFileName);
2148         target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
2149         break;
2150 
2151     case ID_PREVIEW:
2152         {
2153             int index = reg_formatindex(fileFormat);
2154             DWORD tmp = barState[index];
2155             barState[index] = 1 << BANDID_STATUSBAR;
2156             set_bar_states();
2157             barState[index] = tmp;
2158             ShowWindow(hEditorWnd, FALSE);
2159 
2160             init_preview(hWnd, wszFileName);
2161 
2162             SetMenu(hWnd, NULL);
2163             InvalidateRect(0, 0, TRUE);
2164         }
2165         break;
2166 
2167     case ID_PRINTSETUP:
2168         dialog_printsetup(hWnd);
2169         target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
2170         break;
2171 
2172     case ID_FORMAT_BOLD:
2173     case ID_FORMAT_ITALIC:
2174     case ID_FORMAT_UNDERLINE:
2175         {
2176         CHARFORMAT2W fmt;
2177         int effects = CFE_BOLD;
2178 
2179         ZeroMemory(&fmt, sizeof(fmt));
2180         fmt.cbSize = sizeof(fmt);
2181         SendMessageW(hwndEditor, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
2182 
2183         fmt.dwMask = CFM_BOLD;
2184 
2185         if (LOWORD(wParam) == ID_FORMAT_ITALIC)
2186         {
2187             effects = CFE_ITALIC;
2188             fmt.dwMask = CFM_ITALIC;
2189         } else if (LOWORD(wParam) == ID_FORMAT_UNDERLINE)
2190         {
2191             effects = CFE_UNDERLINE;
2192             fmt.dwMask = CFM_UNDERLINE;
2193         }
2194 
2195         fmt.dwEffects ^= effects;
2196 
2197         SendMessageW(hwndEditor, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
2198         break;
2199         }
2200 
2201     case ID_FORMAT_COLOR:
2202     {
2203         HWND hReBarWnd = GetDlgItem(hWnd, IDC_REBAR);
2204         HWND hFormatBarWnd = GetDlgItem(hReBarWnd, IDC_FORMATBAR);
2205         HMENU hPop;
2206         RECT itemrc;
2207         POINT pt;
2208         int mid;
2209         int itemidx = SendMessageW(hFormatBarWnd, TB_COMMANDTOINDEX, ID_FORMAT_COLOR, 0);
2210 
2211         SendMessageW(hFormatBarWnd, TB_GETITEMRECT, itemidx, (LPARAM)&itemrc);
2212         pt.x = itemrc.left;
2213         pt.y = itemrc.bottom;
2214         ClientToScreen(hFormatBarWnd, &pt);
2215         hPop = GetSubMenu(hColorPopupMenu, 0);
2216         mid = TrackPopupMenu(hPop, TPM_LEFTALIGN | TPM_TOPALIGN | TPM_LEFTBUTTON |
2217                                    TPM_RETURNCMD | TPM_NONOTIFY,
2218                              pt.x, pt.y, 0, hWnd, 0);
2219         if (mid >= ID_COLOR_FIRST && mid <= ID_COLOR_AUTOMATIC)
2220         {
2221             CHARFORMAT2W fmt;
2222 
2223             ZeroMemory(&fmt, sizeof(fmt));
2224             fmt.cbSize = sizeof(fmt);
2225             SendMessageW(hwndEditor, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
2226 
2227             fmt.dwMask = CFM_COLOR;
2228 
2229             if (mid < ID_COLOR_AUTOMATIC) {
2230                 fmt.crTextColor = textcolors[mid - ID_COLOR_FIRST];
2231                 fmt.dwEffects &= ~CFE_AUTOCOLOR;
2232             } else {
2233                 fmt.dwEffects |= CFE_AUTOCOLOR;
2234             }
2235 
2236             SendMessageW(hwndEditor, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&fmt);
2237         }
2238         break;
2239     }
2240 
2241     case ID_EDIT_CUT:
2242         PostMessageW(hwndEditor, WM_CUT, 0, 0);
2243         break;
2244 
2245     case ID_EDIT_COPY:
2246         PostMessageW(hwndEditor, WM_COPY, 0, 0);
2247         break;
2248 
2249     case ID_EDIT_PASTE:
2250         PostMessageW(hwndEditor, WM_PASTE, 0, 0);
2251         break;
2252 
2253     case ID_EDIT_CLEAR:
2254         PostMessageW(hwndEditor, WM_CLEAR, 0, 0);
2255         break;
2256 
2257     case ID_EDIT_SELECTALL:
2258         {
2259         CHARRANGE range = {0, -1};
2260         SendMessageW(hwndEditor, EM_EXSETSEL, 0, (LPARAM)&range);
2261         /* SendMessage(hwndEditor, EM_SETSEL, 0, -1); */
2262         return 0;
2263         }
2264 
2265     case ID_EDIT_GETTEXT:
2266         {
2267         int nLen = GetWindowTextLengthW(hwndEditor);
2268         LPWSTR data = HeapAlloc( GetProcessHeap(), 0, (nLen+1)*sizeof(WCHAR) );
2269         TEXTRANGEW tr;
2270 
2271         GetWindowTextW(hwndEditor, data, nLen+1);
2272         MessageBoxW(NULL, data, wszAppTitle, MB_OK);
2273 
2274         HeapFree( GetProcessHeap(), 0, data);
2275         data = HeapAlloc(GetProcessHeap(), 0, (nLen+1)*sizeof(WCHAR));
2276         tr.chrg.cpMin = 0;
2277         tr.chrg.cpMax = nLen;
2278         tr.lpstrText = data;
2279         SendMessageW(hwndEditor, EM_GETTEXTRANGE, 0, (LPARAM)&tr);
2280         MessageBoxW(NULL, data, wszAppTitle, MB_OK);
2281         HeapFree( GetProcessHeap(), 0, data );
2282 
2283         /* SendMessage(hwndEditor, EM_SETSEL, 0, -1); */
2284         return 0;
2285         }
2286 
2287     case ID_EDIT_CHARFORMAT:
2288     case ID_EDIT_DEFCHARFORMAT:
2289         {
2290         CHARFORMAT2W cf;
2291 
2292         ZeroMemory(&cf, sizeof(cf));
2293         cf.cbSize = sizeof(cf);
2294         cf.dwMask = 0;
2295         SendMessageW(hwndEditor, EM_GETCHARFORMAT,
2296                      LOWORD(wParam) == ID_EDIT_CHARFORMAT, (LPARAM)&cf);
2297         return 0;
2298         }
2299 
2300     case ID_EDIT_PARAFORMAT:
2301         {
2302         PARAFORMAT2 pf;
2303         ZeroMemory(&pf, sizeof(pf));
2304         pf.cbSize = sizeof(pf);
2305         SendMessageW(hwndEditor, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
2306         return 0;
2307         }
2308 
2309     case ID_EDIT_SELECTIONINFO:
2310         {
2311         CHARRANGE range = {0, -1};
2312         char buf[128];
2313         WCHAR *data = NULL;
2314 
2315         SendMessageW(hwndEditor, EM_EXGETSEL, 0, (LPARAM)&range);
2316         data = HeapAlloc(GetProcessHeap(), 0, sizeof(*data) * (range.cpMax-range.cpMin+1));
2317         SendMessageW(hwndEditor, EM_GETSELTEXT, 0, (LPARAM)data);
2318         sprintf(buf, "Start = %d, End = %d", range.cpMin, range.cpMax);
2319         MessageBoxA(hWnd, buf, "Editor", MB_OK);
2320         MessageBoxW(hWnd, data, wszAppTitle, MB_OK);
2321         HeapFree( GetProcessHeap(), 0, data);
2322         /* SendMessage(hwndEditor, EM_SETSEL, 0, -1); */
2323         return 0;
2324         }
2325 
2326     case ID_EDIT_READONLY:
2327         {
2328         LONG nStyle = GetWindowLongW(hwndEditor, GWL_STYLE);
2329         if (nStyle & ES_READONLY)
2330             SendMessageW(hwndEditor, EM_SETREADONLY, 0, 0);
2331         else
2332             SendMessageW(hwndEditor, EM_SETREADONLY, 1, 0);
2333         return 0;
2334         }
2335 
2336     case ID_EDIT_MODIFIED:
2337         if (SendMessageW(hwndEditor, EM_GETMODIFY, 0, 0))
2338             SendMessageW(hwndEditor, EM_SETMODIFY, 0, 0);
2339         else
2340             SendMessageW(hwndEditor, EM_SETMODIFY, 1, 0);
2341         return 0;
2342 
2343     case ID_EDIT_UNDO:
2344         SendMessageW(hwndEditor, EM_UNDO, 0, 0);
2345         return 0;
2346 
2347     case ID_EDIT_REDO:
2348         SendMessageW(hwndEditor, EM_REDO, 0, 0);
2349         return 0;
2350 
2351     case ID_BULLET:
2352         {
2353             PARAFORMAT pf;
2354 
2355             pf.cbSize = sizeof(pf);
2356             pf.dwMask = PFM_NUMBERING;
2357             SendMessageW(hwndEditor, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
2358 
2359             pf.dwMask |=  PFM_OFFSET;
2360 
2361             if(pf.wNumbering == PFN_BULLET)
2362             {
2363                 pf.wNumbering = 0;
2364                 pf.dxOffset = 0;
2365             } else
2366             {
2367                 pf.wNumbering = PFN_BULLET;
2368                 pf.dxOffset = 720;
2369             }
2370 
2371             SendMessageW(hwndEditor, EM_SETPARAFORMAT, 0, (LPARAM)&pf);
2372         }
2373         break;
2374 
2375     case ID_ALIGN_LEFT:
2376     case ID_ALIGN_CENTER:
2377     case ID_ALIGN_RIGHT:
2378         {
2379         PARAFORMAT2 pf;
2380 
2381         pf.cbSize = sizeof(pf);
2382         pf.dwMask = PFM_ALIGNMENT;
2383         switch(LOWORD(wParam)) {
2384         case ID_ALIGN_LEFT: pf.wAlignment = PFA_LEFT; break;
2385         case ID_ALIGN_CENTER: pf.wAlignment = PFA_CENTER; break;
2386         case ID_ALIGN_RIGHT: pf.wAlignment = PFA_RIGHT; break;
2387         }
2388         SendMessageW(hwndEditor, EM_SETPARAFORMAT, 0, (LPARAM)&pf);
2389         break;
2390         }
2391 
2392     case ID_BACK_1:
2393         SendMessageW(hwndEditor, EM_SETBKGNDCOLOR, 1, 0);
2394         break;
2395 
2396     case ID_BACK_2:
2397         SendMessageW(hwndEditor, EM_SETBKGNDCOLOR, 0, RGB(255,255,192));
2398         break;
2399 
2400     case ID_TOGGLE_TOOLBAR:
2401         set_toolbar_state(BANDID_TOOLBAR, !is_bar_visible(BANDID_TOOLBAR));
2402         update_window();
2403         break;
2404 
2405     case ID_TOGGLE_FORMATBAR:
2406         set_toolbar_state(BANDID_FONTLIST, !is_bar_visible(BANDID_FORMATBAR));
2407         set_toolbar_state(BANDID_SIZELIST, !is_bar_visible(BANDID_FORMATBAR));
2408         set_toolbar_state(BANDID_FORMATBAR, !is_bar_visible(BANDID_FORMATBAR));
2409         update_window();
2410         break;
2411 
2412     case ID_TOGGLE_STATUSBAR:
2413         set_statusbar_state(!is_bar_visible(BANDID_STATUSBAR));
2414         update_window();
2415         break;
2416 
2417     case ID_TOGGLE_RULER:
2418         set_toolbar_state(BANDID_RULER, !is_bar_visible(BANDID_RULER));
2419         update_window();
2420         break;
2421 
2422     case ID_DATETIME:
2423         DialogBoxW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDD_DATETIME), hWnd, datetime_proc);
2424         break;
2425 
2426     case ID_PARAFORMAT:
2427         DialogBoxW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDD_PARAFORMAT), hWnd, paraformat_proc);
2428         break;
2429 
2430     case ID_TABSTOPS:
2431         DialogBoxW(GetModuleHandleW(0), MAKEINTRESOURCEW(IDD_TABSTOPS), hWnd, tabstops_proc);
2432         break;
2433 
2434     case ID_ABOUT:
2435         dialog_about();
2436         break;
2437 
2438     case ID_VIEWPROPERTIES:
2439         dialog_viewproperties();
2440         break;
2441 
2442     case IDC_FONTLIST:
2443         if (HIWORD(wParam) == CBN_SELENDOK)
2444         {
2445             WCHAR buffer[LF_FACESIZE];
2446             HWND hwndFontList = (HWND)lParam;
2447             get_comboexlist_selection(hwndFontList, buffer, LF_FACESIZE);
2448             on_fontlist_modified(buffer);
2449         }
2450         break;
2451 
2452     case IDC_SIZELIST:
2453         if (HIWORD(wParam) == CBN_SELENDOK)
2454         {
2455             WCHAR buffer[MAX_STRING_LEN+1];
2456             HWND hwndSizeList = (HWND)lParam;
2457             get_comboexlist_selection(hwndSizeList, buffer, MAX_STRING_LEN+1);
2458             on_sizelist_modified(hwndSizeList, buffer);
2459         }
2460         break;
2461 
2462     default:
2463         SendMessageW(hwndEditor, WM_COMMAND, wParam, lParam);
2464         break;
2465     }
2466     return 0;
2467 }
2468 
2469 static LRESULT OnInitPopupMenu( HWND hWnd, WPARAM wParam )
2470 {
2471     HMENU hMenu = (HMENU)wParam;
2472     HWND hwndEditor = GetDlgItem(hWnd, IDC_EDITOR);
2473     HWND hwndStatus = GetDlgItem(hWnd, IDC_STATUSBAR);
2474     PARAFORMAT pf;
2475     int nAlignment = -1;
2476     int selFrom, selTo;
2477     GETTEXTLENGTHEX gt;
2478     LRESULT textLength;
2479     MENUITEMINFOW mi;
2480 
2481     SendMessageW(hEditorWnd, EM_GETSEL, (WPARAM)&selFrom, (LPARAM)&selTo);
2482     EnableMenuItem(hMenu, ID_EDIT_COPY, (selFrom == selTo) ? MF_GRAYED : MF_ENABLED);
2483     EnableMenuItem(hMenu, ID_EDIT_CUT, (selFrom == selTo) ? MF_GRAYED : MF_ENABLED);
2484 
2485     pf.cbSize = sizeof(PARAFORMAT);
2486     SendMessageW(hwndEditor, EM_GETPARAFORMAT, 0, (LPARAM)&pf);
2487     CheckMenuItem(hMenu, ID_EDIT_READONLY,
2488       (GetWindowLongW(hwndEditor, GWL_STYLE) & ES_READONLY) ? MF_CHECKED : MF_UNCHECKED);
2489     CheckMenuItem(hMenu, ID_EDIT_MODIFIED,
2490       SendMessageW(hwndEditor, EM_GETMODIFY, 0, 0) ? MF_CHECKED : MF_UNCHECKED);
2491     if (pf.dwMask & PFM_ALIGNMENT)
2492         nAlignment = pf.wAlignment;
2493     CheckMenuItem(hMenu, ID_ALIGN_LEFT, (nAlignment == PFA_LEFT) ?  MF_CHECKED : MF_UNCHECKED);
2494     CheckMenuItem(hMenu, ID_ALIGN_CENTER, (nAlignment == PFA_CENTER) ?  MF_CHECKED : MF_UNCHECKED);
2495     CheckMenuItem(hMenu, ID_ALIGN_RIGHT, (nAlignment == PFA_RIGHT) ?  MF_CHECKED : MF_UNCHECKED);
2496     CheckMenuItem(hMenu, ID_BULLET, ((pf.wNumbering == PFN_BULLET) ?  MF_CHECKED : MF_UNCHECKED));
2497     EnableMenuItem(hMenu, ID_EDIT_UNDO, SendMessageW(hwndEditor, EM_CANUNDO, 0, 0) ?
2498             MF_ENABLED : MF_GRAYED);
2499     EnableMenuItem(hMenu, ID_EDIT_REDO, SendMessageW(hwndEditor, EM_CANREDO, 0, 0) ?
2500             MF_ENABLED : MF_GRAYED);
2501 
2502     CheckMenuItem(hMenu, ID_TOGGLE_TOOLBAR, is_bar_visible(BANDID_TOOLBAR) ?
2503             MF_CHECKED : MF_UNCHECKED);
2504 
2505     CheckMenuItem(hMenu, ID_TOGGLE_FORMATBAR, is_bar_visible(BANDID_FORMATBAR) ?
2506             MF_CHECKED : MF_UNCHECKED);
2507 
2508     CheckMenuItem(hMenu, ID_TOGGLE_STATUSBAR, IsWindowVisible(hwndStatus) ?
2509             MF_CHECKED : MF_UNCHECKED);
2510 
2511     CheckMenuItem(hMenu, ID_TOGGLE_RULER, is_bar_visible(BANDID_RULER) ? MF_CHECKED : MF_UNCHECKED);
2512 
2513     gt.flags = GTL_NUMCHARS;
2514     gt.codepage = 1200;
2515     textLength = SendMessageW(hEditorWnd, EM_GETTEXTLENGTHEX, (WPARAM)&gt, 0);
2516     EnableMenuItem(hMenu, ID_FIND, textLength ? MF_ENABLED : MF_GRAYED);
2517 
2518     mi.cbSize = sizeof(mi);
2519     mi.fMask = MIIM_DATA;
2520 
2521     GetMenuItemInfoW(hMenu, ID_FIND_NEXT, FALSE, &mi);
2522 
2523     EnableMenuItem(hMenu, ID_FIND_NEXT, (textLength && mi.dwItemData) ?  MF_ENABLED : MF_GRAYED);
2524 
2525     EnableMenuItem(hMenu, ID_REPLACE, textLength ? MF_ENABLED : MF_GRAYED);
2526 
2527     return 0;
2528 }
2529 
2530 static LRESULT OnSize( HWND hWnd, WPARAM wParam, LPARAM lParam )
2531 {
2532     int nStatusSize = 0;
2533     RECT rc;
2534     HWND hwndEditor = preview_isactive() ? GetDlgItem(hWnd, IDC_PREVIEW) : GetDlgItem(hWnd, IDC_EDITOR);
2535     HWND hwndStatusBar = GetDlgItem(hWnd, IDC_STATUSBAR);
2536     HWND hwndReBar = GetDlgItem(hWnd, IDC_REBAR);
2537     HWND hRulerWnd = GetDlgItem(hwndReBar, IDC_RULER);
2538     int rebarHeight = 0;
2539 
2540     if (hwndStatusBar)
2541     {
2542         SendMessageW(hwndStatusBar, WM_SIZE, 0, 0);
2543         if (IsWindowVisible(hwndStatusBar))
2544         {
2545             GetClientRect(hwndStatusBar, &rc);
2546             nStatusSize = rc.bottom - rc.top;
2547         } else
2548         {
2549             nStatusSize = 0;
2550         }
2551     }
2552     if (hwndReBar)
2553     {
2554         rebarHeight = SendMessageW(hwndReBar, RB_GETBARHEIGHT, 0, 0);
2555 
2556         MoveWindow(hwndReBar, 0, 0, LOWORD(lParam), rebarHeight, TRUE);
2557     }
2558     if (hwndEditor)
2559     {
2560         GetClientRect(hWnd, &rc);
2561         MoveWindow(hwndEditor, 0, rebarHeight, rc.right, rc.bottom-nStatusSize-rebarHeight, TRUE);
2562     }
2563 
2564     redraw_ruler(hRulerWnd);
2565 
2566     return DefWindowProcW(hWnd, WM_SIZE, wParam, lParam);
2567 }
2568 
2569 static LRESULT CALLBACK WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
2570 {
2571     if(msg == ID_FINDMSGSTRING)
2572         return handle_findmsg((LPFINDREPLACEW)lParam);
2573 
2574     switch(msg)
2575     {
2576     case WM_CREATE:
2577         return OnCreate( hWnd );
2578 
2579     case WM_USER:
2580         return OnUser( hWnd );
2581 
2582     case WM_NOTIFY:
2583         return OnNotify( hWnd, lParam );
2584 
2585     case WM_COMMAND:
2586         if(preview_isactive())
2587         {
2588             return preview_command( hWnd, wParam );
2589         }
2590 
2591         return OnCommand( hWnd, wParam, lParam );
2592 
2593     case WM_DESTROY:
2594         PostQuitMessage(0);
2595         break;
2596 
2597     case WM_CLOSE:
2598         if(preview_isactive())
2599         {
2600             preview_exit(hWnd);
2601         } else if(prompt_save_changes())
2602         {
2603             registry_set_options(hMainWnd);
2604             registry_set_formatopts_all(barState, wordWrap);
2605             PostQuitMessage(0);
2606         }
2607         break;
2608 
2609     case WM_ACTIVATE:
2610         if (LOWORD(wParam))
2611             SetFocus(GetDlgItem(hWnd, IDC_EDITOR));
2612         return 0;
2613 
2614     case WM_INITMENUPOPUP:
2615         return OnInitPopupMenu( hWnd, wParam );
2616 
2617     case WM_SIZE:
2618         return OnSize( hWnd, wParam, lParam );
2619 
2620     case WM_CONTEXTMENU:
2621         return DefWindowProcW(hWnd, msg, wParam, lParam);
2622 
2623     case WM_DROPFILES:
2624         {
2625             WCHAR file[MAX_PATH];
2626             DragQueryFileW((HDROP)wParam, 0, file, MAX_PATH);
2627             DragFinish((HDROP)wParam);
2628 
2629             if(prompt_save_changes())
2630                 DoOpenFile(file);
2631         }
2632         break;
2633     case WM_PAINT:
2634         if(!preview_isactive())
2635             return DefWindowProcW(hWnd, msg, wParam, lParam);
2636 
2637     default:
2638         return DefWindowProcW(hWnd, msg, wParam, lParam);
2639     }
2640 
2641     return 0;
2642 }
2643 
2644 int CALLBACK WinMain(HINSTANCE hInstance, HINSTANCE hOldInstance, LPSTR szCmdParagraph, int nCmdShow)
2645 {
2646     INITCOMMONCONTROLSEX classes = {8, ICC_BAR_CLASSES|ICC_COOL_CLASSES|ICC_USEREX_CLASSES};
2647     HACCEL hAccel;
2648     WNDCLASSEXW wc;
2649     MSG msg;
2650     RECT rc;
2651     UINT_PTR hPrevRulerProc;
2652     HWND hRulerWnd;
2653     POINTL EditPoint;
2654     DWORD bMaximized;
2655     static const WCHAR wszAccelTable[] = {'M','A','I','N','A','C','C','E','L',
2656                                           'T','A','B','L','E','\0'};
2657 
2658     InitCommonControlsEx(&classes);
2659 
2660     hAccel = LoadAcceleratorsW(hInstance, wszAccelTable);
2661 
2662     wc.cbSize = sizeof(wc);
2663     wc.style = 0;
2664     wc.lpfnWndProc = WndProc;
2665     wc.cbClsExtra = 0;
2666     wc.cbWndExtra = 4;
2667     wc.hInstance = hInstance;
2668     wc.hIcon = LoadIconW(hInstance, MAKEINTRESOURCEW(IDI_WORDPAD));
2669     wc.hIconSm = LoadImageW(hInstance, MAKEINTRESOURCEW(IDI_WORDPAD), IMAGE_ICON,
2670                             GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), LR_SHARED);
2671     wc.hCursor = LoadCursorW(NULL, (LPWSTR)IDC_IBEAM);
2672     wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
2673     wc.lpszMenuName = MAKEINTRESOURCEW(IDM_MAINMENU);
2674     wc.lpszClassName = wszMainWndClass;
2675     RegisterClassExW(&wc);
2676 
2677     wc.style = 0;
2678     wc.lpfnWndProc = preview_proc;
2679     wc.cbClsExtra = 0;
2680     wc.cbWndExtra = 0;
2681     wc.hInstance = hInstance;
2682     wc.hIcon = NULL;
2683     wc.hIconSm = NULL;
2684     wc.hCursor = LoadCursorW(NULL, (LPWSTR)IDC_IBEAM);
2685     wc.hbrBackground = NULL;
2686     wc.lpszMenuName = NULL;
2687     wc.lpszClassName = wszPreviewWndClass;
2688     RegisterClassExW(&wc);
2689 
2690     registry_read_winrect(&rc);
2691     hMainWnd = CreateWindowExW(0, wszMainWndClass, wszAppTitle, WS_CLIPCHILDREN|WS_OVERLAPPEDWINDOW,
2692       rc.left, rc.top, rc.right-rc.left, rc.bottom-rc.top, NULL, NULL, hInstance, NULL);
2693     registry_read_maximized(&bMaximized);
2694     if ((nCmdShow == SW_SHOWNORMAL || nCmdShow == SW_SHOWDEFAULT)
2695 	     && bMaximized)
2696         nCmdShow = SW_SHOWMAXIMIZED;
2697     ShowWindow(hMainWnd, nCmdShow);
2698 
2699     set_caption(NULL);
2700     set_bar_states();
2701     set_fileformat(SF_RTF);
2702     hColorPopupMenu = LoadMenuW(hInstance, MAKEINTRESOURCEW(IDM_COLOR_POPUP));
2703     get_default_printer_opts();
2704     target_device(hMainWnd, wordWrap[reg_formatindex(fileFormat)]);
2705 
2706     hRulerWnd = GetDlgItem(GetDlgItem(hMainWnd, IDC_REBAR), IDC_RULER);
2707     SendMessageW(GetDlgItem(hMainWnd, IDC_EDITOR), EM_POSFROMCHAR, (WPARAM)&EditPoint, 0);
2708     hPrevRulerProc = SetWindowLongPtrW(hRulerWnd, GWLP_WNDPROC, (UINT_PTR)ruler_proc);
2709     SendMessageW(hRulerWnd, WM_USER, (WPARAM)&EditPoint, hPrevRulerProc);
2710 
2711     HandleCommandLine(GetCommandLineW());
2712 
2713     while(GetMessageW(&msg,0,0,0))
2714     {
2715         if (IsDialogMessageW(hFindWnd, &msg))
2716             continue;
2717 
2718         if (TranslateAcceleratorW(hMainWnd, hAccel, &msg))
2719             continue;
2720         TranslateMessage(&msg);
2721         DispatchMessageW(&msg);
2722         if (!PeekMessageW(&msg, 0, 0, 0, PM_NOREMOVE))
2723             SendMessageW(hMainWnd, WM_USER, 0, 0);
2724     }
2725 
2726     return 0;
2727 }
2728