| /* |
| * Unit test suite for rich edit control |
| * |
| * Copyright 2006 Google (Thomas Kho) |
| * Copyright 2007 Matt Finnicum |
| * |
| * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <wine/test.h> |
| #include <windows.h> |
| #include <richedit.h> |
| #include <time.h> |
| #include <stdio.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; |
| const 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, 0}, |
| {10, 20, "Wine", FR_MATCHCASE, 4, 0}, |
| {20, 10, "Wine", FR_MATCHCASE, 13, 0}, |
| |
| /* Case-insensitive */ |
| {1, 31, "wInE", FR_DOWN, 4, 0}, |
| {1, 31, "Wine", FR_DOWN, 4, 0}, |
| |
| /* High-to-low ranges */ |
| {20, 5, "Wine", FR_DOWN, -1, 0}, |
| {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, 0}, |
| {0, -1, "win", FR_DOWN | FR_WHOLEWORD, -1, 0}, |
| {13, -1, "wine", FR_DOWN | FR_WHOLEWORD, 18, 0}, |
| {0, -1, "winewine", FR_DOWN | FR_WHOLEWORD, 0, 0}, |
| {10, -1, "winewine", FR_DOWN | FR_WHOLEWORD, 23, 0}, |
| {11, -1, "winewine", FR_WHOLEWORD, 0, 0}, |
| {31, -1, "winewine", FR_WHOLEWORD, 23, 0}, |
| |
| /* Bad ranges */ |
| {5, 200, "XXX", FR_DOWN, -1, 0}, |
| {-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, 0}, |
| {0, 20, "WINE", FR_MATCHCASE, -1, 0} |
| }; |
| |
| static void check_EM_FINDTEXT(HWND hwnd, const 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, const 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 %d\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 %d\n", |
| name, id, f->needle, f->start, f->end, f->flags, ft.chrgText.cpMax); |
| } |
| |
| static void run_tests_EM_FINDTEXT(HWND hwnd, const 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 const struct getline_s { |
| int line; |
| size_t buffer_len; |
| const char *text; |
| } gl[] = { |
| {0, 10, "foo bar\r"}, |
| {1, 10, "\r"}, |
| {2, 10, "bar\r"}, |
| {3, 10, "\r"}, |
| |
| /* Buffer smaller than line length */ |
| {0, 2, "foo bar\r"}, |
| {0, 1, "foo bar\r"}, |
| {0, 0, "foo bar\r"} |
| }; |
| |
| static void test_EM_GETLINE(void) |
| { |
| int i; |
| HWND hwndRichEdit = new_richedit(NULL); |
| static const int nBuf = 1024; |
| char dest[1024], origdest[1024]; |
| const char text[] = "foo bar\n" |
| "\n" |
| "bar\n"; |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| |
| memset(origdest, 0xBB, nBuf); |
| for (i = 0; i < sizeof(gl)/sizeof(struct getline_s); i++) |
| { |
| int nCopied; |
| int expected_nCopied = min(gl[i].buffer_len, strlen(gl[i].text)); |
| int expected_bytes_written = min(gl[i].buffer_len, strlen(gl[i].text) + 1); |
| memset(dest, 0xBB, nBuf); |
| *(WORD *) dest = gl[i].buffer_len; |
| |
| /* EM_GETLINE appends a "\r\0" to the end of the line |
| * nCopied counts up to and including the '\r' */ |
| nCopied = SendMessage(hwndRichEdit, EM_GETLINE, gl[i].line, (LPARAM) dest); |
| ok(nCopied == expected_nCopied, "%d: %d!=%d\n", i, nCopied, |
| expected_nCopied); |
| /* two special cases since a parameter is passed via dest */ |
| if (gl[i].buffer_len == 0) |
| ok(!dest[0] && !dest[1] && !strncmp(dest+2, origdest+2, nBuf-2), |
| "buffer_len=0\n"); |
| else if (gl[i].buffer_len == 1) |
| ok(dest[0] == gl[i].text[0] && !dest[1] && |
| !strncmp(dest+2, origdest+2, nBuf-2), "buffer_len=1\n"); |
| else |
| { |
| ok(!strncmp(dest, gl[i].text, expected_bytes_written), |
| "%d: expected_bytes_written=%d\n", i, expected_bytes_written); |
| ok(!strncmp(dest + expected_bytes_written, origdest |
| + expected_bytes_written, nBuf - expected_bytes_written), |
| "%d: expected_bytes_written=%d\n", i, expected_bytes_written); |
| } |
| } |
| |
| 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:%d p.y:%d\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); |
| } |
| |
| static void test_EM_SETTEXTMODE(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| CHARFORMAT2 cf2, cf2test; |
| CHARRANGE cr; |
| int rc = 0; |
| |
| /*Test that EM_SETTEXTMODE fails if text exists within the control*/ |
| /*Insert text into the control*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Attempt to change the control to plain text mode*/ |
| rc = SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_PLAINTEXT, 0); |
| ok(rc != 0, "EM_SETTEXTMODE: changed text mode in control containing text - returned: %d\n", rc); |
| |
| /*Test that EM_SETTEXTMODE does not allow rich edit text to be pasted. |
| If rich text is pasted, it should have the same formatting as the rest |
| of the text in the control*/ |
| |
| /*Italicize the text |
| *NOTE: If the default text was already italicized, the test will simply |
| reverse; in other words, it will copy a regular "wine" into a plain |
| text window that uses an italicized format*/ |
| cf2.cbSize = sizeof(CHARFORMAT2); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT, |
| (LPARAM) &cf2); |
| |
| cf2.dwMask = CFM_ITALIC | cf2.dwMask; |
| cf2.dwEffects = CFE_ITALIC ^ cf2.dwEffects; |
| |
| /*EM_SETCHARFORMAT is not yet fully implemented for all WPARAMs in wine; |
| however, SCF_ALL has been implemented*/ |
| SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Select the string "wine"*/ |
| cr.cpMin = 0; |
| cr.cpMax = 4; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /*Copy the italicized "wine" to the clipboard*/ |
| SendMessage(hwndRichEdit, WM_COPY, 0, 0); |
| |
| /*Reset the formatting to default*/ |
| cf2.dwEffects = CFE_ITALIC^cf2.dwEffects; |
| SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2); |
| |
| /*Clear the text in the control*/ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) ""); |
| |
| /*Switch to Plain Text Mode*/ |
| rc = SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_PLAINTEXT, 0); |
| ok(rc == 0, "EM_SETTEXTMODE: unable to switch to plain text mode with empty control: returned: %d\n", rc); |
| |
| /*Input "wine" again in normal format*/ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Paste the italicized "wine" into the control*/ |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| |
| /*Select a character from the first "wine" string*/ |
| cr.cpMin = 2; |
| cr.cpMax = 3; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /*Retrieve its formatting*/ |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, |
| (LPARAM) &cf2); |
| |
| /*Select a character from the second "wine" string*/ |
| cr.cpMin = 5; |
| cr.cpMax = 6; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /*Retrieve its formatting*/ |
| cf2test.cbSize = sizeof(CHARFORMAT2); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, |
| (LPARAM) &cf2test); |
| |
| /*Compare the two formattings*/ |
| ok((cf2.dwMask == cf2test.dwMask) && (cf2.dwEffects == cf2test.dwEffects), |
| "two formats found in plain text mode - cf2.dwEffects: %x cf2test.dwEffects: %x\n", |
| cf2.dwEffects, cf2test.dwEffects); |
| /*Test TM_RICHTEXT by: switching back to Rich Text mode |
| printing "wine" in the current format(normal) |
| pasting "wine" from the clipboard(italicized) |
| comparing the two formats(should differ)*/ |
| |
| /*Attempt to switch with text in control*/ |
| rc = SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_RICHTEXT, 0); |
| ok(rc != 0, "EM_SETTEXTMODE: changed from plain text to rich text with text in control - returned: %d\n", rc); |
| |
| /*Clear control*/ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) ""); |
| |
| /*Switch into Rich Text mode*/ |
| rc = SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_RICHTEXT, 0); |
| ok(rc == 0, "EM_SETTEXTMODE: unable to change to rich text with empty control - returned: %d\n", rc); |
| |
| /*Print "wine" in normal formatting into the control*/ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Paste italicized "wine" into the control*/ |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| |
| /*Select text from the first "wine" string*/ |
| cr.cpMin = 1; |
| cr.cpMax = 3; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /*Retrieve its formatting*/ |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, |
| (LPARAM) &cf2); |
| |
| /*Select text from the second "wine" string*/ |
| cr.cpMin = 6; |
| cr.cpMax = 7; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /*Retrieve its formatting*/ |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, |
| (LPARAM) &cf2test); |
| |
| /*Test that the two formattings are not the same*/ |
| todo_wine ok((cf2.dwMask == cf2test.dwMask) && (cf2.dwEffects != cf2test.dwEffects), |
| "expected different formats - cf2.dwMask: %x, cf2test.dwMask: %x, cf2.dwEffects: %x, cf2test.dwEffects: %x\n", |
| cf2.dwMask, cf2test.dwMask, cf2.dwEffects, cf2test.dwEffects); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_TM_PLAINTEXT(void) |
| { |
| /*Tests plain text properties*/ |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| CHARFORMAT2 cf2, cf2test; |
| CHARRANGE cr; |
| |
| /*Switch to plain text mode*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) ""); |
| SendMessage(hwndRichEdit, EM_SETTEXTMODE, TM_PLAINTEXT, 0); |
| |
| /*Fill control with text*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "Is Wine an emulator? No it's not"); |
| |
| /*Select some text and bold it*/ |
| |
| cr.cpMin = 10; |
| cr.cpMax = 20; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| cf2.cbSize = sizeof(CHARFORMAT2); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT, |
| (LPARAM) &cf2); |
| |
| cf2.dwMask = CFM_BOLD | cf2.dwMask; |
| cf2.dwEffects = CFE_BOLD ^ cf2.dwEffects; |
| |
| SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2); |
| |
| /*Get the formatting of those characters*/ |
| |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2); |
| |
| /*Get the formatting of some other characters*/ |
| cf2test.cbSize = sizeof(CHARFORMAT2); |
| cr.cpMin = 21; |
| cr.cpMax = 30; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2test); |
| |
| /*Test that they are the same as plain text allows only one formatting*/ |
| |
| ok((cf2.dwMask == cf2test.dwMask) && (cf2.dwEffects == cf2test.dwEffects), |
| "two selections' formats differ - cf2.dwMask: %x, cf2test.dwMask %x, cf2.dwEffects: %x, cf2test.dwEffects: %x\n", |
| cf2.dwMask, cf2test.dwMask, cf2.dwEffects, cf2test.dwEffects); |
| |
| /*Fill the control with a "wine" string, which when inserted will be bold*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Copy the bolded "wine" string*/ |
| |
| cr.cpMin = 0; |
| cr.cpMax = 4; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| SendMessage(hwndRichEdit, WM_COPY, 0, 0); |
| |
| /*Swap back to rich text*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) ""); |
| SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_RICHTEXT, 0); |
| |
| /*Set the default formatting to bold italics*/ |
| |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT, (LPARAM) &cf2); |
| cf2.dwMask |= CFM_ITALIC; |
| cf2.dwEffects ^= CFE_ITALIC; |
| SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2); |
| |
| /*Set the text in the control to "wine", which will be bold and italicized*/ |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "wine"); |
| |
| /*Paste the plain text "wine" string, which should take the insert |
| formatting, which at the moment is bold italics*/ |
| |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| |
| /*Select the first "wine" string and retrieve its formatting*/ |
| |
| cr.cpMin = 1; |
| cr.cpMax = 3; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2); |
| |
| /*Select the second "wine" string and retrieve its formatting*/ |
| |
| cr.cpMin = 5; |
| cr.cpMax = 7; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2test); |
| |
| /*Compare the two formattings. They should be the same.*/ |
| |
| ok((cf2.dwMask == cf2test.dwMask) && (cf2.dwEffects == cf2test.dwEffects), |
| "Copied text retained formatting - cf2.dwMask: %x, cf2test.dwMask: %x, cf2.dwEffects: %x, cf2test.dwEffects: %x\n", |
| cf2.dwMask, cf2test.dwMask, cf2.dwEffects, cf2test.dwEffects); |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_WM_GETTEXT(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| static const char text[] = "Hello. My name is RichEdit!"; |
| char buffer[1024] = {0}; |
| int result; |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp(buffer,text); |
| ok(result == 0, |
| "WM_GETTEXT: settext and gettext differ. strcmp: %d\n", result); |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| /* FIXME: need to test unimplemented options and robustly test wparam */ |
| static void test_EM_SETOPTIONS(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| static const char text[] = "Hello. My name is RichEdit!"; |
| char buffer[1024] = {0}; |
| |
| /* NEGATIVE TESTING - NO OPTIONS SET */ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| SendMessage(hwndRichEdit, EM_SETOPTIONS, ECOOP_SET, 0); |
| |
| /* testing no readonly by sending 'a' to the control*/ |
| SetFocus(hwndRichEdit); |
| SendMessage(hwndRichEdit, WM_CHAR, 'a', 0x1E0001); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| ok(buffer[0]=='a', |
| "EM_SETOPTIONS: Text not changed! s1:%s s2:%s\n", text, buffer); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| |
| /* READONLY - sending 'a' to the control */ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| SendMessage(hwndRichEdit, EM_SETOPTIONS, ECOOP_SET, ECO_READONLY); |
| SetFocus(hwndRichEdit); |
| SendMessage(hwndRichEdit, WM_CHAR, 'a', 0x1E0001); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| ok(buffer[0]==text[0], |
| "EM_SETOPTIONS: Text changed! s1:%s s2:%s\n", text, buffer); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void check_CFE_LINK_rcvd(HWND hwnd, int is_url) |
| { |
| CHARFORMAT2W text_format; |
| int link_present = 0; |
| text_format.cbSize = sizeof(text_format); |
| SendMessage(hwnd, EM_SETSEL, 0, 0); |
| SendMessage(hwnd, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM) &text_format); |
| link_present = text_format.dwEffects & CFE_LINK; |
| if (is_url) |
| { /* control text is url; should get CFE_LINK */ |
| ok(0 != link_present, "URL Case: CFE_LINK not set.\n"); |
| } |
| else |
| { |
| ok(0 == link_present, "Non-URL Case: CFE_LINK set.\n"); |
| } |
| } |
| |
| static HWND new_static_wnd(HWND parent) { |
| return new_window("Static", 0, parent); |
| } |
| |
| static void test_EM_AUTOURLDETECT(void) |
| { |
| struct urls_s { |
| const char *text; |
| int is_url; |
| } urls[12] = { |
| {"winehq.org", 0}, |
| {"http://www.winehq.org", 1}, |
| {"http//winehq.org", 0}, |
| {"ww.winehq.org", 0}, |
| {"www.winehq.org", 1}, |
| {"ftp://192.168.1.1", 1}, |
| {"ftp//192.168.1.1", 0}, |
| {"mailto:your@email.com", 1}, |
| {"prospero:prosperoserver", 1}, |
| {"telnet:test", 1}, |
| {"news:newserver", 1}, |
| {"wais:waisserver", 1} |
| }; |
| |
| int i; |
| int urlRet=-1; |
| HWND hwndRichEdit, parent; |
| |
| parent = new_static_wnd(NULL); |
| hwndRichEdit = new_richedit(parent); |
| /* Try and pass EM_AUTOURLDETECT some test wParam values */ |
| urlRet=SendMessage(hwndRichEdit, EM_AUTOURLDETECT, FALSE, 0); |
| ok(urlRet==0, "Good wParam: urlRet is: %d\n", urlRet); |
| urlRet=SendMessage(hwndRichEdit, EM_AUTOURLDETECT, 1, 0); |
| ok(urlRet==0, "Good wParam2: urlRet is: %d\n", urlRet); |
| /* Windows returns -2147024809 (0x80070057) on bad wParam values */ |
| urlRet=SendMessage(hwndRichEdit, EM_AUTOURLDETECT, 8, 0); |
| ok(urlRet==E_INVALIDARG, "Bad wParam: urlRet is: %d\n", urlRet); |
| urlRet=SendMessage(hwndRichEdit, EM_AUTOURLDETECT, (WPARAM)"h", (LPARAM)"h"); |
| ok(urlRet==E_INVALIDARG, "Bad wParam2: urlRet is: %d\n", urlRet); |
| /* for each url, check the text to see if CFE_LINK effect is present */ |
| for (i = 0; i < sizeof(urls)/sizeof(struct urls_s); i++) { |
| SendMessage(hwndRichEdit, EM_AUTOURLDETECT, FALSE, 0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) urls[i].text); |
| SendMessage(hwndRichEdit, WM_CHAR, 0, 0); |
| check_CFE_LINK_rcvd(hwndRichEdit, 0); |
| SendMessage(hwndRichEdit, EM_AUTOURLDETECT, TRUE, 0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) urls[i].text); |
| SendMessage(hwndRichEdit, WM_CHAR, 0, 0); |
| check_CFE_LINK_rcvd(hwndRichEdit, urls[i].is_url); |
| } |
| DestroyWindow(hwndRichEdit); |
| DestroyWindow(parent); |
| } |
| |
| static void test_EM_SCROLL(void) |
| { |
| int i, j; |
| int r; /* return value */ |
| int expr; /* expected return value */ |
| HWND hwndRichEdit = new_richedit(NULL); |
| int y_before, y_after; /* units of lines of text */ |
| |
| /* test a richedit box containing a single line of text */ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "a");/* one line of text */ |
| expr = 0x00010000; |
| for (i = 0; i < 4; i++) { |
| static const int cmd[4] = { SB_PAGEDOWN, SB_PAGEUP, SB_LINEDOWN, SB_LINEUP }; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, cmd[i], 0); |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| ok(expr == r, "EM_SCROLL improper return value returned (i == %d). " |
| "Got 0x%08x, expected 0x%08x\n", i, r, expr); |
| ok(y_after == 0, "EM_SCROLL improper scroll. scrolled to line %d, not 1 " |
| "(i == %d)\n", y_after, i); |
| } |
| |
| /* |
| * test a richedit box that will scroll. There are two general |
| * cases: the case without any long lines and the case with a long |
| * line. |
| */ |
| for (i = 0; i < 2; i++) { /* iterate through different bodies of text */ |
| if (i == 0) |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "a\nb\nc\nd\ne"); |
| else |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) |
| "a LONG LINE LONG LINE LONG LINE LONG LINE LONG LINE " |
| "LONG LINE LONG LINE LONG LINE LONG LINE LONG LINE " |
| "LONG LINE \nb\nc\nd\ne"); |
| for (j = 0; j < 12; j++) /* reset scrol position to top */ |
| SendMessage(hwndRichEdit, EM_SCROLL, SB_PAGEUP, 0); |
| |
| /* get first visible line */ |
| y_before = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| r = SendMessage(hwndRichEdit, EM_SCROLL, SB_PAGEDOWN, 0); /* page down */ |
| |
| /* get new current first visible line */ |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(((r & 0xffffff00) == 0x00010000) && |
| ((r & 0x000000ff) != 0x00000000), |
| "EM_SCROLL page down didn't scroll by a small positive number of " |
| "lines (r == 0x%08x)\n", r); |
| ok(y_after > y_before, "EM_SCROLL page down not functioning " |
| "(line %d scrolled to line %d\n", y_before, y_after); |
| |
| y_before = y_after; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, SB_PAGEUP, 0); /* page up */ |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| ok(((r & 0xffffff00) == 0x0001ff00), |
| "EM_SCROLL page up didn't scroll by a small negative number of lines " |
| "(r == 0x%08x)\n", r); |
| ok(y_after < y_before, "EM_SCROLL page up not functioning (line " |
| "%d scrolled to line %d\n", y_before, y_after); |
| |
| y_before = y_after; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEDOWN, 0); /* line down */ |
| |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x00010001, "EM_SCROLL line down didn't scroll by one line " |
| "(r == 0x%08x)\n", r); |
| ok(y_after -1 == y_before, "EM_SCROLL line down didn't go down by " |
| "1 line (%d scrolled to %d)\n", y_before, y_after); |
| |
| y_before = y_after; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEUP, 0); /* line up */ |
| |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x0001ffff, "EM_SCROLL line up didn't scroll by one line " |
| "(r == 0x%08x)\n", r); |
| ok(y_after +1 == y_before, "EM_SCROLL line up didn't go up by 1 " |
| "line (%d scrolled to %d)\n", y_before, y_after); |
| |
| y_before = y_after; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, |
| SB_LINEUP, 0); /* lineup beyond top */ |
| |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x00010000, |
| "EM_SCROLL line up returned indicating movement (0x%08x)\n", r); |
| ok(y_before == y_after, |
| "EM_SCROLL line up beyond top worked (%d)\n", y_after); |
| |
| y_before = y_after; |
| |
| r = SendMessage(hwndRichEdit, EM_SCROLL, |
| SB_PAGEUP, 0);/*page up beyond top */ |
| |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x00010000, |
| "EM_SCROLL page up returned indicating movement (0x%08x)\n", r); |
| ok(y_before == y_after, |
| "EM_SCROLL page up beyond top worked (%d)\n", y_after); |
| |
| for (j = 0; j < 12; j++) /* page down all the way to the bottom */ |
| SendMessage(hwndRichEdit, EM_SCROLL, SB_PAGEDOWN, 0); |
| y_before = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| r = SendMessage(hwndRichEdit, EM_SCROLL, |
| SB_PAGEDOWN, 0); /* page down beyond bot */ |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x00010000, |
| "EM_SCROLL page down returned indicating movement (0x%08x)\n", r); |
| ok(y_before == y_after, |
| "EM_SCROLL page down beyond bottom worked (%d -> %d)\n", |
| y_before, y_after); |
| |
| y_before = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| SendMessage(hwndRichEdit, EM_SCROLL, |
| SB_LINEDOWN, 0); /* line down beyond bot */ |
| y_after = SendMessage(hwndRichEdit, EM_GETFIRSTVISIBLELINE, 0, 0); |
| |
| ok(r == 0x00010000, |
| "EM_SCROLL line down returned indicating movement (0x%08x)\n", r); |
| ok(y_before == y_after, |
| "EM_SCROLL line down beyond bottom worked (%d -> %d)\n", |
| y_before, y_after); |
| } |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_EM_SETUNDOLIMIT(void) |
| { |
| /* cases we test for: |
| * default behaviour - limiting at 100 undo's |
| * undo disabled - setting a limit of 0 |
| * undo limited - undo limit set to some to some number, like 2 |
| * bad input - sending a negative number should default to 100 undo's */ |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| CHARRANGE cr; |
| int i; |
| int result; |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "x"); |
| cr.cpMin = 0; |
| cr.cpMax = 1; |
| SendMessage(hwndRichEdit, WM_COPY, 0, 0); |
| /*Load "x" into the clipboard. Paste is an easy, undo'able operation. |
| also, multiple pastes don't combine like WM_CHAR would */ |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| /* first case - check the default */ |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0,0); |
| for (i=0; i<101; i++) /* Put 101 undo's on the stack */ |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| for (i=0; i<100; i++) /* Undo 100 of them */ |
| SendMessage(hwndRichEdit, WM_UNDO, 0, 0); |
| ok(!SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0), |
| "EM_SETUNDOLIMIT allowed more than a hundred undo's by default.\n"); |
| |
| /* second case - cannot undo */ |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0, 0); |
| SendMessage(hwndRichEdit, EM_SETUNDOLIMIT, 0, 0); |
| SendMessage(hwndRichEdit, |
| WM_PASTE, 0, 0); /* Try to put something in the undo stack */ |
| ok(!SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0), |
| "EM_SETUNDOLIMIT allowed undo with UNDOLIMIT set to 0\n"); |
| |
| /* third case - set it to an arbitrary number */ |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0, 0); |
| SendMessage(hwndRichEdit, EM_SETUNDOLIMIT, 2, 0); |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| /* If SETUNDOLIMIT is working, there should only be two undo's after this */ |
| ok(SendMessage(hwndRichEdit, EM_CANUNDO, 0,0), |
| "EM_SETUNDOLIMIT didn't allow the first undo with UNDOLIMIT set to 2\n"); |
| SendMessage(hwndRichEdit, WM_UNDO, 0, 0); |
| ok(SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0), |
| "EM_SETUNDOLIMIT didn't allow a second undo with UNDOLIMIT set to 2\n"); |
| SendMessage(hwndRichEdit, WM_UNDO, 0, 0); |
| ok(!SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0), |
| "EM_SETUNDOLIMIT allowed a third undo with UNDOLIMIT set to 2\n"); |
| |
| /* fourth case - setting negative numbers should default to 100 undos */ |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0,0); |
| result = SendMessage(hwndRichEdit, EM_SETUNDOLIMIT, -1, 0); |
| ok (result == 100, |
| "EM_SETUNDOLIMIT returned %d when set to -1, instead of 100\n",result); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_ES_PASSWORD(void) |
| { |
| /* This isn't hugely testable, so we're just going to run it through it's paces. */ |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| WCHAR result; |
| |
| /* First, check the default of a regular control */ |
| result = SendMessage(hwndRichEdit, EM_GETPASSWORDCHAR, 0, 0); |
| ok (result == 0, |
| "EM_GETPASSWORDCHAR returned %c by default, instead of NULL\n",result); |
| |
| /* Now, set it to something normal */ |
| SendMessage(hwndRichEdit, EM_SETPASSWORDCHAR, 'x', 0); |
| result = SendMessage(hwndRichEdit, EM_GETPASSWORDCHAR, 0, 0); |
| ok (result == 120, |
| "EM_GETPASSWORDCHAR returned %c (%d) when set to 'x', instead of x (120)\n",result,result); |
| |
| /* Now, set it to something odd */ |
| SendMessage(hwndRichEdit, EM_SETPASSWORDCHAR, (WCHAR)1234, 0); |
| result = SendMessage(hwndRichEdit, EM_GETPASSWORDCHAR, 0, 0); |
| ok (result == 1234, |
| "EM_GETPASSWORDCHAR returned %c (%d) when set to 'x', instead of x (120)\n",result,result); |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_EM_SETTEXTEX(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| SETTEXTEX setText; |
| GETTEXTEX getText; |
| WCHAR TestItem1[] = {'T', 'e', 's', 't', |
| 'S', 'o', 'm', 'e', |
| 'T', 'e', 'x', 't', 0}; |
| #define MAX_BUF_LEN 1024 |
| WCHAR buf[MAX_BUF_LEN]; |
| int result; |
| CHARRANGE cr; |
| |
| setText.codepage = 1200; /* no constant for unicode */ |
| getText.codepage = 1200; /* no constant for unicode */ |
| getText.cb = MAX_BUF_LEN; |
| getText.flags = GT_DEFAULT; |
| |
| setText.flags = 0; |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) TestItem1); |
| SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf); |
| ok(lstrcmpW(buf, TestItem1) == 0, |
| "EM_GETTEXTEX results not what was set by EM_SETTEXTEX\n"); |
| |
| result = SendMessage(hwndRichEdit, EM_SETTEXTEX, |
| (WPARAM)&setText, (LPARAM) NULL); |
| SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf); |
| |
| ok (result == 1, |
| "EM_SETTEXTEX returned %d, instead of 1\n",result); |
| ok(lstrlenW(buf) == 0, |
| "EM_SETTEXTEX with NULL lParam should clear rich edit.\n"); |
| |
| /* put some text back */ |
| setText.flags = 0; |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) TestItem1); |
| /* select some text */ |
| cr.cpMax = 1; |
| cr.cpMin = 3; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| /* replace current selection */ |
| setText.flags = ST_SELECTION; |
| result = SendMessage(hwndRichEdit, EM_SETTEXTEX, |
| (WPARAM)&setText, (LPARAM) NULL); |
| ok(result == 0, |
| "EM_SETTEXTEX with NULL lParam to replace selection" |
| " with no text should return 0. Got %i\n", |
| result); |
| |
| /* put some text back */ |
| setText.flags = 0; |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) TestItem1); |
| /* select some text */ |
| cr.cpMax = 1; |
| cr.cpMin = 3; |
| SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr); |
| /* replace current selection */ |
| setText.flags = ST_SELECTION; |
| result = SendMessage(hwndRichEdit, EM_SETTEXTEX, |
| (WPARAM)&setText, (LPARAM) TestItem1); |
| /* get text */ |
| SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf); |
| ok(result == lstrlenW(TestItem1), |
| "EM_SETTEXTEX with NULL lParam to replace selection" |
| " with no text should return 0. Got %i\n", |
| result); |
| ok(lstrlenW(buf) == 22, |
| "EM_SETTEXTEX to replace selection with more text failed: %i.\n", |
| lstrlenW(buf) ); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_EM_LIMITTEXT(void) |
| { |
| int ret; |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| |
| /* The main purpose of this test is to demonstrate that the nonsense in MSDN |
| * about setting the length to -1 for multiline edit controls doesn't happen. |
| */ |
| |
| /* Don't check default gettextlimit case. That's done in other tests */ |
| |
| /* Set textlimit to 100 */ |
| SendMessage (hwndRichEdit, EM_LIMITTEXT, 100, 0); |
| ret = SendMessage (hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok (ret == 100, |
| "EM_LIMITTEXT: set to 100, returned: %d, expected: 100\n", ret); |
| |
| /* Set textlimit to 0 */ |
| SendMessage (hwndRichEdit, EM_LIMITTEXT, 0, 0); |
| ret = SendMessage (hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok (ret == 65536, |
| "EM_LIMITTEXT: set to 0, returned: %d, expected: 65536\n", ret); |
| |
| /* Set textlimit to -1 */ |
| SendMessage (hwndRichEdit, EM_LIMITTEXT, -1, 0); |
| ret = SendMessage (hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok (ret == -1, |
| "EM_LIMITTEXT: set to -1, returned: %d, expected: -1\n", ret); |
| |
| /* Set textlimit to -2 */ |
| SendMessage (hwndRichEdit, EM_LIMITTEXT, -2, 0); |
| ret = SendMessage (hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok (ret == -2, |
| "EM_LIMITTEXT: set to -2, returned: %d, expected: -2\n", ret); |
| |
| DestroyWindow (hwndRichEdit); |
| } |
| |
| |
| static void test_EM_EXLIMITTEXT(void) |
| { |
| int i, selBegin, selEnd, len1, len2; |
| char text[1024 + 1]; |
| int textlimit = 0; /* multiple of 100 */ |
| HWND hwndRichEdit = new_richedit(NULL); |
| |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok(32767 == i, "EM_EXLIMITTEXT: expected: %d, actual: %d\n", 32767, i); /* default */ |
| |
| textlimit = 256000; |
| SendMessage(hwndRichEdit, EM_EXLIMITTEXT, 0, textlimit); |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| /* set higher */ |
| ok(textlimit == i, "EM_EXLIMITTEXT: expected: %d, actual: %d\n", textlimit, i); |
| |
| textlimit = 1000; |
| SendMessage(hwndRichEdit, EM_EXLIMITTEXT, 0, textlimit); |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| /* set lower */ |
| ok(textlimit == i, "EM_EXLIMITTEXT: expected: %d, actual: %d\n", textlimit, i); |
| |
| SendMessage(hwndRichEdit, EM_EXLIMITTEXT, 0, 0); |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| /* default for WParam = 0 */ |
| ok(65536 == i, "EM_EXLIMITTEXT: expected: %d, actual: %d\n", 65536, i); |
| |
| textlimit = sizeof(text)-1; |
| memset(text, 'W', textlimit); |
| text[sizeof(text)-1] = 0; |
| SendMessage(hwndRichEdit, EM_EXLIMITTEXT, 0, textlimit); |
| /* maxed out text */ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text); |
| |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, -1); /* select everything */ |
| SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&selBegin, (LPARAM)&selEnd); |
| len1 = selEnd - selBegin; |
| |
| SendMessage(hwndRichEdit, WM_KEYDOWN, VK_BACK, 1); |
| SendMessage(hwndRichEdit, WM_CHAR, VK_BACK, 1); |
| SendMessage(hwndRichEdit, WM_KEYUP, VK_BACK, 1); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, -1); |
| SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&selBegin, (LPARAM)&selEnd); |
| len2 = selEnd - selBegin; |
| |
| ok(len1 != len2, |
| "EM_EXLIMITTEXT: Change Expected\nOld Length: %d, New Length: %d, Limit: %d\n", |
| len1,len2,i); |
| |
| SendMessage(hwndRichEdit, WM_KEYDOWN, 'A', 1); |
| SendMessage(hwndRichEdit, WM_CHAR, 'A', 1); |
| SendMessage(hwndRichEdit, WM_KEYUP, 'A', 1); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, -1); |
| SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&selBegin, (LPARAM)&selEnd); |
| len1 = selEnd - selBegin; |
| |
| ok(len1 != len2, |
| "EM_EXLIMITTEXT: Change Expected\nOld Length: %d, New Length: %d, Limit: %d\n", |
| len1,len2,i); |
| |
| SendMessage(hwndRichEdit, WM_KEYDOWN, 'A', 1); |
| SendMessage(hwndRichEdit, WM_CHAR, 'A', 1); |
| SendMessage(hwndRichEdit, WM_KEYUP, 'A', 1); /* full; should be no effect */ |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, -1); |
| SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&selBegin, (LPARAM)&selEnd); |
| len2 = selEnd - selBegin; |
| |
| ok(len1 == len2, |
| "EM_EXLIMITTEXT: No Change Expected\nOld Length: %d, New Length: %d, Limit: %d\n", |
| len1,len2,i); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_EM_GETLIMITTEXT(void) |
| { |
| int i; |
| HWND hwndRichEdit = new_richedit(NULL); |
| |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok(32767 == i, "expected: %d, actual: %d\n", 32767, i); /* default value */ |
| |
| SendMessage(hwndRichEdit, EM_EXLIMITTEXT, 0, 50000); |
| i = SendMessage(hwndRichEdit, EM_GETLIMITTEXT, 0, 0); |
| ok(50000 == i, "expected: %d, actual: %d\n", 50000, i); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_WM_SETFONT(void) |
| { |
| /* There is no invalid input or error conditions for this function. |
| * NULL wParam and lParam just fall back to their default values |
| * It should be noted that even if you use a gibberish name for your fonts |
| * here, it will still work because the name is stored. They will display as |
| * System, but will report their name to be whatever they were created as */ |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| HFONT testFont1 = CreateFontA (0,0,0,0,FW_LIGHT, 0, 0, 0, ANSI_CHARSET, |
| OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | |
| FF_DONTCARE, "Marlett"); |
| HFONT testFont2 = CreateFontA (0,0,0,0,FW_LIGHT, 0, 0, 0, ANSI_CHARSET, |
| OUT_TT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | |
| FF_DONTCARE, "MS Sans Serif"); |
| HFONT testFont3 = CreateFontA (0,0,0,0,FW_LIGHT, 0, 0, 0, ANSI_CHARSET, |
| OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | |
| FF_DONTCARE, "Courier"); |
| LOGFONTA sentLogFont; |
| CHARFORMAT2A returnedCF2A; |
| |
| returnedCF2A.cbSize = sizeof(returnedCF2A); |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "x"); |
| SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont1,(LPARAM) MAKELONG((WORD) TRUE, 0)); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A); |
| |
| GetObjectA(testFont1, sizeof(LOGFONTA), &sentLogFont); |
| ok (!strcmp(sentLogFont.lfFaceName,returnedCF2A.szFaceName), |
| "EM_GETCHARFOMAT: Returned wrong font on test 1. Sent: %s, Returned: %s\n", |
| sentLogFont.lfFaceName,returnedCF2A.szFaceName); |
| |
| SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont2,(LPARAM) MAKELONG((WORD) TRUE, 0)); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A); |
| GetObjectA(testFont2, sizeof(LOGFONTA), &sentLogFont); |
| ok (!strcmp(sentLogFont.lfFaceName,returnedCF2A.szFaceName), |
| "EM_GETCHARFOMAT: Returned wrong font on test 2. Sent: %s, Returned: %s\n", |
| sentLogFont.lfFaceName,returnedCF2A.szFaceName); |
| |
| SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont3,(LPARAM) MAKELONG((WORD) TRUE, 0)); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A); |
| GetObjectA(testFont3, sizeof(LOGFONTA), &sentLogFont); |
| ok (!strcmp(sentLogFont.lfFaceName,returnedCF2A.szFaceName), |
| "EM_GETCHARFOMAT: Returned wrong font on test 3. Sent: %s, Returned: %s\n", |
| sentLogFont.lfFaceName,returnedCF2A.szFaceName); |
| |
| /* This last test is special since we send in NULL. We clear the variables |
| * and just compare to "System" instead of the sent in font name. */ |
| ZeroMemory(&returnedCF2A,sizeof(returnedCF2A)); |
| ZeroMemory(&sentLogFont,sizeof(sentLogFont)); |
| returnedCF2A.cbSize = sizeof(returnedCF2A); |
| |
| SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)NULL,(LPARAM) MAKELONG((WORD) TRUE, 0)); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A); |
| GetObjectA(NULL, sizeof(LOGFONTA), &sentLogFont); |
| ok (!strcmp("System",returnedCF2A.szFaceName), |
| "EM_GETCHARFOMAT: Returned wrong font on test 4. Sent: NULL, Returned: %s. Expected \"System\".\n",returnedCF2A.szFaceName); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| |
| static DWORD CALLBACK test_EM_GETMODIFY_esCallback(DWORD_PTR dwCookie, |
| LPBYTE pbBuff, |
| LONG cb, |
| LONG *pcb) |
| { |
| const char** str = (const char**)dwCookie; |
| int size = strlen(*str); |
| if(size > 3) /* let's make it peice-meal for fun */ |
| size = 3; |
| *pcb = cb; |
| if (*pcb > size) { |
| *pcb = size; |
| } |
| if (*pcb > 0) { |
| memcpy(pbBuff, *str, *pcb); |
| *str += *pcb; |
| } |
| return 0; |
| } |
| |
| static void test_EM_GETMODIFY(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| LRESULT result; |
| SETTEXTEX setText; |
| WCHAR TestItem1[] = {'T', 'e', 's', 't', |
| 'S', 'o', 'm', 'e', |
| 'T', 'e', 'x', 't', 0}; |
| WCHAR TestItem2[] = {'T', 'e', 's', 't', |
| 'S', 'o', 'm', 'e', |
| 'O', 't', 'h', 'e', 'r', |
| 'T', 'e', 'x', 't', 0}; |
| const char* streamText = "hello world"; |
| CHARFORMAT2 cf2; |
| PARAFORMAT2 pf2; |
| EDITSTREAM es; |
| |
| HFONT testFont = CreateFontA (0,0,0,0,FW_LIGHT, 0, 0, 0, ANSI_CHARSET, |
| OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | |
| FF_DONTCARE, "Courier"); |
| |
| setText.codepage = 1200; /* no constant for unicode */ |
| setText.flags = ST_KEEPUNDO; |
| |
| |
| /* modify flag shouldn't be set when richedit is first created */ |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero, instead of zero on create\n"); |
| |
| /* setting modify flag should actually set it */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, TRUE, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero on EM_SETMODIFY\n"); |
| |
| /* clearing modify flag should actually clear it */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero, instead of zero on EM_SETMODIFY\n"); |
| |
| /* setting font doesn't change modify flag */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont,(LPARAM) MAKELONG((WORD) TRUE, 0)); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero, instead of zero on setting font\n"); |
| |
| /* setting text should set modify flag */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM)TestItem1); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero on setting text\n"); |
| |
| /* undo previous text doesn't reset modify flag */ |
| SendMessage(hwndRichEdit, WM_UNDO, 0, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero on undo after setting text\n"); |
| |
| /* set text with no flag to keep undo stack should not set modify flag */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| setText.flags = 0; |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM)TestItem1); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero, instead of zero when setting text while not keeping undo stack\n"); |
| |
| /* WM_SETTEXT doesn't modify */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)TestItem2); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| todo_wine { |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero for WM_SETTEXT\n"); |
| } |
| |
| /* clear the text */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, WM_CLEAR, 0, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned non-zero, instead of zero for WM_CLEAR\n"); |
| |
| /* replace text */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM)TestItem1); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 2); |
| SendMessage(hwndRichEdit, EM_REPLACESEL, TRUE, (LPARAM)TestItem2); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero when replacing text\n"); |
| |
| /* copy/paste text 1 */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 2); |
| SendMessage(hwndRichEdit, WM_COPY, 0, 0); |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero when pasting identical text\n"); |
| |
| /* copy/paste text 2 */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 2); |
| SendMessage(hwndRichEdit, WM_COPY, 0, 0); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 3); |
| SendMessage(hwndRichEdit, WM_PASTE, 0, 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero when pasting different text\n"); |
| |
| /* press char */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 1); |
| SendMessage(hwndRichEdit, WM_CHAR, 'A', 0); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero for WM_CHAR\n"); |
| |
| /* set char format */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| cf2.cbSize = sizeof(CHARFORMAT2); |
| SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT, |
| (LPARAM) &cf2); |
| cf2.dwMask = CFM_ITALIC | cf2.dwMask; |
| cf2.dwEffects = CFE_ITALIC ^ cf2.dwEffects; |
| SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero for EM_SETCHARFORMAT\n"); |
| |
| /* set para format */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| pf2.cbSize = sizeof(PARAFORMAT2); |
| SendMessage(hwndRichEdit, EM_GETPARAFORMAT, 0, |
| (LPARAM) &pf2); |
| pf2.dwMask = PFM_ALIGNMENT | pf2.dwMask; |
| pf2.wAlignment = PFA_RIGHT; |
| SendMessage(hwndRichEdit, EM_SETPARAFORMAT, 0, (LPARAM) &pf2); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result == 0, |
| "EM_GETMODIFY returned zero, instead of non-zero for EM_SETPARAFORMAT\n"); |
| |
| /* EM_STREAM */ |
| SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0); |
| es.dwCookie = (DWORD_PTR)&streamText; |
| es.dwError = 0; |
| es.pfnCallback = test_EM_GETMODIFY_esCallback; |
| SendMessage(hwndRichEdit, EM_STREAMIN, |
| (WPARAM)(SF_TEXT), (LPARAM)&es); |
| result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0); |
| ok (result != 0, |
| "EM_GETMODIFY returned zero, instead of non-zero for EM_STREAM\n"); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| struct exsetsel_s { |
| long min; |
| long max; |
| long expected_retval; |
| int expected_getsel_start; |
| int expected_getsel_end; |
| int _exsetsel_todo_wine; |
| int _getsel_todo_wine; |
| }; |
| |
| const struct exsetsel_s exsetsel_tests[] = { |
| /* sanity tests */ |
| {5, 10, 10, 5, 10, 0, 0}, |
| {15, 17, 17, 15, 17, 0, 0}, |
| /* test cpMax > strlen() */ |
| {0, 100, 18, 0, 18, 0, 1}, |
| /* test cpMin == cpMax */ |
| {5, 5, 5, 5, 5, 0, 0}, |
| /* test cpMin < 0 && cpMax >= 0 (bug 4462) */ |
| {-1, 0, 5, 5, 5, 0, 0}, |
| {-1, 17, 5, 5, 5, 0, 0}, |
| {-1, 18, 5, 5, 5, 0, 0}, |
| /* test cpMin < 0 && cpMax < 0 */ |
| {-1, -1, 17, 17, 17, 0, 0}, |
| {-4, -5, 17, 17, 17, 0, 0}, |
| /* test cMin >=0 && cpMax < 0 (bug 6814) */ |
| {0, -1, 18, 0, 18, 0, 1}, |
| {17, -5, 18, 17, 18, 0, 1}, |
| {18, -3, 17, 17, 17, 0, 0}, |
| /* test if cpMin > cpMax */ |
| {15, 19, 18, 15, 18, 0, 1}, |
| {19, 15, 18, 15, 18, 0, 1} |
| }; |
| |
| static void check_EM_EXSETSEL(HWND hwnd, const struct exsetsel_s *setsel, int id) { |
| CHARRANGE cr; |
| long result; |
| int start, end; |
| |
| cr.cpMin = setsel->min; |
| cr.cpMax = setsel->max; |
| result = SendMessage(hwnd, EM_EXSETSEL, 0, (LPARAM) &cr); |
| |
| if (setsel->_exsetsel_todo_wine) { |
| todo_wine { |
| ok(result == setsel->expected_retval, "EM_EXSETSEL(%d): expected: %ld actual: %ld\n", id, setsel->expected_retval, result); |
| } |
| } else { |
| ok(result == setsel->expected_retval, "EM_EXSETSEL(%d): expected: %ld actual: %ld\n", id, setsel->expected_retval, result); |
| } |
| |
| SendMessage(hwnd, EM_GETSEL, (WPARAM) &start, (LPARAM) &end); |
| |
| if (setsel->_getsel_todo_wine) { |
| todo_wine { |
| ok(start == setsel->expected_getsel_start && end == setsel->expected_getsel_end, "EM_EXSETSEL(%d): expected (%d,%d) actual:(%d,%d)\n", id, setsel->expected_getsel_start, setsel->expected_getsel_end, start, end); |
| } |
| } else { |
| ok(start == setsel->expected_getsel_start && end == setsel->expected_getsel_end, "EM_EXSETSEL(%d): expected (%d,%d) actual:(%d,%d)\n", id, setsel->expected_getsel_start, setsel->expected_getsel_end, start, end); |
| } |
| } |
| |
| static void test_EM_EXSETSEL(void) |
| { |
| HWND hwndRichEdit = new_richedit(NULL); |
| int i; |
| const int num_tests = sizeof(exsetsel_tests)/sizeof(struct exsetsel_s); |
| |
| /* sending some text to the window */ |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "testing selection"); |
| /* 01234567890123456*/ |
| /* 10 */ |
| |
| for (i = 0; i < num_tests; i++) { |
| check_EM_EXSETSEL(hwndRichEdit, &exsetsel_tests[i], i); |
| } |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static void test_WM_PASTE(void) |
| { |
| int result; |
| char buffer[1024] = {0}; |
| const char* text1 = "testing paste\r"; |
| const char* text2 = "testing paste\r\rtesting paste"; |
| const char* text3 = "testing paste\rpaste\rtesting paste"; |
| HWND hwndRichEdit = new_richedit(NULL); |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text1); |
| SendMessage(hwndRichEdit, EM_SETSEL, 0, 14); |
| SendMessage(hwndRichEdit, WM_CHAR, 3, 0); /* ctrl-c */ |
| SendMessage(hwndRichEdit, EM_SETSEL, 14, 14); |
| SendMessage(hwndRichEdit, WM_CHAR, 22, 0); /* ctrl-v */ |
| SendMessage(hwndRichEdit, WM_CHAR, 26, 0); /* ctrl-z */ |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp(text1, buffer); |
| ok(result == 0, |
| "test paste: strcmp = %i\n", result); |
| |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text2); |
| SendMessage(hwndRichEdit, EM_SETSEL, 8, 13); |
| SendMessage(hwndRichEdit, WM_CHAR, 3, 0); /* ctrl-c */ |
| SendMessage(hwndRichEdit, EM_SETSEL, 14, 14); |
| SendMessage(hwndRichEdit, WM_CHAR, 22, 0); /* ctrl-v */ |
| SendMessage(hwndRichEdit, WM_CHAR, 26, 0); /* ctrl-z */ |
| SendMessage(hwndRichEdit, WM_CHAR, 25, 0); /* ctrl-y */ |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp(buffer,text3); |
| ok(result == 0, |
| "test paste: strcmp = %i\n", result); |
| |
| DestroyWindow(hwndRichEdit); |
| } |
| |
| static int nCallbackCount = 0; |
| |
| static DWORD CALLBACK EditStreamCallback(DWORD_PTR dwCookie, LPBYTE pbBuff, |
| LONG cb, LONG* pcb) |
| { |
| const char text[] = {'t','e','s','t'}; |
| |
| if (sizeof(text) <= cb) |
| { |
| if ((int)dwCookie != nCallbackCount) |
| { |
| *pcb = 0; |
| return 0; |
| } |
| |
| memcpy (pbBuff, text, sizeof(text)); |
| *pcb = sizeof(text); |
| |
| nCallbackCount++; |
| |
| return 0; |
| } |
| else |
| return 1; /* indicates callback failed */ |
| } |
| |
| static void test_EM_StreamIn_Undo(void) |
| { |
| /* The purpose of this test is to determine when a EM_StreamIn should be |
| * undoable. This is important because WM_PASTE currently uses StreamIn and |
| * pasting should always be undoable but streaming isn't always. |
| * |
| * cases to test: |
| * StreamIn plain text without SFF_SELECTION. |
| * StreamIn plain text with SFF_SELECTION set but a zero-length selection |
| * StreamIn plain text with SFF_SELECTION and a valid, normal selection |
| * StreamIn plain text with SFF_SELECTION and a backwards-selection (from>to) |
| * Feel free to add tests for other text modes or StreamIn things. |
| */ |
| |
| |
| HWND hwndRichEdit = new_richedit(NULL); |
| LRESULT result; |
| EDITSTREAM es; |
| char buffer[1024] = {0}; |
| const char randomtext[] = "Some text"; |
| |
| es.pfnCallback = (EDITSTREAMCALLBACK) EditStreamCallback; |
| |
| /* StreamIn, no SFF_SELECTION */ |
| es.dwCookie = nCallbackCount; |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0,0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) randomtext); |
| SendMessage(hwndRichEdit, EM_SETSEL,0,0); |
| SendMessage(hwndRichEdit, EM_STREAMIN, (WPARAM)SF_TEXT, (LPARAM)&es); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp (buffer,"test"); |
| ok (result == 0, |
| "EM_STREAMIN: Test 1 set wrong text: Result: %s\n",buffer); |
| |
| result = SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0); |
| ok (result == FALSE, |
| "EM_STREAMIN without SFF_SELECTION wrongly allows undo\n"); |
| |
| /* StreamIn, SFF_SELECTION, but nothing selected */ |
| es.dwCookie = nCallbackCount; |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0,0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) randomtext); |
| SendMessage(hwndRichEdit, EM_SETSEL,0,0); |
| SendMessage(hwndRichEdit, EM_STREAMIN, |
| (WPARAM)(SF_TEXT|SFF_SELECTION), (LPARAM)&es); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp (buffer,"testSome text"); |
| ok (result == 0, |
| "EM_STREAMIN: Test 2 set wrong text: Result: %s\n",buffer); |
| |
| result = SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0); |
| ok (result == TRUE, |
| "EM_STREAMIN with SFF_SELECTION but no selection set " |
| "should create an undo\n"); |
| |
| /* StreamIn, SFF_SELECTION, with a selection */ |
| es.dwCookie = nCallbackCount; |
| SendMessage(hwndRichEdit,EM_EMPTYUNDOBUFFER, 0,0); |
| SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) randomtext); |
| SendMessage(hwndRichEdit, EM_SETSEL,4,5); |
| SendMessage(hwndRichEdit, EM_STREAMIN, |
| (WPARAM)(SF_TEXT|SFF_SELECTION), (LPARAM)&es); |
| SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer); |
| result = strcmp (buffer,"Sometesttext"); |
| ok (result == 0, |
| "EM_STREAMIN: Test 2 set wrong text: Result: %s\n",buffer); |
| |
| result = SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0); |
| ok (result == TRUE, |
| "EM_STREAMIN with SFF_SELECTION and selection set " |
| "should create an undo\n"); |
| |
| } |
| |
| static void test_unicode_conversions(void) |
| { |
| static const WCHAR textW[] = {'t','e','s','t',0}; |
| static const char textA[] = "test"; |
| char bufA[64]; |
| WCHAR bufW[64]; |
| HWND hwnd; |
| int is_win9x, ret; |
| |
| is_win9x = GetVersion() & 0x80000000; |
| |
| #define set_textA(hwnd, txt) \ |
| do { \ |
| ret = SendMessageA(hwnd, WM_SETTEXT, 0, (LPARAM)txt); \ |
| ok(ret, "SendMessageA(WM_SETTEXT) error %u\n", GetLastError()); \ |
| } while(0) |
| #define expect_textA(hwnd, txt) \ |
| do { \ |
| memset(bufA, 0xAA, sizeof(bufA)); \ |
| ret = SendMessageA(hwnd, WM_GETTEXT, 64, (LPARAM)bufA); \ |
| ok(ret, "SendMessageA(WM_GETTEXT) error %u\n", GetLastError()); \ |
| ret = lstrcmpA(bufA, txt); \ |
| ok(!ret, "strings not match: expected %s got %s\n", txt, bufA); \ |
| } while(0) |
| |
| #define set_textW(hwnd, txt) \ |
| do { \ |
| ret = SendMessageW(hwnd, WM_SETTEXT, 0, (LPARAM)txt); \ |
| ok(ret, "SendMessageW(WM_SETTEXT) error %u\n", GetLastError()); \ |
| } while(0) |
| #define expect_textW(hwnd, txt) \ |
| do { \ |
| memset(bufW, 0xAA, sizeof(bufW)); \ |
| ret = SendMessageW(hwnd, WM_GETTEXT, 64, (LPARAM)bufW); \ |
| ok(ret, "SendMessageW(WM_GETTEXT) error %u\n", GetLastError()); \ |
| ret = lstrcmpW(bufW, txt); \ |
| ok(!ret, "strings not match expected[0] %x got[0] %x\n", txt[0], bufW[0]); \ |
| } while(0) |
| |
| hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP, |
| 0, 0, 200, 60, 0, 0, 0, 0); |
| ok(hwnd != 0, "CreateWindowExA error %u\n", GetLastError()); |
| |
| ret = IsWindowUnicode(hwnd); |
| if (is_win9x) |
| ok(!ret, "RichEdit20W should NOT be unicode under Win9x\n"); |
| else |
| ok(ret, "RichEdit20W should be unicode under NT\n"); |
| |
| memset(bufA, 0xAA, sizeof(bufA)); |
| ret = SendMessageA(hwnd, WM_GETTEXT, 64, (LPARAM)bufA); |
| ok(!ret, "empty richedit should return 0, got %d\n", ret); |
| ok(!*bufA, "empty richedit should return empty string, got %s\n", bufA); |
| |
| ret = SendMessageA(hwnd, WM_CHAR, (WPARAM)textW[0], 0); |
| ok(!ret, "SendMessageA(WM_CHAR) should return 0, got %d\n", ret); |
| expect_textA(hwnd, "t"); |
| |
| ret = SendMessageA(hwnd, WM_CHAR, (WPARAM)textA[1], 0); |
| ok(!ret, "SendMessageA(WM_CHAR) should return 0, got %d\n", ret); |
| expect_textA(hwnd, "te"); |
| |
| set_textA(hwnd, NULL); |
| memset(bufA, 0xAA, sizeof(bufA)); |
| ret = SendMessageA(hwnd, WM_GETTEXT, 64, (LPARAM)bufA); |
| ok(!ret, "empty richedit should return 0, got %d\n", ret); |
| ok(!*bufA, "empty richedit should return empty string, got %s\n", bufA); |
| |
| if (is_win9x) |
| set_textA(hwnd, textW); |
| else |
| set_textA(hwnd, textA); |
| expect_textA(hwnd, textA); |
| |
| if (!is_win9x) |
| { |
| set_textW(hwnd, textW); |
| expect_textW(hwnd, textW); |
| } |
| DestroyWindow(hwnd); |
| |
| hwnd = CreateWindowExA(0, "RichEdit20A", NULL, WS_POPUP, |
| 0, 0, 200, 60, 0, 0, 0, 0); |
| ok(hwnd != 0, "CreateWindowExA error %u\n", GetLastError()); |
| |
| ret = IsWindowUnicode(hwnd); |
| ok(!ret, "RichEdit20A should NOT be unicode\n"); |
| |
| set_textA(hwnd, textA); |
| expect_textA(hwnd, textA); |
| |
| if (!is_win9x) |
| { |
| set_textW(hwnd, textW); |
| expect_textW(hwnd, textW); |
| } |
| DestroyWindow(hwnd); |
| } |
| |
| 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_GETLINE(); |
| test_EM_SCROLLCARET(); |
| test_EM_SCROLL(); |
| test_EM_SETTEXTMODE(); |
| test_TM_PLAINTEXT(); |
| test_EM_SETOPTIONS(); |
| test_WM_GETTEXT(); |
| test_EM_AUTOURLDETECT(); |
| test_EM_SETUNDOLIMIT(); |
| test_ES_PASSWORD(); |
| test_EM_SETTEXTEX(); |
| test_EM_LIMITTEXT(); |
| test_EM_EXLIMITTEXT(); |
| test_EM_GETLIMITTEXT(); |
| test_WM_SETFONT(); |
| test_EM_GETMODIFY(); |
| test_EM_EXSETSEL(); |
| test_WM_PASTE(); |
| test_EM_StreamIn_Undo(); |
| test_unicode_conversions(); |
| |
| /* 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); |
| } |
| } |
| } |
| |
| OleFlushClipboard(); |
| ok(FreeLibrary(hmoduleRichEdit) != 0, "error: %d\n", (int) GetLastError()); |
| } |