1 /* Unit test suite for edit control.
3 * Copyright 2004 Vitaliy Margolen
4 * Copyright 2005 C. Scott Ananian
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26 #include "wine/test.h"
29 #define ES_COMBO 0x200
32 #define ID_EDITTEST2 99
36 int en_change, en_maxtext, en_update;
39 static struct edit_notify notifications;
41 static HINSTANCE hinst;
43 static char szEditTest2Class[] = "EditTest2Class";
44 static char szEditTest3Class[] = "EditTest3Class";
45 static char szEditTextPositionClass[] = "EditTextPositionWindowClass";
47 static HWND create_editcontrol (DWORD style, DWORD exstyle)
51 handle = CreateWindowEx(exstyle,
54 ES_AUTOHSCROLL | ES_AUTOVSCROLL | style,
56 NULL, NULL, hinst, NULL);
58 if (winetest_interactive)
59 ShowWindow (handle, SW_SHOW);
63 static HWND create_child_editcontrol (DWORD style, DWORD exstyle)
73 assert(AdjustWindowRect(&rect, WS_OVERLAPPEDWINDOW, FALSE));
75 parentWnd = CreateWindowEx(0,
76 szEditTextPositionClass,
79 CW_USEDEFAULT, CW_USEDEFAULT,
80 rect.right - rect.left, rect.bottom - rect.top,
81 NULL, NULL, hinst, NULL);
84 editWnd = CreateWindowEx(exstyle,
87 WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL | ES_AUTOVSCROLL | style,
89 parentWnd, NULL, hinst, NULL);
91 if (winetest_interactive)
92 ShowWindow (parentWnd, SW_SHOW);
96 static void destroy_child_editcontrol (HWND hwndEdit)
98 if (GetParent(hwndEdit))
99 DestroyWindow(GetParent(hwndEdit));
101 trace("Edit control has no parent!\n");
102 DestroyWindow(hwndEdit);
106 static LONG get_edit_style (HWND hwnd)
108 return GetWindowLongA( hwnd, GWL_STYLE ) & (
110 /* FIXME: not implemented
129 static void set_client_height(HWND Wnd, unsigned Height)
131 RECT ClientRect, WindowRect;
133 GetWindowRect(Wnd, &WindowRect);
134 GetClientRect(Wnd, &ClientRect);
135 SetWindowPos(Wnd, NULL, 0, 0,
136 WindowRect.right - WindowRect.left,
137 Height + (WindowRect.bottom - WindowRect.top) -
138 (ClientRect.bottom - ClientRect.top),
139 SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOZORDER);
141 /* Workaround for a bug in Windows' edit control
143 GetWindowRect(Wnd, &WindowRect);
144 SetWindowPos(Wnd, NULL, 0, 0,
145 WindowRect.right - WindowRect.left + 1,
146 WindowRect.bottom - WindowRect.top + 1,
147 SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOZORDER);
148 SetWindowPos(Wnd, NULL, 0, 0,
149 WindowRect.right - WindowRect.left,
150 WindowRect.bottom - WindowRect.top,
151 SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOZORDER);
153 GetClientRect(Wnd, &ClientRect);
154 ok(ClientRect.bottom - ClientRect.top == Height,
155 "The client height should be %ld, but is %ld\n",
156 (long)Height, (long)(ClientRect.bottom - ClientRect.top));
159 static void test_edit_control_1(void)
166 msMessage.message = WM_KEYDOWN;
168 trace("EDIT: Single line\n");
169 hwEdit = create_editcontrol(0, 0);
170 r = get_edit_style(hwEdit);
171 ok(r == (ES_AUTOVSCROLL | ES_AUTOHSCROLL), "Wrong style expected 0xc0 got: 0x%lx\n", r);
172 for (i=0;i<65535;i++)
174 msMessage.wParam = i;
175 r = SendMessage(hwEdit, WM_GETDLGCODE, 0, (LPARAM) &msMessage);
176 ok(r == (DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTARROWS),
177 "Expected DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTARROWS got %lx\n", r);
179 DestroyWindow (hwEdit);
181 trace("EDIT: Single line want returns\n");
182 hwEdit = create_editcontrol(ES_WANTRETURN, 0);
183 r = get_edit_style(hwEdit);
184 ok(r == (ES_AUTOVSCROLL | ES_AUTOHSCROLL | ES_WANTRETURN), "Wrong style expected 0x10c0 got: 0x%lx\n", r);
185 for (i=0;i<65535;i++)
187 msMessage.wParam = i;
188 r = SendMessage(hwEdit, WM_GETDLGCODE, 0, (LPARAM) &msMessage);
189 ok(r == (DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTARROWS),
190 "Expected DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTARROWS got %lx\n", r);
192 DestroyWindow (hwEdit);
194 trace("EDIT: Multiline line\n");
195 hwEdit = create_editcontrol(ES_MULTILINE | WS_VSCROLL | ES_AUTOVSCROLL, 0);
196 r = get_edit_style(hwEdit);
197 ok(r == (ES_AUTOHSCROLL | ES_AUTOVSCROLL | ES_MULTILINE), "Wrong style expected 0xc4 got: 0x%lx\n", r);
198 for (i=0;i<65535;i++)
200 msMessage.wParam = i;
201 r = SendMessage(hwEdit, WM_GETDLGCODE, 0, (LPARAM) &msMessage);
202 ok(r == (DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTALLKEYS | DLGC_WANTARROWS),
203 "Expected DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTALLKEYS | DLGC_WANTARROWS got %lx\n", r);
205 DestroyWindow (hwEdit);
207 trace("EDIT: Multi line want returns\n");
208 hwEdit = create_editcontrol(ES_MULTILINE | WS_VSCROLL | ES_AUTOVSCROLL | ES_WANTRETURN, 0);
209 r = get_edit_style(hwEdit);
210 ok(r == (ES_WANTRETURN | ES_AUTOHSCROLL | ES_AUTOVSCROLL | ES_MULTILINE), "Wrong style expected 0x10c4 got: 0x%lx\n", r);
211 for (i=0;i<65535;i++)
213 msMessage.wParam = i;
214 r = SendMessage(hwEdit, WM_GETDLGCODE, 0, (LPARAM) &msMessage);
215 ok(r == (DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTALLKEYS | DLGC_WANTARROWS),
216 "Expected DLGC_WANTCHARS | DLGC_HASSETSEL | DLGC_WANTALLKEYS | DLGC_WANTARROWS got %lx\n", r);
218 DestroyWindow (hwEdit);
221 /* WM_SETTEXT is implemented by selecting all text, and then replacing the
222 * selection. This test checks that the first 'select all' doesn't generate
223 * an UPDATE message which can escape and (via a handler) change the
224 * selection, which would cause WM_SETTEXT to break. This old bug
225 * was fixed 18-Mar-2005; we check here to ensure it doesn't regress.
227 static void test_edit_control_2(void)
230 char szLocalString[MAXLEN];
232 /* Create main and edit windows. */
233 hwndMain = CreateWindow(szEditTest2Class, "ET2", WS_OVERLAPPEDWINDOW,
234 0, 0, 200, 200, NULL, NULL, hinst, NULL);
236 if (winetest_interactive)
237 ShowWindow (hwndMain, SW_SHOW);
239 hwndET2 = CreateWindow("EDIT", NULL,
240 WS_CHILD|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL,
241 0, 0, 150, 50, /* important this not be 0 size. */
242 hwndMain, (HMENU) ID_EDITTEST2, hinst, NULL);
244 if (winetest_interactive)
245 ShowWindow (hwndET2, SW_SHOW);
247 trace("EDIT: SETTEXT atomicity\n");
248 /* Send messages to "type" in the word 'foo'. */
249 SendMessage(hwndET2, WM_CHAR, 'f', 1);
250 SendMessage(hwndET2, WM_CHAR, 'o', 1);
251 SendMessage(hwndET2, WM_CHAR, 'o', 1);
252 /* 'foo' should have been changed to 'bar' by the UPDATE handler. */
253 GetWindowText(hwndET2, szLocalString, MAXLEN);
254 ok(lstrcmp(szLocalString, "bar")==0,
255 "Wrong contents of edit: %s\n", szLocalString);
258 DestroyWindow (hwndET2);
259 DestroyWindow (hwndMain);
262 static void ET2_check_change(void) {
263 char szLocalString[MAXLEN];
264 /* This EN_UPDATE handler changes any 'foo' to 'bar'. */
265 GetWindowText(hwndET2, szLocalString, MAXLEN);
266 if (lstrcmp(szLocalString, "foo")==0) {
267 lstrcpy(szLocalString, "bar");
268 SendMessage(hwndET2, WM_SETTEXT, 0, (LPARAM) szLocalString);
270 /* always leave the cursor at the end. */
271 SendMessage(hwndET2, EM_SETSEL, MAXLEN - 1, MAXLEN - 1);
273 static void ET2_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)
275 if (id==ID_EDITTEST2 && codeNotify == EN_UPDATE)
278 static LRESULT CALLBACK ET2_WndProc(HWND hwnd, UINT iMsg, WPARAM wParam, LPARAM lParam)
281 HANDLE_MSG(hwnd, WM_COMMAND, ET2_OnCommand);
283 return DefWindowProc(hwnd, iMsg, wParam, lParam);
286 static void zero_notify(void)
288 notifications.en_change = 0;
289 notifications.en_maxtext = 0;
290 notifications.en_update = 0;
293 #define test_notify(enchange, enmaxtext, enupdate) \
294 ok(notifications.en_change == enchange, "expected %d EN_CHANGE notifications, " \
295 "got %d\n", enchange, notifications.en_change); \
296 ok(notifications.en_maxtext == enmaxtext, "expected %d EN_MAXTEXT notifications, " \
297 "got %d\n", enmaxtext, notifications.en_maxtext); \
298 ok(notifications.en_update == enupdate, "expected %d EN_UPDATE notifications, " \
299 "got %d\n", enupdate, notifications.en_update)
302 static LRESULT CALLBACK edit3_wnd_procA(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
306 switch (HIWORD(wParam)) {
308 notifications.en_maxtext++;
311 notifications.en_update++;
314 notifications.en_change++;
319 return DefWindowProcA(hWnd, msg, wParam, lParam);
322 /* Test behaviour of WM_SETTEXT, WM_REPLACESEL and notificatisons sent in response
325 static void test_edit_control_3(void)
330 static const char *str = "this is a long string.";
331 static const char *str2 = "this is a long string.\r\nthis is a long string.\r\nthis is a long string.\r\nthis is a long string.";
333 trace("EDIT: Test notifications\n");
335 hParent = CreateWindowExA(0,
339 CW_USEDEFAULT, CW_USEDEFAULT, 10, 10,
340 NULL, NULL, NULL, NULL);
343 trace("EDIT: Single line, no ES_AUTOHSCROLL\n");
344 hWnd = CreateWindowExA(0,
349 hParent, NULL, NULL, NULL);
353 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
354 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
355 ok(lstrlenA(str) > len, "text should have been truncated\n");
356 test_notify(1, 1, 1);
358 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
360 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)"a");
361 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
362 ok(1 == len, "wrong text length, expected 1, got %d\n", len);
363 test_notify(1, 0, 1);
366 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
367 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
368 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
369 test_notify(1, 0, 1);
371 SendMessageA(hWnd, EM_SETLIMITTEXT, 5, 0);
373 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
375 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
376 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
377 ok(5 == len, "text should have been truncated to limit, expected 5, got %d\n", len);
378 test_notify(1, 1, 1);
381 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
382 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
383 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
384 test_notify(1, 0, 1);
388 trace("EDIT: Single line, ES_AUTOHSCROLL\n");
389 hWnd = CreateWindowExA(0,
394 hParent, NULL, NULL, NULL);
398 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
399 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
400 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
401 test_notify(1, 0, 1);
404 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
405 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
406 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
407 test_notify(1, 0, 1);
409 SendMessageA(hWnd, EM_SETLIMITTEXT, 5, 0);
411 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
413 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
414 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
415 ok(5 == len, "text should have been truncated to limit, expected 5, got %d\n", len);
416 test_notify(1, 1, 1);
419 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
420 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
421 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
422 test_notify(1, 0, 1);
426 trace("EDIT: Multline, no ES_AUTOHSCROLL, no ES_AUTOVSCROLL\n");
427 hWnd = CreateWindowExA(0,
432 hParent, NULL, NULL, NULL);
436 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
437 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
438 ok(0 == len, "text should have been truncated, expected 0, got %d\n", len);
439 test_notify(1, 1, 1);
441 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
443 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)"a");
444 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
445 ok(1 == SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0), "wrong text length, expected 1, got %d\n", len);
446 test_notify(1, 0, 1);
449 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
450 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
451 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
452 test_notify(0, 0, 0);
454 SendMessageA(hWnd, EM_SETLIMITTEXT, 5, 0);
456 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
458 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str);
459 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
460 ok(5 == len, "text should have been truncated to limit, expected 5, got %d\n", len);
461 test_notify(1, 1, 1);
464 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str);
465 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
466 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
467 test_notify(0, 0, 0);
471 trace("EDIT: Multline, ES_AUTOHSCROLL, no ES_AUTOVSCROLL\n");
472 hWnd = CreateWindowExA(0,
475 ES_MULTILINE | ES_AUTOHSCROLL,
477 hParent, NULL, NULL, NULL);
481 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str2);
482 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
483 ok(0 == len, "text should have been truncated, expected 0, got %d\n", len);
484 test_notify(1, 1, 1);
486 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
488 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)"a");
489 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
490 ok(1 == SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0), "wrong text length, expected 1, got %d\n", len);
491 test_notify(1, 0, 1);
494 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str2);
495 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
496 ok(lstrlenA(str2) == len, "text shouldn't have been truncated\n");
497 test_notify(0, 0, 0);
499 SendMessageA(hWnd, EM_SETLIMITTEXT, 5, 0);
501 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
503 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str2);
504 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
505 ok(5 == len, "text should have been truncated to limit, expected 5, got %d\n", len);
506 test_notify(1, 1, 1);
509 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str2);
510 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
511 ok(lstrlenA(str2) == len, "text shouldn't have been truncated\n");
512 test_notify(0, 0, 0);
516 trace("EDIT: Multline, ES_AUTOHSCROLL and ES_AUTOVSCROLL\n");
517 hWnd = CreateWindowExA(0,
520 ES_MULTILINE | ES_AUTOHSCROLL | ES_AUTOVSCROLL,
522 hParent, NULL, NULL, NULL);
526 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str2);
527 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
528 ok(lstrlenA(str2) == len, "text shouldn't have been truncated\n");
529 test_notify(1, 0, 1);
532 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str2);
533 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
534 ok(lstrlenA(str2) == len, "text shouldn't have been truncated\n");
535 test_notify(0, 0, 0);
537 SendMessageA(hWnd, EM_SETLIMITTEXT, 5, 0);
539 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)"");
541 SendMessageA(hWnd, EM_REPLACESEL, 0, (LPARAM)str2);
542 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
543 ok(5 == len, "text should have been truncated to limit, expected 5, got %d\n", len);
544 test_notify(1, 1, 1);
547 SendMessageA(hWnd, WM_SETTEXT, 0, (LPARAM)str2);
548 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
549 ok(lstrlenA(str2) == len, "text shouldn't have been truncated\n");
550 test_notify(0, 0, 0);
555 /* Test EM_CHARFROMPOS and EM_POSFROMCHAR
557 static void test_edit_control_4(void)
564 trace("EDIT: Test EM_CHARFROMPOS and EM_POSFROMCHAR\n");
565 hwEdit = create_editcontrol(0, 0);
566 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
567 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
568 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
569 mid = lo + (hi - lo) / 2;
571 for (i = lo; i < mid; i++) {
572 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
573 ok(0 == ret, "expected 0 got %d\n", ret);
575 for (i = mid; i <= hi; i++) {
576 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
577 ok(1 == ret, "expected 1 got %d\n", ret);
579 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
580 ok(-1 == ret, "expected -1 got %d\n", ret);
581 DestroyWindow(hwEdit);
583 hwEdit = create_editcontrol(ES_RIGHT, 0);
584 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
585 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
586 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
587 mid = lo + (hi - lo) / 2;
589 for (i = lo; i < mid; i++) {
590 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
591 ok(0 == ret, "expected 0 got %d\n", ret);
593 for (i = mid; i <= hi; i++) {
594 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
595 ok(1 == ret, "expected 1 got %d\n", ret);
597 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
598 ok(-1 == ret, "expected -1 got %d\n", ret);
599 DestroyWindow(hwEdit);
601 hwEdit = create_editcontrol(ES_CENTER, 0);
602 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
603 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
604 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
605 mid = lo + (hi - lo) / 2;
607 for (i = lo; i < mid; i++) {
608 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
609 ok(0 == ret, "expected 0 got %d\n", ret);
611 for (i = mid; i <= hi; i++) {
612 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
613 ok(1 == ret, "expected 1 got %d\n", ret);
615 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
616 ok(-1 == ret, "expected -1 got %d\n", ret);
617 DestroyWindow(hwEdit);
619 hwEdit = create_editcontrol(ES_MULTILINE, 0);
620 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
621 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
622 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
623 mid = lo + (hi - lo) / 2 +1;
625 for (i = lo; i < mid; i++) {
626 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
627 ok(0 == ret, "expected 0 got %d\n", ret);
629 for (i = mid; i <= hi; i++) {
630 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
631 ok(1 == ret, "expected 1 got %d\n", ret);
633 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
634 ok(-1 == ret, "expected -1 got %d\n", ret);
635 DestroyWindow(hwEdit);
637 hwEdit = create_editcontrol(ES_MULTILINE | ES_RIGHT, 0);
638 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
639 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
640 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
641 mid = lo + (hi - lo) / 2 +1;
643 for (i = lo; i < mid; i++) {
644 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
645 ok(0 == ret, "expected 0 got %d\n", ret);
647 for (i = mid; i <= hi; i++) {
648 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
649 ok(1 == ret, "expected 1 got %d\n", ret);
651 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
652 ok(-1 == ret, "expected -1 got %d\n", ret);
653 DestroyWindow(hwEdit);
655 hwEdit = create_editcontrol(ES_MULTILINE | ES_CENTER, 0);
656 SendMessage(hwEdit, WM_SETTEXT, 0, (LPARAM) "aa");
657 lo = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 0, 0));
658 hi = LOWORD(SendMessage(hwEdit, EM_POSFROMCHAR, 1, 0));
659 mid = lo + (hi - lo) / 2 +1;
661 for (i = lo; i < mid; i++) {
662 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
663 ok(0 == ret, "expected 0 got %d\n", ret);
665 for (i = mid; i <= hi; i++) {
666 ret = LOWORD(SendMessage(hwEdit, EM_CHARFROMPOS, 0, (LPARAM) i));
667 ok(1 == ret, "expected 1 got %d\n", ret);
669 ret = SendMessage(hwEdit, EM_POSFROMCHAR, 2, 0);
670 ok(-1 == ret, "expected -1 got %d\n", ret);
671 DestroyWindow(hwEdit);
674 /* Test if creating edit control without ES_AUTOHSCROLL and ES_AUTOVSCROLL
675 * truncates text that doesn't fit.
677 static void test_edit_control_5(void)
679 static const char *str = "test\r\ntest";
683 hWnd = CreateWindowEx(0,
688 NULL, NULL, NULL, NULL);
691 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
692 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
695 hWnd = CreateWindowEx(0,
700 NULL, NULL, NULL, NULL);
703 len = SendMessageA(hWnd, WM_GETTEXTLENGTH, 0, 0);
704 ok(lstrlenA(str) == len, "text shouldn't have been truncated\n");
708 static void test_margins(void)
711 RECT old_rect, new_rect;
712 INT old_left_margin, old_right_margin;
713 DWORD old_margins, new_margins;
715 hwEdit = create_editcontrol(WS_BORDER, 0);
717 old_margins = SendMessage(hwEdit, EM_GETMARGINS, 0, 0);
718 old_left_margin = LOWORD(old_margins);
719 old_right_margin = HIWORD(old_margins);
721 /* Check if setting the margins works */
723 SendMessage(hwEdit, EM_SETMARGINS, EC_LEFTMARGIN, MAKELONG(10, 0));
724 new_margins = SendMessage(hwEdit, EM_GETMARGINS, 0, 0);
725 ok(LOWORD(new_margins) == 10, "Wrong left margin: %d\n", LOWORD(new_margins));
726 ok(HIWORD(new_margins) == old_right_margin, "Wrong right margin: %d\n", HIWORD(new_margins));
728 SendMessage(hwEdit, EM_SETMARGINS, EC_RIGHTMARGIN, MAKELONG(0, 10));
729 new_margins = SendMessage(hwEdit, EM_GETMARGINS, 0, 0);
730 ok(LOWORD(new_margins) == 10, "Wrong left margin: %d\n", LOWORD(new_margins));
731 ok(HIWORD(new_margins) == 10, "Wrong right margin: %d\n", HIWORD(new_margins));
734 /* The size of the rectangle must decrease if we increase the margin */
736 SendMessage(hwEdit, EM_SETMARGINS, EC_LEFTMARGIN | EC_RIGHTMARGIN, MAKELONG(5, 5));
737 SendMessage(hwEdit, EM_GETRECT, 0, (LPARAM)&old_rect);
738 SendMessage(hwEdit, EM_SETMARGINS, EC_LEFTMARGIN | EC_RIGHTMARGIN, MAKELONG(15, 20));
739 SendMessage(hwEdit, EM_GETRECT, 0, (LPARAM)&new_rect);
740 ok(new_rect.left == old_rect.left + 10, "The left border of the rectangle is wrong\n");
741 ok(new_rect.right == old_rect.right - 15, "The right border of the rectangle is wrong\n");
742 ok(new_rect.top == old_rect.top, "The top border of the rectangle must not change\n");
743 ok(new_rect.bottom == old_rect.bottom, "The bottom border of the rectangle must not change\n");
746 /* If we set the margin to same value as the current margin,
747 the rectangle must not change */
749 SendMessage(hwEdit, EM_SETMARGINS, EC_LEFTMARGIN | EC_RIGHTMARGIN, MAKELONG(10, 10));
753 old_rect.bottom = 99;
754 SendMessage(hwEdit, EM_SETRECT, 0, (LPARAM)&old_rect);
755 SendMessage(hwEdit, EM_GETRECT, 0, (LPARAM)&old_rect);
756 SendMessage(hwEdit, EM_SETMARGINS, EC_LEFTMARGIN | EC_RIGHTMARGIN, MAKELONG(10, 10));
757 SendMessage(hwEdit, EM_GETRECT, 0, (LPARAM)&new_rect);
758 ok(new_rect.left == old_rect.left, "The left border of the rectangle has changed\n");
759 ok(new_rect.right == old_rect.right, "The right border of the rectangle has changed\n");
760 ok(new_rect.top == old_rect.top, "The top border of the rectangle has changed\n");
761 ok(new_rect.bottom == old_rect.bottom, "The bottom border of the rectangle has changed\n");
763 DestroyWindow (hwEdit);
766 #define edit_pos_ok(exp, got, txt) \
767 ok(exp == got, "wrong " #txt " expected %d got %ld\n", exp, got);
769 #define check_pos(hwEdit, set_height, test_top, test_height, test_left) \
773 set_client_height(hwEdit, set_height); \
774 SendMessage(hwEdit, EM_GETRECT, 0, (LPARAM) &format_rect); \
775 left_margin = LOWORD(SendMessage(hwEdit, EM_GETMARGINS, 0, 0)); \
776 edit_pos_ok(test_top, format_rect.top, vertical position); \
777 edit_pos_ok((int)test_height, format_rect.bottom - format_rect.top, height); \
778 edit_pos_ok(test_left, format_rect.left - left_margin, left); \
781 void test_text_position_style(DWORD style)
788 BOOL single_line = !(style & ES_MULTILINE);
790 b = GetSystemMetrics(SM_CYBORDER) + 1;
795 /* Get a stock font for which we can determine the metrics */
796 assert(font = GetStockObject(SYSTEM_FONT));
797 assert(dc = GetDC(NULL));
798 oldFont = SelectObject(dc, font);
799 assert(GetTextMetrics(dc, &metrics));
800 SelectObject(dc, oldFont);
803 /* Windows' edit control has some bugs in multi-line mode:
804 * - Sometimes the format rectangle doesn't get updated
805 * (see workaround in set_client_height())
806 * - If the height of the control is smaller than the height of a text
807 * line, the format rectangle is still as high as a text line
808 * (higher than the client rectangle) and the caret is not shown
811 /* Edit controls that are in a parent window */
813 hwEdit = create_child_editcontrol(style, 0);
814 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
816 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 0);
817 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 0);
818 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 0);
819 check_pos(hwEdit, metrics.tmHeight + 2, 0, metrics.tmHeight , 0);
820 check_pos(hwEdit, metrics.tmHeight + 10, 0, metrics.tmHeight , 0);
821 destroy_child_editcontrol(hwEdit);
823 hwEdit = create_child_editcontrol(style | WS_BORDER, 0);
824 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
826 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, b);
827 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , b);
828 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , b);
829 check_pos(hwEdit, metrics.tmHeight + bm, 0, metrics.tmHeight , b);
830 check_pos(hwEdit, metrics.tmHeight + b2, b, metrics.tmHeight , b);
831 check_pos(hwEdit, metrics.tmHeight + b3, b, metrics.tmHeight , b);
832 destroy_child_editcontrol(hwEdit);
834 hwEdit = create_child_editcontrol(style, WS_EX_CLIENTEDGE);
835 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
837 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 1);
838 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 1);
839 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 1);
840 check_pos(hwEdit, metrics.tmHeight + 2, 1, metrics.tmHeight , 1);
841 check_pos(hwEdit, metrics.tmHeight + 10, 1, metrics.tmHeight , 1);
842 destroy_child_editcontrol(hwEdit);
844 hwEdit = create_child_editcontrol(style | WS_BORDER, WS_EX_CLIENTEDGE);
845 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
847 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 1);
848 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 1);
849 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 1);
850 check_pos(hwEdit, metrics.tmHeight + 2, 1, metrics.tmHeight , 1);
851 check_pos(hwEdit, metrics.tmHeight + 10, 1, metrics.tmHeight , 1);
852 destroy_child_editcontrol(hwEdit);
855 /* Edit controls that are popup windows */
857 hwEdit = create_editcontrol(style | WS_POPUP, 0);
858 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
860 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 0);
861 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 0);
862 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 0);
863 check_pos(hwEdit, metrics.tmHeight + 2, 0, metrics.tmHeight , 0);
864 check_pos(hwEdit, metrics.tmHeight + 10, 0, metrics.tmHeight , 0);
865 DestroyWindow(hwEdit);
867 hwEdit = create_editcontrol(style | WS_POPUP | WS_BORDER, 0);
868 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
870 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, b);
871 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , b);
872 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , b);
873 check_pos(hwEdit, metrics.tmHeight + bm, 0, metrics.tmHeight , b);
874 check_pos(hwEdit, metrics.tmHeight + b2, b, metrics.tmHeight , b);
875 check_pos(hwEdit, metrics.tmHeight + b3, b, metrics.tmHeight , b);
876 DestroyWindow(hwEdit);
878 hwEdit = create_editcontrol(style | WS_POPUP, WS_EX_CLIENTEDGE);
879 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
881 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 1);
882 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 1);
883 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 1);
884 check_pos(hwEdit, metrics.tmHeight + 2, 1, metrics.tmHeight , 1);
885 check_pos(hwEdit, metrics.tmHeight + 10, 1, metrics.tmHeight , 1);
886 DestroyWindow(hwEdit);
888 hwEdit = create_editcontrol(style | WS_POPUP | WS_BORDER, WS_EX_CLIENTEDGE);
889 SendMessage(hwEdit, WM_SETFONT, (WPARAM) font, (LPARAM) FALSE);
891 check_pos(hwEdit, metrics.tmHeight - 1, 0, metrics.tmHeight - 1, 1);
892 check_pos(hwEdit, metrics.tmHeight , 0, metrics.tmHeight , 1);
893 check_pos(hwEdit, metrics.tmHeight + 1, 0, metrics.tmHeight , 1);
894 check_pos(hwEdit, metrics.tmHeight + 2, 1, metrics.tmHeight , 1);
895 check_pos(hwEdit, metrics.tmHeight + 10, 1, metrics.tmHeight , 1);
896 DestroyWindow(hwEdit);
899 void test_text_position(void)
901 trace("EDIT: Text position (Single line)\n");
902 test_text_position_style(0);
903 trace("EDIT: Text position (Multi line)\n");
904 test_text_position_style(ES_MULTILINE);
907 static BOOL RegisterWindowClasses (void)
911 WNDCLASSA text_position;
914 test2.lpfnWndProc = ET2_WndProc;
915 test2.cbClsExtra = 0;
916 test2.cbWndExtra = 0;
917 test2.hInstance = hinst;
919 test2.hCursor = LoadCursorA (NULL, IDC_ARROW);
920 test2.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
921 test2.lpszMenuName = NULL;
922 test2.lpszClassName = szEditTest2Class;
923 if (!RegisterClassA(&test2)) return FALSE;
926 test3.lpfnWndProc = edit3_wnd_procA;
927 test3.cbClsExtra = 0;
928 test3.cbWndExtra = 0;
929 test3.hInstance = hinst;
931 test3.hCursor = LoadCursorA(0, (LPSTR)IDC_ARROW);
932 test3.hbrBackground = GetStockObject(WHITE_BRUSH);
933 test3.lpszMenuName = NULL;
934 test3.lpszClassName = szEditTest3Class;
935 if (!RegisterClassA(&test3)) return FALSE;
937 text_position.style = CS_HREDRAW | CS_VREDRAW;
938 text_position.cbClsExtra = 0;
939 text_position.cbWndExtra = 0;
940 text_position.hInstance = hinst;
941 text_position.hIcon = NULL;
942 text_position.hCursor = LoadCursorA(NULL, MAKEINTRESOURCEA(IDC_ARROW));
943 text_position.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1);
944 text_position.lpszMenuName = NULL;
945 text_position.lpszClassName = szEditTextPositionClass;
946 text_position.lpfnWndProc = DefWindowProc;
947 if (!RegisterClassA(&text_position)) return FALSE;
952 static void UnregisterWindowClasses (void)
954 UnregisterClassA(szEditTest2Class, hinst);
955 UnregisterClassA(szEditTest3Class, hinst);
956 UnregisterClassA(szEditTextPositionClass, hinst);
961 hinst = GetModuleHandleA(NULL);
962 assert(RegisterWindowClasses());
964 test_edit_control_1();
965 test_edit_control_2();
966 test_edit_control_3();
967 test_edit_control_4();
968 test_edit_control_5();
970 test_text_position();
972 UnregisterWindowClasses();