| /* |
| * Unit test suite for rich edit control |
| * |
| * Copyright 2006 Google (Thomas Kho) |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include <wine/test.h> |
| #include <windows.h> |
| #include <richedit.h> |
| #include <time.h> |
| |
| static HMODULE hmoduleRichEdit; |
| |
| static HWND new_window(LPCTSTR lpClassName, DWORD dwStyle, HWND parent) { |
| HWND hwnd; |
| hwnd = CreateWindow(lpClassName, NULL, dwStyle|WS_POPUP|WS_HSCROLL|WS_VSCROLL |
| |WS_VISIBLE, 0, 0, 200, 60, parent, NULL, |
| hmoduleRichEdit, NULL); |
| ok(hwnd != NULL, "class: %s, error: %d\n", lpClassName, (int) GetLastError()); |
| return hwnd; |
| } |
| |
| static HWND new_richedit(HWND parent) { |
| return new_window(RICHEDIT_CLASS, ES_MULTILINE, parent); |
| } |
| |
| static const char haystack[] = "WINEWine wineWine wine WineWine"; |
| /* ^0 ^10 ^20 ^30 */ |
| |
| struct find_s { |
| int start; |
| int end; |
| char *needle; |
| int flags; |
| int expected_loc; |
| int _todo_wine; |
| }; |
| |
| struct find_s find_tests[] = { |
| /* Find in empty text */ |
| {0, -1, "foo", FR_DOWN, -1, 0}, |
| {0, -1, "foo", 0, -1, 0}, |
| {0, -1, "", FR_DOWN, -1, 0}, |
| {20, 5, "foo", FR_DOWN, -1, 0}, |
| {5, 20, "foo", FR_DOWN, -1, 0} |
| }; |
| |
| struct find_s find_tests2[] = { |
| /* No-result find */ |
| {0, -1, "foo", FR_DOWN | FR_MATCHCASE, -1, 0}, |
| {5, 20, "WINE", FR_DOWN | FR_MATCHCASE, -1, 0}, |
| |
| /* Subsequent finds */ |
| {0, -1, "Wine", FR_DOWN | FR_MATCHCASE, 4, 0}, |
| {5, 31, "Wine", FR_DOWN | FR_MATCHCASE, 13, 0}, |
| {14, 31, "Wine", FR_DOWN | FR_MATCHCASE, 23, 0}, |
| {24, 31, "Wine", FR_DOWN | FR_MATCHCASE, 27, 0}, |
| |
| /* Find backwards */ |
| {19, 20, "Wine", FR_MATCHCASE, 13, 1}, |
| {10, 20, "Wine", FR_MATCHCASE, 4, 1}, |
| {20, 10, "Wine", FR_MATCHCASE, 13, 0}, |
| |
| /* Case-insensitive */ |
| {1, 31, "wInE", FR_DOWN, 4, 1}, |
| {1, 31, "Wine", FR_DOWN, 4, 0}, |
| |
| /* High-to-low ranges */ |
| {20, 5, "Wine", FR_DOWN, -1, 1}, |
| {2, 1, "Wine", FR_DOWN, -1, 0}, |
| {30, 29, "Wine", FR_DOWN, -1, 0}, |
| {20, 5, "Wine", 0, 13, 0}, |
| |
| /* Find nothing */ |
| {5, 10, "", FR_DOWN, -1, 0}, |
| {10, 5, "", FR_DOWN, -1, 0}, |
| {0, -1, "", FR_DOWN, -1, 0}, |
| {10, 5, "", 0, -1, 0}, |
| |
| /* Whole-word search */ |
| {0, -1, "wine", FR_DOWN | FR_WHOLEWORD, 18, 1}, |
| {0, -1, "win", FR_DOWN | FR_WHOLEWORD, -1, 1}, |
| |
| /* Bad ranges */ |
| {-20, 20, "Wine", FR_DOWN, -1, 0}, |
| {-20, 20, "Wine", FR_DOWN, -1, 0}, |
| {-15, -20, "Wine", FR_DOWN, -1, 0}, |
| {1<<12, 1<<13, "Wine", FR_DOWN, -1, 0}, |
| |
| /* Check the case noted in bug 4479 where matches at end aren't recognized */ |
| {23, 31, "Wine", FR_DOWN | FR_MATCHCASE, 23, 0}, |
| {27, 31, "Wine", FR_DOWN | FR_MATCHCASE, 27, 0}, |
| {27, 32, "Wine", FR_DOWN | FR_MATCHCASE, 27, 0}, |
| {13, 31, "WineWine", FR_DOWN | FR_MATCHCASE, 23, 0}, |
| {13, 32, "WineWine", FR_DOWN | FR_MATCHCASE, 23, 0}, |
| |
| /* The backwards case of bug 4479; bounds look right |
| * Fails because backward find is wrong */ |
| {19, 20, "WINE", FR_MATCHCASE, 0, 1}, |
| {0, 20, "WINE", FR_MATCHCASE, -1, 1} |
| }; |
| |
| static void check_EM_FINDTEXT(HWND hwnd, char *name, struct find_s *f, int id) { |
| int findloc; |
| FINDTEXT ft; |
| memset(&ft, 0, sizeof(ft)); |
| ft.chrg.cpMin = f->start; |
| ft.chrg.cpMax = f->end; |
| ft.lpstrText = f->needle; |
| findloc = SendMessage(hwnd, EM_FINDTEXT, f->flags, (LPARAM) &ft); |
| ok(findloc == f->expected_loc, |
| "EM_FINDTEXT(%s,%d) '%s' in range(%d,%d), flags %08x, got start at %d\n", |
| name, id, f->needle, f->start, f->end, f->flags, findloc); |
| } |
| |
| static void check_EM_FINDTEXTEX(HWND hwnd, char *name, struct find_s *f, |
| int id) { |
| int findloc; |
| FINDTEXTEX ft; |
| memset(&ft, 0, sizeof(ft)); |
| ft.chrg.cpMin = f->start; |
| ft.chrg.cpMax = f->end; |
| ft.lpstrText = f->needle; |
| findloc = SendMessage(hwnd, EM_FINDTEXTEX, f->flags, (LPARAM) &ft); |
| ok(findloc == f->expected_loc, |
| "EM_FINDTEXTEX(%s,%d) '%s' in range(%d,%d), flags %08x, start at %d\n", |
| name, id, f->needle, f->start, f->end, f->flags, findloc); |
| ok(ft.chrgText.cpMin == f->expected_loc, |
| "EM_FINDTEXTEX(%s,%d) '%s' in range(%d,%d), flags %08x, start at %ld\n", |
| name, id, f->needle, f->start, f->end, f->flags, ft.chrgText.cpMin); |
| ok(ft.chrgText.cpMax == ((f->expected_loc == -1) ? -1 |
| : f->expected_loc + strlen(f->needle)), |
| "EM_FINDTEXTEX(%s,%d) '%s' in range(%d,%d), flags %08x, end at %ld\n", |
| name, id, f->needle, f->start, f->end, f->flags, ft.chrgText.cpMax); |
| } |
| |
| static void run_tests_EM_FINDTEXT(HWND hwnd, char *name, struct find_s *find, |
| int num_tests) |
| { |
| int i; |
| |
| for (i = 0; i < num_tests; i++) { |
| if (find[i]._todo_wine) { |
| todo_wine { |
| check_EM_FINDTEXT(hwnd, name, &find[i], i); |
| check_EM_FINDTEXTEX(hwnd, name, &find[i], i); |
| } |
| } else { |
| check_EM_FINDTEXT(hwnd, name, &find[i], i); |
| check_EM_FINDTEXTEX(hwnd, name, &find[i], i); |
| } |
| } |
| } |
| |
| static void test_EM_FINDTEXT(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| |
| /* Empty rich edit control */ |
| run_tests_EM_FINDTEXT(hwndRichEdit, "1", find_tests, |
| sizeof(find_tests)/sizeof(struct find_s)); |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) haystack); |
| |
| /* Haystack text */ |
| run_tests_EM_FINDTEXT(hwndRichEdit, "2", find_tests2, |
| sizeof(find_tests2)/sizeof(struct find_s)); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static int get_scroll_pos_y(HWND hwnd) |
| { |
| POINT p = {-1, -1}; |
| SendMessage(hwnd, EM_GETSCROLLPOS, 0, (LPARAM) &p); |
| ok(p.x != -1 && p.y != -1, "p.x:%ld p.y:%ld\n", p.x, p.y); |
| return p.y; |
| } |
| |
| static void move_cursor(HWND hwnd, long charindex) |
| { |
| CHARRANGE cr; |
| cr.cpMax = charindex; |
| cr.cpMin = charindex; |
| SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM) &cr); |
| } |
| |
| static void line_scroll(HWND hwnd, int amount) |
| { |
| SendMessage(hwnd, EM_LINESCROLL, 0, amount); |
| } |
| |
| static void test_EM_SCROLLCARET(void) |
| { |
| int prevY, curY; |
| HWND hwndRichEdit = new_richedit(NULL); |
| const char text[] = "aa\n" |
| "this is a long line of text that should be longer than the " |
| "control's width\n" |
| "cc\n" |
| "dd\n" |
| "ee\n" |
| "ff\n" |
| "gg\n" |
| "hh\n"; |
| |
| /* Can't verify this */ |
| SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0); |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| |
| /* Caret above visible window */ |
| line_scroll(hwndRichEdit, 3); |
| prevY = get_scroll_pos_y(hwndRichEdit); |
| SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0); |
| curY = get_scroll_pos_y(hwndRichEdit); |
| ok(prevY != curY, "%d == %d\n", prevY, curY); |
| |
| /* Caret below visible window */ |
| move_cursor(hwndRichEdit, sizeof(text) - 1); |
| line_scroll(hwndRichEdit, -3); |
| prevY = get_scroll_pos_y(hwndRichEdit); |
| SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0); |
| curY = get_scroll_pos_y(hwndRichEdit); |
| ok(prevY != curY, "%d == %d\n", prevY, curY); |
| |
| /* Caret in visible window */ |
| move_cursor(hwndRichEdit, sizeof(text) - 2); |
| prevY = get_scroll_pos_y(hwndRichEdit); |
| SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0); |
| curY = get_scroll_pos_y(hwndRichEdit); |
| ok(prevY == curY, "%d != %d\n", prevY, curY); |
| |
| /* Caret still in visible window */ |
| line_scroll(hwndRichEdit, -1); |
| prevY = get_scroll_pos_y(hwndRichEdit); |
| SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0); |
| curY = get_scroll_pos_y(hwndRichEdit); |
| ok(prevY == curY, "%d != %d\n", prevY, curY); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| START_TEST( editor ) |
| { |
| MSG msg; |
| time_t end; |
| |
| /* Must explicitly LoadLibrary(). The test has no references to functions in |
| * RICHED20.DLL, so the linker doesn't actually link to it. */ |
| hmoduleRichEdit = LoadLibrary("RICHED20.DLL"); |
| ok(hmoduleRichEdit != NULL, "error: %d\n", (int) GetLastError()); |
| |
| test_EM_FINDTEXT(); |
| test_EM_SCROLLCARET(); |
| |
| /* Set the environment variable WINETEST_RICHED20 to keep windows |
| * responsive and open for 30 seconds. This is useful for debugging. |
| * |
| * The message pump uses PeekMessage() to empty the queue and then sleeps for |
| * 50ms before retrying the queue. */ |
| end = time(NULL) + 30; |
| if (getenv( "WINETEST_RICHED20" )) { |
| while (time(NULL) < end) { |
| if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { |
| TranslateMessage(&msg); |
| DispatchMessage(&msg); |
| } else { |
| Sleep(50); |
| } |
| } |
| } |
| |
| ok(FreeLibrary(hmoduleRichEdit) != 0, "error: %d\n", (int) GetLastError()); |
| } |