*/
#include <stdarg.h>
+#include <stdio.h>
#include <assert.h>
#include <windef.h>
#include <winbase.h>
#include <time.h>
#include <wine/test.h>
+#define ID_RICHEDITTESTDBUTTON 0x123
+
+static CHAR string1[MAX_PATH], string2[MAX_PATH], string3[MAX_PATH];
+
+#define ok_w3(format, szString1, szString2, szString3) \
+ WideCharToMultiByte(CP_ACP, 0, szString1, -1, string1, MAX_PATH, NULL, NULL); \
+ WideCharToMultiByte(CP_ACP, 0, szString2, -1, string2, MAX_PATH, NULL, NULL); \
+ WideCharToMultiByte(CP_ACP, 0, szString3, -1, string3, MAX_PATH, NULL, NULL); \
+ ok(!lstrcmpW(szString3, szString1) || !lstrcmpW(szString3, szString2), \
+ format, string1, string2, string3);
+
static HMODULE hmoduleRichEdit;
+static int is_win9x = 0;
+
static HWND new_window(LPCTSTR lpClassName, DWORD dwStyle, HWND parent) {
HWND hwnd;
hwnd = CreateWindow(lpClassName, NULL, dwStyle|WS_POPUP|WS_HSCROLL|WS_VSCROLL
return new_window(RICHEDIT_CLASS, ES_MULTILINE, parent);
}
+/* Keeps the window reponsive for the deley_time in seconds.
+ * This is useful for debugging a test to see what is happening. */
+static void keep_responsive(time_t delay_time)
+{
+ MSG msg;
+ time_t end;
+
+ /* The message pump uses PeekMessage() to empty the queue and then
+ * sleeps for 50ms before retrying the queue. */
+ end = time(NULL) + delay_time;
+ while (time(NULL) < end) {
+ if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
+ TranslateMessage(&msg);
+ DispatchMessage(&msg);
+ } else {
+ Sleep(50);
+ }
+ }
+}
+
static void processPendingMessages(void)
{
MSG msg;
}
}
+static BOOL hold_key(int vk)
+{
+ BYTE key_state[256];
+ BOOL result;
+
+ result = GetKeyboardState(key_state);
+ ok(result, "GetKeyboardState failed.\n");
+ if (!result) return FALSE;
+ key_state[vk] |= 0x80;
+ result = SetKeyboardState(key_state);
+ ok(result, "SetKeyboardState failed.\n");
+ return result != 0;
+}
+
+static BOOL release_key(int vk)
+{
+ BYTE key_state[256];
+ BOOL result;
+
+ result = GetKeyboardState(key_state);
+ ok(result, "GetKeyboardState failed.\n");
+ if (!result) return FALSE;
+ key_state[vk] &= ~0x80;
+ result = SetKeyboardState(key_state);
+ ok(result, "SetKeyboardState failed.\n");
+ return result != 0;
+}
+
static const char haystack[] = "WINEWine wineWine wine WineWine";
/* ^0 ^10 ^20 ^30 */
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}
+ {0, -1, "foo", FR_DOWN, -1},
+ {0, -1, "foo", 0, -1},
+ {0, -1, "", FR_DOWN, -1},
+ {20, 5, "foo", FR_DOWN, -1},
+ {5, 20, "foo", FR_DOWN, -1}
};
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},
+ {0, -1, "foo", FR_DOWN | FR_MATCHCASE, -1},
+ {5, 20, "WINE", FR_DOWN | FR_MATCHCASE, -1},
/* 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},
+ {0, -1, "Wine", FR_DOWN | FR_MATCHCASE, 4},
+ {5, 31, "Wine", FR_DOWN | FR_MATCHCASE, 13},
+ {14, 31, "Wine", FR_DOWN | FR_MATCHCASE, 23},
+ {24, 31, "Wine", FR_DOWN | FR_MATCHCASE, 27},
/* Find backwards */
- {19, 20, "Wine", FR_MATCHCASE, 13, 0},
- {10, 20, "Wine", FR_MATCHCASE, 4, 0},
- {20, 10, "Wine", FR_MATCHCASE, 13, 0},
+ {19, 20, "Wine", FR_MATCHCASE, 13},
+ {10, 20, "Wine", FR_MATCHCASE, 4},
+ {20, 10, "Wine", FR_MATCHCASE, 13},
/* Case-insensitive */
- {1, 31, "wInE", FR_DOWN, 4, 0},
- {1, 31, "Wine", FR_DOWN, 4, 0},
+ {1, 31, "wInE", FR_DOWN, 4},
+ {1, 31, "Wine", FR_DOWN, 4},
/* 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},
+ {20, 5, "Wine", FR_DOWN, -1},
+ {2, 1, "Wine", FR_DOWN, -1},
+ {30, 29, "Wine", FR_DOWN, -1},
+ {20, 5, "Wine", 0, 13},
/* 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},
+ {5, 10, "", FR_DOWN, -1},
+ {10, 5, "", FR_DOWN, -1},
+ {0, -1, "", FR_DOWN, -1},
+ {10, 5, "", 0, -1},
/* 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},
+ {0, -1, "wine", FR_DOWN | FR_WHOLEWORD, 18},
+ {0, -1, "win", FR_DOWN | FR_WHOLEWORD, -1},
+ {13, -1, "wine", FR_DOWN | FR_WHOLEWORD, 18},
+ {0, -1, "winewine", FR_DOWN | FR_WHOLEWORD, 0},
+ {10, -1, "winewine", FR_DOWN | FR_WHOLEWORD, 23},
+ {11, -1, "winewine", FR_WHOLEWORD, 0},
+ {31, -1, "winewine", FR_WHOLEWORD, 23},
/* 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},
+ {5, 200, "XXX", FR_DOWN, -1},
+ {-20, 20, "Wine", FR_DOWN, -1},
+ {-20, 20, "Wine", FR_DOWN, -1},
+ {-15, -20, "Wine", FR_DOWN, -1},
+ {1<<12, 1<<13, "Wine", FR_DOWN, -1},
/* 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},
+ {23, 31, "Wine", FR_DOWN | FR_MATCHCASE, 23},
+ {27, 31, "Wine", FR_DOWN | FR_MATCHCASE, 27},
+ {27, 32, "Wine", FR_DOWN | FR_MATCHCASE, 27},
+ {13, 31, "WineWine", FR_DOWN | FR_MATCHCASE, 23},
+ {13, 32, "WineWine", FR_DOWN | FR_MATCHCASE, 23},
/* 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},
+ {19, 20, "WINE", FR_MATCHCASE, 0},
+ {0, 20, "WINE", FR_MATCHCASE, -1},
- {0, -1, "wineWine wine", 0, -1, 0},
+ {0, -1, "wineWine wine", 0, -1},
};
static void check_EM_FINDTEXT(HWND hwnd, const char *name, struct find_s *f, int id) {
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);
- }
+ check_EM_FINDTEXT(hwnd, name, &find[i], i);
+ check_EM_FINDTEXTEX(hwnd, name, &find[i], i);
}
}
/* Setting a format on an arbitrary range should have no effect in search
results. This tests correct offset reporting across runs. */
cf2.cbSize = sizeof(CHARFORMAT2);
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT,
- (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM)&cf2);
cf2.dwMask = CFM_ITALIC | cf2.dwMask;
cf2.dwEffects = CFE_ITALIC ^ cf2.dwEffects;
SendMessage(hwndRichEdit, EM_SETSEL, 6, 20);
{
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);
+ int expected_bytes_written = min(gl[i].buffer_len, strlen(gl[i].text));
memset(dest, 0xBB, nBuf);
*(WORD *) dest = gl[i].buffer_len;
!strncmp(dest+2, origdest+2, nBuf-2), "buffer_len=1\n");
else
{
+ /* Prepare hex strings of buffers to dump on failure. */
+ char expectedbuf[1024];
+ char resultbuf[1024];
+ int j;
+ resultbuf[0] = '\0';
+ for (j = 0; j < 32; j++)
+ sprintf(resultbuf+strlen(resultbuf), "%02x", dest[j] & 0xFF);
+ expectedbuf[0] = '\0';
+ for (j = 0; j < expected_bytes_written; j++) /* Written bytes */
+ sprintf(expectedbuf+strlen(expectedbuf), "%02x", gl[i].text[j] & 0xFF);
+ for (; j < gl[i].buffer_len; j++) /* Ignored bytes */
+ sprintf(expectedbuf+strlen(expectedbuf), "??");
+ for (; j < 32; j++) /* Bytes after declared buffer size */
+ sprintf(expectedbuf+strlen(expectedbuf), "%02x", origdest[j] & 0xFF);
+
+ /* Test the part of the buffer that is expected to be written according
+ * to the MSDN documentation fo EM_GETLINE, which does not state that
+ * a NULL terminating character will be added unless no text is copied.
+ *
+ * Windows 95, 98 & NT do not append a NULL terminating character, but
+ * Windows 2000 and up do append a NULL terminating character if there
+ * is space in the buffer. The test will ignore this difference. */
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);
+ "%d: expected_bytes_written=%d\n" "expected=0x%s\n" "but got= 0x%s\n",
+ i, expected_bytes_written, expectedbuf, resultbuf);
+ /* Test the part of the buffer after the declared length to make sure
+ * there are no buffer overruns. */
+ ok(!strncmp(dest + gl[i].buffer_len, origdest + gl[i].buffer_len,
+ nBuf - gl[i].buffer_len),
+ "%d: expected_bytes_written=%d\n" "expected=0x%s\n" "but got= 0x%s\n",
+ i, expected_bytes_written, expectedbuf, resultbuf);
}
}
return p.y;
}
-static void move_cursor(HWND hwnd, long charindex)
+static void move_cursor(HWND hwnd, LONG charindex)
{
CHARRANGE cr;
cr.cpMax = charindex;
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"
"ff\n"
"gg\n"
"hh\n";
+ /* The richedit window height needs to be large enough vertically to fit in
+ * more than two lines of text, so the new_richedit function can't be used
+ * since a height of 60 was not large enough on some systems.
+ */
+ HWND hwndRichEdit = CreateWindow(RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwndRichEdit != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
/* Can't verify this */
SendMessage(hwndRichEdit, EM_SCROLLCARET, 0, 0);
LRESULT result;
unsigned int height = 0;
int xpos = 0;
+ POINTL pt;
static const char text[] = "aa\n"
"this is a long line of text that should be longer than the "
"control's width\n"
if (i == 0)
{
ok(HIWORD(result) == 0, "EM_POSFROMCHAR reports y=%d, expected 0\n", HIWORD(result));
- todo_wine {
ok(LOWORD(result) == 1, "EM_POSFROMCHAR reports x=%d, expected 1\n", LOWORD(result));
- }
xpos = LOWORD(result);
}
else if (i == 1)
result = SendMessage(hwndRichEdit, EM_POSFROMCHAR, 0, 0);
ok(HIWORD(result) == 0, "EM_POSFROMCHAR reports y=%d, expected 0\n", HIWORD(result));
- todo_wine {
ok(LOWORD(result) == 1, "EM_POSFROMCHAR reports x=%d, expected 1\n", LOWORD(result));
- }
xpos = LOWORD(result);
SendMessage(hwndRichEdit, WM_HSCROLL, SB_LINERIGHT, 0);
result = SendMessage(hwndRichEdit, EM_POSFROMCHAR, 0, 0);
ok(HIWORD(result) == 0, "EM_POSFROMCHAR reports y=%d, expected 0\n", HIWORD(result));
- todo_wine {
- /* Fails on builtin because horizontal scrollbar is not being shown */
ok((signed short)(LOWORD(result)) < xpos,
"EM_POSFROMCHAR reports x=%hd, expected value less than %d\n",
(signed short)(LOWORD(result)), xpos);
- }
+ SendMessage(hwndRichEdit, WM_HSCROLL, SB_LINELEFT, 0);
+
+ /* Test around end of text that doesn't end in a newline. */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "12345678901234");
+ SendMessage(hwndRichEdit, EM_POSFROMCHAR, (WPARAM)&pt,
+ SendMessage(hwndRichEdit, WM_GETTEXTLENGTH, 0, 0)-1);
+ ok(pt.x > 1, "pt.x = %d\n", pt.x);
+ xpos = pt.x;
+ SendMessage(hwndRichEdit, EM_POSFROMCHAR, (WPARAM)&pt,
+ SendMessage(hwndRichEdit, WM_GETTEXTLENGTH, 0, 0));
+ ok(pt.x > xpos, "pt.x = %d\n", pt.x);
+ xpos = pt.x;
+ SendMessage(hwndRichEdit, EM_POSFROMCHAR, (WPARAM)&pt,
+ SendMessage(hwndRichEdit, WM_GETTEXTLENGTH, 0, 0)+1);
+ ok(pt.x == xpos, "pt.x = %d\n", pt.x);
+
+ /* Try a negative position. */
+ SendMessage(hwndRichEdit, EM_POSFROMCHAR, (WPARAM)&pt, -1);
+ ok(pt.x == 1, "pt.x = %d\n", pt.x);
+
DestroyWindow(hwndRichEdit);
}
(LPARAM) &cf2);
ok(rc == 1, "EM_SETCHARFORMAT returned %d instead of 1\n", rc);
rc = SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0);
- todo_wine ok(rc == FALSE, "Should not be able to undo here.\n");
+ ok(rc == FALSE, "Should not be able to undo here.\n");
SendMessage(hwndRichEdit, EM_EMPTYUNDOBUFFER, 0, 0);
/* A valid flag, CHARFORMAT2 structure minimally filled */
(LPARAM) &cf2);
ok(rc == 1, "EM_SETCHARFORMAT returned %d instead of 1\n", rc);
rc = SendMessage(hwndRichEdit, EM_CANUNDO, 0, 0);
- todo_wine ok(rc == TRUE, "Should not be able to undo here.\n");
+ ok(rc == TRUE, "Should not be able to undo here.\n");
SendMessage(hwndRichEdit, EM_EMPTYUNDOBUFFER, 0, 0);
cf2.cbSize = sizeof(CHARFORMAT2);
cf2.dwEffects = CFE_ITALIC ^ cf2.dwEffects;
/* wParam==0 is default char format, does not set modify */
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
rc = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0);
ok(rc == 0, "Text marked as modified, expected not modified!\n");
rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, 0, (LPARAM) &cf2);
ok(rc == 0, "Text marked as modified, expected not modified!\n");
/* wParam==SCF_SELECTION sets modify if nonempty selection */
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
rc = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0);
ok(rc == 0, "Text marked as modified, expected not modified!\n");
rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM) &cf2);
ok(rc == -1, "Text not marked as modified, expected modified! (%d)\n", rc);
/* wParam==SCF_ALL sets modify regardless of whether text is present */
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
rc = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0);
ok(rc == 0, "Text marked as modified, expected not modified!\n");
rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2);
DestroyWindow(hwndRichEdit);
}
+static void test_SETPARAFORMAT(void)
+{
+ HWND hwndRichEdit = new_richedit(NULL);
+ PARAFORMAT2 fmt;
+ HRESULT ret;
+ LONG expectedMask = PFM_ALL2 & ~PFM_TABLEROWDELIMITER;
+ fmt.cbSize = sizeof(PARAFORMAT2);
+ fmt.dwMask = PFM_ALIGNMENT;
+ fmt.wAlignment = PFA_LEFT;
+
+ ret = SendMessage(hwndRichEdit, EM_SETPARAFORMAT, 0, (LPARAM) &fmt);
+ ok(ret != 0, "expected non-zero got %d\n", ret);
+
+ fmt.cbSize = sizeof(PARAFORMAT2);
+ fmt.dwMask = -1;
+ ret = SendMessage(hwndRichEdit, EM_GETPARAFORMAT, 0, (LPARAM) &fmt);
+ /* Ignore the PFM_TABLEROWDELIMITER bit because it changes
+ * between richedit different native builds of riched20.dll
+ * used on different Windows versions. */
+ ret &= ~PFM_TABLEROWDELIMITER;
+ fmt.dwMask &= ~PFM_TABLEROWDELIMITER;
+
+ ok(ret == expectedMask, "expected %x got %x\n", expectedMask, ret);
+ ok(fmt.dwMask == expectedMask, "expected %x got %x\n", expectedMask, fmt.dwMask);
+
+ DestroyWindow(hwndRichEdit);
+}
+
static void test_TM_PLAINTEXT(void)
{
/*Tests plain text properties*/
cr.cpMax = 20;
SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr);
cf2.cbSize = sizeof(CHARFORMAT2);
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT,
- (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM)&cf2);
cf2.dwMask = CFM_BOLD | cf2.dwMask;
cf2.dwEffects = CFE_BOLD ^ cf2.dwEffects;
- rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2);
+ rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&cf2);
ok(rc == 0, "EM_SETCHARFORMAT returned %d instead of 0\n", rc);
- rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_WORD | SCF_SELECTION, (LPARAM) &cf2);
+ rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_WORD | SCF_SELECTION, (LPARAM)&cf2);
ok(rc == 0, "EM_SETCHARFORMAT returned %d instead of 0\n", rc);
- rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM)&cf2);
+ rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_ALL, (LPARAM)&cf2);
ok(rc == 1, "EM_SETCHARFORMAT returned %d instead of 1\n", rc);
/*Get the formatting of those characters*/
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, 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);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&cf2test);
/*Test that they are the same as plain text allows only one formatting*/
/*Swap back to rich text*/
SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "");
- SendMessage(hwndRichEdit, EM_SETTEXTMODE, (WPARAM) TM_RICHTEXT, 0);
+ SendMessage(hwndRichEdit, EM_SETTEXTMODE, TM_RICHTEXT, 0);
/*Set the default formatting to bold italics*/
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT, (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM)&cf2);
cf2.dwMask |= CFM_ITALIC;
cf2.dwEffects ^= CFE_ITALIC;
- rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2);
+ rc = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_ALL, (LPARAM)&cf2);
ok(rc == 1, "EM_SETCHARFORMAT returned %d instead of 1\n", rc);
/*Set the text in the control to "wine", which will be bold and italicized*/
cr.cpMin = 1;
cr.cpMax = 3;
SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr);
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_SELECTION, (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, 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);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_SELECTION, (LPARAM)&cf2test);
/*Compare the two formattings. They should be the same.*/
ok(result == 7, "EM_GETTEXTRANGE returned %ld\n", result);
ok(!strcmp(expect, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+ /* cpMax of text length is used instead of -1 in this case */
+ textRange.lpstrText = buffer;
+ textRange.chrg.cpMin = 0;
+ textRange.chrg.cpMax = -1;
+ result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
+ ok(result == strlen(text2), "EM_GETTEXTRANGE returned %ld\n", result);
+ ok(!strcmp(text2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+
+ /* cpMin < 0 causes no text to be copied, and 0 to be returned */
+ textRange.lpstrText = buffer;
+ textRange.chrg.cpMin = -1;
+ textRange.chrg.cpMax = 1;
+ result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
+ ok(result == 0, "EM_GETTEXTRANGE returned %ld\n", result);
+ ok(!strcmp(text2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+
+ /* cpMax of -1 is not replaced with text length if cpMin != 0 */
+ textRange.lpstrText = buffer;
+ textRange.chrg.cpMin = 1;
+ textRange.chrg.cpMax = -1;
+ result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
+ ok(result == 0, "EM_GETTEXTRANGE returned %ld\n", result);
+ ok(!strcmp(text2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+
+ /* no end character is copied if cpMax - cpMin < 0 */
+ textRange.lpstrText = buffer;
+ textRange.chrg.cpMin = 5;
+ textRange.chrg.cpMax = 5;
+ result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
+ ok(result == 0, "EM_GETTEXTRANGE returned %ld\n", result);
+ ok(!strcmp(text2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+
+ /* cpMax of text length is used if cpMax > text length*/
+ textRange.lpstrText = buffer;
+ textRange.chrg.cpMin = 0;
+ textRange.chrg.cpMax = 1000;
+ result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
+ ok(result == strlen(text2), "EM_GETTEXTRANGE returned %ld\n", result);
+ ok(!strcmp(text2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
+
DestroyWindow(hwndRichEdit);
}
/* FIXME: need to test unimplemented options and robustly test wparam */
static void test_EM_SETOPTIONS(void)
{
- HWND hwndRichEdit = new_richedit(NULL);
+ HWND hwndRichEdit;
static const char text[] = "Hello. My name is RichEdit!";
char buffer[1024] = {0};
+ DWORD dwStyle, options, oldOptions;
+ DWORD optionStyles = ES_AUTOVSCROLL|ES_AUTOHSCROLL|ES_NOHIDESEL|
+ ES_READONLY|ES_WANTRETURN|ES_SAVESEL|
+ ES_SELECTIONBAR|ES_VERTICAL;
+
+ /* Test initial options. */
+ hwndRichEdit = CreateWindow(RICHEDIT_CLASS, NULL, WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL,
+ hmoduleRichEdit, NULL);
+ ok(hwndRichEdit != NULL, "class: %s, error: %d\n",
+ RICHEDIT_CLASS, (int) GetLastError());
+ options = SendMessage(hwndRichEdit, EM_GETOPTIONS, 0, 0);
+ ok(options == 0, "Incorrect initial options %x\n", options);
+ DestroyWindow(hwndRichEdit);
+
+ hwndRichEdit = CreateWindow(RICHEDIT_CLASS, NULL,
+ WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL,
+ hmoduleRichEdit, NULL);
+ ok(hwndRichEdit != NULL, "class: %s, error: %d\n",
+ RICHEDIT_CLASS, (int) GetLastError());
+ options = SendMessage(hwndRichEdit, EM_GETOPTIONS, 0, 0);
+ /* WS_[VH]SCROLL cause the ECO_AUTO[VH]SCROLL options to be set */
+ ok(options == (ECO_AUTOVSCROLL|ECO_AUTOHSCROLL),
+ "Incorrect initial options %x\n", options);
/* NEGATIVE TESTING - NO OPTIONS SET */
SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text);
ok(buffer[0]==text[0],
"EM_SETOPTIONS: Text changed! s1:%s s2:%s\n", text, buffer);
+ /* EM_SETOPTIONS changes the window style, but changing the
+ * window style does not change the options. */
+ dwStyle = GetWindowLong(hwndRichEdit, GWL_STYLE);
+ ok(dwStyle & ES_READONLY, "Readonly style not set by EM_SETOPTIONS\n");
+ SetWindowLong(hwndRichEdit, GWL_STYLE, dwStyle & ~ES_READONLY);
+ options = SendMessage(hwndRichEdit, EM_GETOPTIONS, 0, 0);
+ ok(options & ES_READONLY, "Readonly option set by SetWindowLong\n");
+ /* Confirm that the text is still read only. */
+ SendMessage(hwndRichEdit, WM_CHAR, 'a', ('a' << 16) | 0x0001);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ ok(buffer[0]==text[0],
+ "EM_SETOPTIONS: Text changed! s1:%s s2:%s\n", text, buffer);
+
+ oldOptions = options;
+ SetWindowLong(hwndRichEdit, GWL_STYLE, dwStyle|optionStyles);
+ options = SendMessage(hwndRichEdit, EM_GETOPTIONS, 0, 0);
+ ok(options == oldOptions,
+ "Options set by SetWindowLong (%x -> %x)\n", oldOptions, options);
+
DestroyWindow(hwndRichEdit);
}
static void test_EM_AUTOURLDETECT(void)
{
+ /* DO NOT change the properties of the first two elements. To shorten the
+ tests, all tests after WM_SETTEXT test just the first two elements -
+ one non-URL and one URL */
struct urls_s {
const char *text;
int is_url;
}
/* Test detection of URLs within normal text - WM_CHAR case. */
- for (i = 0; i < sizeof(urls)/sizeof(struct urls_s); i++) {
+ /* Test only the first two URL examples for brevity */
+ for (i = 0; i < 2; i++) {
hwndRichEdit = new_richedit(parent);
- for (j = 0; j < sizeof(templates_delim) / sizeof(const char *); j++) {
+ /* Also for brevity, test only the first three delimiters */
+ for (j = 0; j < 3; j++) {
char * at_pos;
int at_offset;
int end_offset;
}
}
SendMessage(hwndRichEdit, WM_GETTEXT, sizeof(buffer), (LPARAM)buffer);
- trace("Using template: %s\n", templates_delim[j]);
/* This assumes no templates start with the URL itself, and that they
have at least two characters before the URL text */
}
/* Test detection of URLs within normal text - EM_SETTEXTEX case. */
- for (i = 0; i < sizeof(urls)/sizeof(struct urls_s); i++) {
+ /* Test just the first two URL examples for brevity */
+ for (i = 0; i < 2; i++) {
SETTEXTEX st;
hwndRichEdit = new_richedit(parent);
}
/* Test detection of URLs within normal text - EM_REPLACESEL case. */
- for (i = 0; i < sizeof(urls)/sizeof(struct urls_s); i++) {
+ /* Test just the first two URL examples for brevity */
+ for (i = 0; i < 2; i++) {
hwndRichEdit = new_richedit(parent);
/* Set selection with X to the URL */
DestroyWindow(hwndRichEdit);
}
+unsigned int recursionLevel = 0;
+unsigned int WM_SIZE_recursionLevel = 0;
+BOOL bailedOutOfRecursion = FALSE;
+LRESULT (WINAPI *richeditProc)(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam);
+
+static LRESULT WINAPI RicheditStupidOverrideProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
+{
+ LRESULT r;
+
+ if (bailedOutOfRecursion) return 0;
+ if (recursionLevel >= 32) {
+ bailedOutOfRecursion = TRUE;
+ return 0;
+ }
+
+ recursionLevel++;
+ switch (message) {
+ case WM_SIZE:
+ WM_SIZE_recursionLevel++;
+ r = richeditProc(hwnd, message, wParam, lParam);
+ /* Because, uhhhh... I never heard of ES_DISABLENOSCROLL */
+ ShowScrollBar(hwnd, SB_VERT, TRUE);
+ WM_SIZE_recursionLevel--;
+ break;
+ default:
+ r = richeditProc(hwnd, message, wParam, lParam);
+ break;
+ }
+ recursionLevel--;
+ return r;
+}
+
+static void test_scrollbar_visibility(void)
+{
+ HWND hwndRichEdit;
+ const char * text="a\na\na\na\na\na\na\na\na\na\na\na\na\na\na\na\na\na\na\na\n";
+ SCROLLINFO si;
+ WNDCLASSA cls;
+ BOOL r;
+
+ /* These tests show that richedit should temporarily refrain from automatically
+ hiding or showing its scrollbars (vertical at least) when an explicit request
+ is made via ShowScrollBar() or similar, outside of standard richedit logic.
+ Some applications depend on forced showing (when otherwise richedit would
+ hide the vertical scrollbar) and are thrown on an endless recursive loop
+ if richedit auto-hides the scrollbar again. Apparently they never heard of
+ the ES_DISABLENOSCROLL style... */
+
+ hwndRichEdit = new_richedit(NULL);
+
+ /* Test default scrollbar visibility behavior */
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Oddly, setting text to NULL does *not* reset the scrollbar range,
+ even though it hides the scrollbar */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Setting non-scrolling text again does *not* reset scrollbar range */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+ /* Test again, with ES_DISABLENOSCROLL style */
+ hwndRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE|ES_DISABLENOSCROLL, NULL);
+
+ /* Test default scrollbar visibility behavior */
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 1,
+ "reported page/range is %d (%d..%d) expected 0 (0..1)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 1,
+ "reported page/range is %d (%d..%d) expected 0 (0..1)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Oddly, setting text to NULL does *not* reset the scrollbar range */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Setting non-scrolling text again does *not* reset scrollbar range */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax > 1,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+ /* Test behavior with explicit visibility request, using ShowScrollBar() */
+ hwndRichEdit = new_richedit(NULL);
+
+ /* Previously failed because builtin incorrectly re-hides scrollbar forced visible */
+ ShowScrollBar(hwndRichEdit, SB_VERT, TRUE);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ todo_wine {
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 100,
+ "reported page/range is %d (%d..%d) expected 0 (0..100)\n",
+ si.nPage, si.nMin, si.nMax);
+ }
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ todo_wine {
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 100,
+ "reported page/range is %d (%d..%d) expected 0 (0..100)\n",
+ si.nPage, si.nMin, si.nMax);
+ }
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ todo_wine {
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 100,
+ "reported page/range is %d (%d..%d) expected 0 (0..100)\n",
+ si.nPage, si.nMin, si.nMax);
+ }
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a\na");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ todo_wine {
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 100,
+ "reported page/range is %d (%d..%d) expected 0 (0..100)\n",
+ si.nPage, si.nMin, si.nMax);
+ }
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ todo_wine {
+ ok(si.nPage == 0 && si.nMin == 0 && si.nMax == 100,
+ "reported page/range is %d (%d..%d) expected 0 (0..100)\n",
+ si.nPage, si.nMin, si.nMax);
+ }
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+ hwndRichEdit = new_richedit(NULL);
+
+ ShowScrollBar(hwndRichEdit, SB_VERT, FALSE);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Previously, builtin incorrectly re-shows explicitly hidden scrollbar */
+ ShowScrollBar(hwndRichEdit, SB_VERT, FALSE);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Testing effect of EM_SCROLL on scrollbar visibility. It seems that
+ EM_SCROLL will make visible any forcefully invisible scrollbar */
+ SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEDOWN, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ ShowScrollBar(hwndRichEdit, SB_VERT, FALSE);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Again, EM_SCROLL, with SB_LINEUP */
+ SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEUP, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+
+ /* Test behavior with explicit visibility request, using SetWindowLong()() */
+ hwndRichEdit = new_richedit(NULL);
+
+#define ENABLE_WS_VSCROLL(hwnd) \
+ SetWindowLongA(hwnd, GWL_STYLE, GetWindowLongA(hwnd, GWL_STYLE) | WS_VSCROLL)
+#define DISABLE_WS_VSCROLL(hwnd) \
+ SetWindowLongA(hwnd, GWL_STYLE, GetWindowLongA(hwnd, GWL_STYLE) & ~WS_VSCROLL)
+
+ /* Previously failed because builtin incorrectly re-hides scrollbar forced visible */
+ ENABLE_WS_VSCROLL(hwndRichEdit);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a\na");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Ditto, see above */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+ hwndRichEdit = new_richedit(NULL);
+
+ DISABLE_WS_VSCROLL(hwndRichEdit);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)"a");
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage == 0 && si.nMin == 0 && (si.nMax == 0 || si.nMax == 100),
+ "reported page/range is %d (%d..%d) expected all 0 or nMax=100\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Previously, builtin incorrectly re-shows explicitly hidden scrollbar */
+ DISABLE_WS_VSCROLL(hwndRichEdit);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d) expected nMax/nPage nonzero\n",
+ si.nPage, si.nMin, si.nMax);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DISABLE_WS_VSCROLL(hwndRichEdit);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Testing effect of EM_SCROLL on scrollbar visibility. It seems that
+ EM_SCROLL will make visible any forcefully invisible scrollbar */
+ SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEDOWN, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DISABLE_WS_VSCROLL(hwndRichEdit);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) == 0),
+ "Vertical scrollbar is visible, should be invisible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ /* Again, EM_SCROLL, with SB_LINEUP */
+ SendMessage(hwndRichEdit, EM_SCROLL, SB_LINEUP, 0);
+ memset(&si, 0, sizeof(si));
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_PAGE | SIF_RANGE;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok (((GetWindowLongA(hwndRichEdit, GWL_STYLE) & WS_VSCROLL) != 0),
+ "Vertical scrollbar is invisible, should be visible.\n");
+ ok(si.nPage != 0 && si.nMin == 0 && si.nMax != 0,
+ "reported page/range is %d (%d..%d)\n",
+ si.nPage, si.nMin, si.nMax);
+
+ DestroyWindow(hwndRichEdit);
+
+ /* This window proc models what is going on with Corman Lisp 3.0.
+ At WM_SIZE, this proc unconditionally calls ShowScrollBar() to
+ force the scrollbar into visibility. Recursion should NOT happen
+ as a result of this action.
+ */
+ r = GetClassInfoA(NULL, RICHEDIT_CLASS, &cls);
+ if (r) {
+ richeditProc = cls.lpfnWndProc;
+ cls.lpfnWndProc = RicheditStupidOverrideProcA;
+ cls.lpszClassName = "RicheditStupidOverride";
+ if(!RegisterClassA(&cls)) assert(0);
+
+ recursionLevel = 0;
+ WM_SIZE_recursionLevel = 0;
+ bailedOutOfRecursion = FALSE;
+ hwndRichEdit = new_window(cls.lpszClassName, ES_MULTILINE, NULL);
+ ok(!bailedOutOfRecursion,
+ "WM_SIZE/scrollbar mutual recursion detected, expected none!\n");
+
+ recursionLevel = 0;
+ WM_SIZE_recursionLevel = 0;
+ bailedOutOfRecursion = FALSE;
+ MoveWindow(hwndRichEdit, 0, 0, 250, 100, TRUE);
+ ok(!bailedOutOfRecursion,
+ "WM_SIZE/scrollbar mutual recursion detected, expected none!\n");
+
+ /* Unblock window in order to process WM_DESTROY */
+ recursionLevel = 0;
+ bailedOutOfRecursion = FALSE;
+ WM_SIZE_recursionLevel = 0;
+ DestroyWindow(hwndRichEdit);
+ }
+}
+
static void test_EM_SETUNDOLIMIT(void)
{
/* cases we test for:
return 0;
}
-static void test_WM_SETTEXT()
+static void test_WM_SETTEXT(void)
{
HWND hwndRichEdit = new_richedit(NULL);
const char * TestItem1 = "TestSomeText";
const char * TestItem7 = "TestSomeText\r\n\r\r\n\rTestSomeText";
const char * TestItem7_after = "TestSomeText\r\n \r\nTestSomeText";
+ const char rtftextA[] = "{\\rtf sometext}";
+ const char urtftextA[] = "{\\urtf sometext}";
+ const WCHAR rtftextW[] = {'{','\\','r','t','f',' ','s','o','m','e','t','e','x','t','}',0};
+ const WCHAR urtftextW[] = {'{','\\','u','r','t','f',' ','s','o','m','e','t','e','x','t','}',0};
+ const WCHAR sometextW[] = {'s','o','m','e','t','e','x','t',0};
+
char buf[1024] = {0};
+ WCHAR bufW[1024] = {0};
LRESULT result;
- EDITSTREAM es;
- char * p;
/* This test attempts to show that WM_SETTEXT on a riched20 control causes
any solitary \r to be converted to \r\n on return. Properly paired
result, lstrlen(buf)); \
result = strcmp(b, buf); \
ok(result == 0, \
- "WM_SETTEXT round trip: strcmp = %ld\n", result);
+ "WM_SETTEXT round trip: strcmp = %ld, text=\"%s\"\n", result, buf);
TEST_SETTEXT(TestItem1, TestItem1)
TEST_SETTEXT(TestItem2, TestItem2_after)
TEST_SETTEXT(TestItem6, TestItem6_after)
TEST_SETTEXT(TestItem7, TestItem7_after)
- /* The following test demonstrates that WM_SETTEXT supports RTF strings */
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) TestItem1);
- p = buf;
- es.dwCookie = (DWORD_PTR)&p;
- es.dwError = 0;
- es.pfnCallback = test_WM_SETTEXT_esCallback;
- memset(buf, 0, sizeof(buf));
- SendMessage(hwndRichEdit, EM_STREAMOUT,
- (WPARAM)(SF_RTF), (LPARAM)&es);
- trace("EM_STREAMOUT produced: \n%s\n", buf);
- TEST_SETTEXT(buf, TestItem1)
-
+ /* The following tests demonstrate that WM_SETTEXT supports RTF strings */
+ TEST_SETTEXT(rtftextA, "sometext") /* interpreted as ascii rtf */
+ TEST_SETTEXT(urtftextA, "sometext") /* interpreted as ascii rtf */
+ TEST_SETTEXT(rtftextW, "{") /* interpreted as ascii text */
+ TEST_SETTEXT(urtftextW, "{") /* interpreted as ascii text */
+ DestroyWindow(hwndRichEdit);
#undef TEST_SETTEXT
+
+#define TEST_SETTEXTW(a, b) \
+ result = SendMessageW(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) a); \
+ ok (result == 1, "WM_SETTEXT returned %ld instead of 1\n", result); \
+ result = SendMessageW(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) bufW); \
+ ok (result == lstrlenW(bufW), \
+ "WM_GETTEXT returned %ld instead of expected %u\n", \
+ result, lstrlenW(bufW)); \
+ result = lstrcmpW(b, bufW); \
+ ok(result == 0, "WM_SETTEXT round trip: strcmp = %ld\n", result);
+
+ if (is_win9x)
+ {
+ skip("Cannot perform unicode tests\n");
+ return;
+ }
+hwndRichEdit = CreateWindowW(RICHEDIT_CLASS20W, NULL,
+ ES_MULTILINE|WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwndRichEdit != NULL, "class: RichEdit20W, error: %d\n", (int) GetLastError());
+ TEST_SETTEXTW(rtftextA, sometextW) /* interpreted as ascii rtf */
+ TEST_SETTEXTW(urtftextA, sometextW) /* interpreted as ascii rtf */
+ TEST_SETTEXTW(rtftextW, rtftextW) /* interpreted as ascii text */
+ TEST_SETTEXTW(urtftextW, urtftextW) /* interpreted as ascii text */
DestroyWindow(hwndRichEdit);
+#undef TEST_SETTEXTW
}
static void test_EM_STREAMOUT(void)
es.dwError = 0;
es.pfnCallback = test_WM_SETTEXT_esCallback;
memset(buf, 0, sizeof(buf));
- SendMessage(hwndRichEdit, EM_STREAMOUT,
- (WPARAM)(SF_TEXT), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMOUT, SF_TEXT, (LPARAM)&es);
r = strlen(buf);
ok(r == 12, "streamed text length is %d, expecting 12\n", r);
ok(strcmp(buf, TestItem1) == 0,
es.dwError = 0;
es.pfnCallback = test_WM_SETTEXT_esCallback;
memset(buf, 0, sizeof(buf));
- SendMessage(hwndRichEdit, EM_STREAMOUT,
- (WPARAM)(SF_TEXT), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMOUT, SF_TEXT, (LPARAM)&es);
r = strlen(buf);
/* Here again, \r gets converted to \r\n, like WM_GETTEXT */
ok(r == 14, "streamed text length is %d, expecting 14\n", r);
es.dwError = 0;
es.pfnCallback = test_WM_SETTEXT_esCallback;
memset(buf, 0, sizeof(buf));
- SendMessage(hwndRichEdit, EM_STREAMOUT,
- (WPARAM)(SF_TEXT), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMOUT, SF_TEXT, (LPARAM)&es);
r = strlen(buf);
ok(r == 14, "streamed text length is %d, expecting 14\n", r);
ok(strcmp(buf, TestItem3) == 0,
DestroyWindow(hwndRichEdit);
}
-static void test_EM_SETTEXTEX(void)
+static void test_EM_STREAMOUT_FONTTBL(void)
{
HWND hwndRichEdit = new_richedit(NULL);
+ EDITSTREAM es;
+ char buf[1024] = {0};
+ char * p;
+ char * fontTbl;
+ int brackCount;
+
+ const char * TestItem = "TestSomeText";
+
+ /* fills in the richedit control with some text */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) TestItem);
+
+ /* streams out the text in rtf format */
+ p = buf;
+ es.dwCookie = (DWORD_PTR)&p;
+ es.dwError = 0;
+ es.pfnCallback = test_WM_SETTEXT_esCallback;
+ memset(buf, 0, sizeof(buf));
+ SendMessage(hwndRichEdit, EM_STREAMOUT, SF_RTF, (LPARAM)&es);
+
+ /* scans for \fonttbl, error if not found */
+ fontTbl = strstr(buf, "\\fonttbl");
+ ok(fontTbl != NULL, "missing \\fonttbl section\n");
+ if(fontTbl)
+ {
+ /* scans for terminating closing bracket */
+ brackCount = 1;
+ while(*fontTbl && brackCount)
+ {
+ if(*fontTbl == '{')
+ brackCount++;
+ else if(*fontTbl == '}')
+ brackCount--;
+ fontTbl++;
+ }
+ /* checks whether closing bracket is ok */
+ ok(brackCount == 0, "missing closing bracket in \\fonttbl block\n");
+ if(!brackCount)
+ {
+ /* char before closing fonttbl block should be a closed bracket */
+ fontTbl -= 2;
+ ok(*fontTbl == '}', "spurious character '%02x' before \\fonttbl closing bracket\n", *fontTbl);
+
+ /* char after fonttbl block should be a crlf */
+ fontTbl += 2;
+ ok(*fontTbl == 0x0d && *(fontTbl+1) == 0x0a, "missing crlf after \\fonttbl block\n");
+ }
+ }
+ DestroyWindow(hwndRichEdit);
+}
+
+
+static void test_EM_SETTEXTEX(void)
+{
+ HWND hwndRichEdit, parent;
+ SCROLLINFO si;
+ int sel_start, sel_end;
SETTEXTEX setText;
GETTEXTEX getText;
WCHAR TestItem1[] = {'T', 'e', 's', 't',
'S', 'o', 'm', 'e',
'T', 'e', 'x', 't', 0};
+ WCHAR TestItem1alt[] = {'T', 'T', 'e', 's',
+ 't', 'S', 'o', 'm',
+ 'e', 'T', 'e', 'x',
+ 't', 't', 'S', 'o',
+ 'm', 'e', 'T', 'e',
+ 'x', 't', 0};
+ WCHAR TestItem1altn[] = {'T','T','e','s','t','S','o','m','e','T','e','x','t',
+ '\r','t','S','o','m','e','T','e','x','t',0};
WCHAR TestItem2[] = {'T', 'e', 's', 't',
'S', 'o', 'm', 'e',
'T', 'e', 'x', 't',
' ','\r', 0};
#define MAX_BUF_LEN 1024
WCHAR buf[MAX_BUF_LEN];
+ char bufACP[MAX_BUF_LEN];
char * p;
int result;
CHARRANGE cr;
EDITSTREAM es;
+ WNDCLASSA cls;
+
+ /* Test the scroll position with and without a parent window.
+ *
+ * For some reason the scroll position is 0 after EM_SETTEXTEX
+ * with the ST_SELECTION flag only when the control has a parent
+ * window, even though the selection is at the end. */
+ cls.style = 0;
+ cls.lpfnWndProc = DefWindowProcA;
+ cls.cbClsExtra = 0;
+ cls.cbWndExtra = 0;
+ cls.hInstance = GetModuleHandleA(0);
+ cls.hIcon = 0;
+ cls.hCursor = LoadCursorA(0, IDC_ARROW);
+ cls.hbrBackground = GetStockObject(WHITE_BRUSH);
+ cls.lpszMenuName = NULL;
+ cls.lpszClassName = "ParentTestClass";
+ if(!RegisterClassA(&cls)) assert(0);
+
+ parent = CreateWindow(cls.lpszClassName, NULL, WS_POPUP|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL, NULL, NULL);
+ ok (parent != 0, "Failed to create parent window\n");
+
+ hwndRichEdit = CreateWindowEx(0,
+ RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_VSCROLL|WS_VISIBLE|WS_CHILD,
+ 0, 0, 200, 60, parent, NULL,
+ hmoduleRichEdit, NULL);
+
+ setText.codepage = CP_ACP;
+ setText.flags = ST_SELECTION;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText,
+ (LPARAM)"{\\rtf 1\\par 2\\par 3\\par 4\\par 5\\par 6\\par 7\\par 8\\par 9\\par}");
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_ALL;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ todo_wine ok(si.nPos == 0, "Position is incorrectly at %d\n", si.nPos);
+ SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 18, "Selection start incorrectly at %d\n", sel_start);
+ ok(sel_end == 18, "Selection end incorrectly at %d\n", sel_end);
+
+ DestroyWindow(parent);
+
+ /* Test without a parent window */
+ hwndRichEdit = new_richedit(NULL);
+ setText.codepage = CP_ACP;
+ setText.flags = ST_SELECTION;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText,
+ (LPARAM)"{\\rtf 1\\par 2\\par 3\\par 4\\par 5\\par 6\\par 7\\par 8\\par 9\\par}");
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_ALL;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok(si.nPos != 0, "Position is incorrectly at %d\n", si.nPos);
+ SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 18, "Selection start incorrectly at %d\n", sel_start);
+ ok(sel_end == 18, "Selection end incorrectly at %d\n", sel_end);
+
+ /* The scroll position should also be 0 after EM_SETTEXTEX with ST_DEFAULT,
+ * but this time it is because the selection is at the beginning. */
+ setText.codepage = CP_ACP;
+ setText.flags = ST_DEFAULT;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText,
+ (LPARAM)"{\\rtf 1\\par 2\\par 3\\par 4\\par 5\\par 6\\par 7\\par 8\\par 9\\par}");
+ si.cbSize = sizeof(si);
+ si.fMask = SIF_ALL;
+ GetScrollInfo(hwndRichEdit, SB_VERT, &si);
+ ok(si.nPos == 0, "Position is incorrectly at %d\n", si.nPos);
+ SendMessage(hwndRichEdit, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 0, "Selection start incorrectly at %d\n", sel_start);
+ ok(sel_end == 0, "Selection end incorrectly at %d\n", sel_end);
setText.codepage = 1200; /* no constant for unicode */
getText.codepage = 1200; /* no constant for unicode */
"EM_GETTEXTEX results not what was set by EM_SETTEXTEX\n");
/* Unlike WM_SETTEXT/WM_GETTEXT pair, EM_SETTEXTEX/EM_GETTEXTEX does not
- convert \r to \r\n on return
+ convert \r to \r\n on return: !ST_SELECTION && Unicode && !\rtf
*/
setText.codepage = 1200; /* no constant for unicode */
getText.codepage = 1200; /* no constant for unicode */
"EM_GETTEXTEX results not what was set by EM_SETTEXTEX\n");
- /* \r\n pairs get changed into \r */
+ /* \r\n pairs get changed into \r: !ST_SELECTION && Unicode && !\rtf */
setText.codepage = 1200; /* no constant for unicode */
getText.codepage = 1200; /* no constant for unicode */
getText.cb = MAX_BUF_LEN;
ok(lstrcmpW(buf, TestItem3_after) == 0,
"EM_SETTEXTEX did not convert properly\n");
- /* \n also gets changed to \r */
+ /* \n also gets changed to \r: !ST_SELECTION && Unicode && !\rtf */
setText.codepage = 1200; /* no constant for unicode */
getText.codepage = 1200; /* no constant for unicode */
getText.cb = MAX_BUF_LEN;
ok(lstrcmpW(buf, TestItem3_after) == 0,
"EM_SETTEXTEX did not convert properly\n");
- /* \r\r\n gets changed into single space */
+ /* \r\r\n gets changed into single space: !ST_SELECTION && Unicode && !\rtf */
setText.codepage = 1200; /* no constant for unicode */
getText.codepage = 1200; /* no constant for unicode */
getText.cb = MAX_BUF_LEN;
ok(lstrcmpW(buf, TestItem4_after) == 0,
"EM_SETTEXTEX did not convert properly\n");
- result = SendMessage(hwndRichEdit, EM_SETTEXTEX,
- (WPARAM)&setText, (LPARAM) NULL);
+ /* !ST_SELECTION && Unicode && !\rtf */
+ result = SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, 0);
SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf);
ok (result == 1,
ok(lstrlenW(buf) == 0,
"EM_SETTEXTEX with NULL lParam should clear rich edit.\n");
- /* put some text back */
+ /* put some text back: !ST_SELECTION && Unicode && !\rtf */
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 */
+ /* replace current selection: ST_SELECTION && Unicode && !\rtf */
setText.flags = ST_SELECTION;
- result = SendMessage(hwndRichEdit, EM_SETTEXTEX,
- (WPARAM)&setText, (LPARAM) NULL);
+ result = SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, 0);
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 */
+ /* put some text back: !ST_SELECTION && Unicode && !\rtf */
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 */
+ /* replace current selection: ST_SELECTION && Unicode && !\rtf */
setText.flags = ST_SELECTION;
result = SendMessage(hwndRichEdit, EM_SETTEXTEX,
(WPARAM)&setText, (LPARAM) TestItem1);
memset(buf, 0, sizeof(buf));
SendMessage(hwndRichEdit, EM_STREAMOUT,
(WPARAM)(SF_RTF), (LPARAM)&es);
- trace("EM_STREAMOUT produced: \n%s\n", (char *)buf);
+ trace("EM_STREAMOUT produced:\n%s\n", (char *)buf);
+ /* !ST_SELECTION && !Unicode && \rtf */
setText.codepage = CP_ACP;/* EM_STREAMOUT saved as ANSI string */
getText.codepage = 1200; /* no constant for unicode */
getText.cb = MAX_BUF_LEN;
ok(lstrcmpW(buf, TestItem1) == 0,
"EM_GETTEXTEX results not what was set by EM_SETTEXTEX\n");
+ /* The following test demonstrates that EM_SETTEXTEX treats text as ASCII if it
+ * starts with ASCII characters "{\rtf" even when the codepage is unicode. */
+ setText.codepage = 1200; /* Lie about code page (actual ASCII) */
+ getText.codepage = CP_ACP;
+ getText.cb = MAX_BUF_LEN;
+ getText.flags = GT_DEFAULT;
+ getText.lpDefaultChar = NULL;
+ getText.lpUsedDefChar = NULL;
+
+ setText.flags = ST_SELECTION;
+ SendMessage(hwndRichEdit, EM_SETSEL, 0, -1);
+ result = SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) "{\\rtf not unicode}");
+ todo_wine ok(result == 11, "EM_SETTEXTEX incorrectly returned %d\n", result);
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) bufACP);
+ ok(lstrcmpA(bufACP, "not unicode") == 0, "'%s' != 'not unicode'\n", bufACP);
+
+ /* The following test demonstrates that EM_SETTEXTEX supports RTF strings with a selection */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "TestSomeText"); /* TestItem1 */
+ p = (char *)buf;
+ es.dwCookie = (DWORD_PTR)&p;
+ es.dwError = 0;
+ es.pfnCallback = test_WM_SETTEXT_esCallback;
+ memset(buf, 0, sizeof(buf));
+ SendMessage(hwndRichEdit, EM_STREAMOUT,
+ (WPARAM)(SF_RTF), (LPARAM)&es);
+ trace("EM_STREAMOUT produced:\n%s\n", (char *)buf);
+
+ /* select some text */
+ cr.cpMax = 1;
+ cr.cpMin = 3;
+ SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr);
+
+ /* ST_SELECTION && !Unicode && \rtf */
+ setText.codepage = CP_ACP;/* EM_STREAMOUT saved as ANSI string */
+ getText.codepage = 1200; /* no constant for unicode */
+ getText.cb = MAX_BUF_LEN;
+ getText.flags = GT_DEFAULT;
+ getText.lpDefaultChar = NULL;
+ getText.lpUsedDefChar = NULL;
+
+ setText.flags = ST_SELECTION;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) buf);
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf);
+ ok_w3("Expected \"%s\" or \"%s\", got \"%s\"\n", TestItem1alt, TestItem1altn, buf);
+
+ /* The following test demonstrates that EM_SETTEXTEX replacing a selection */
+ setText.codepage = 1200; /* no constant for unicode */
+ getText.codepage = CP_ACP;
+ getText.cb = MAX_BUF_LEN;
+
+ setText.flags = 0;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) TestItem1); /* TestItem1 */
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) bufACP);
+
+ /* select some text */
+ cr.cpMax = 1;
+ cr.cpMin = 3;
+ SendMessage(hwndRichEdit, EM_EXSETSEL, 0, (LPARAM) &cr);
+
+ /* ST_SELECTION && !Unicode && !\rtf */
+ setText.codepage = CP_ACP;
+ getText.codepage = 1200; /* no constant for unicode */
+ getText.cb = MAX_BUF_LEN;
+ getText.flags = GT_DEFAULT;
+ getText.lpDefaultChar = NULL;
+ getText.lpUsedDefChar = NULL;
+
+ setText.flags = ST_SELECTION;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM) bufACP);
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) buf);
+ ok(lstrcmpW(buf, TestItem1alt) == 0,
+ "EM_GETTEXTEX results not what was set by EM_SETTEXTEX when"
+ " using ST_SELECTION and non-Unicode\n");
+
+ /* Test setting text using rich text format */
+ setText.flags = 0;
+ setText.codepage = CP_ACP;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM)"{\\rtf richtext}");
+ getText.codepage = CP_ACP;
+ getText.cb = MAX_BUF_LEN;
+ getText.flags = GT_DEFAULT;
+ getText.lpDefaultChar = NULL;
+ getText.lpUsedDefChar = NULL;
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) bufACP);
+ ok(!strcmp(bufACP, "richtext"), "expected 'richtext' but got '%s'\n", bufACP);
+
+ setText.flags = 0;
+ setText.codepage = CP_ACP;
+ SendMessage(hwndRichEdit, EM_SETTEXTEX, (WPARAM)&setText, (LPARAM)"{\\urtf morerichtext}");
+ getText.codepage = CP_ACP;
+ getText.cb = MAX_BUF_LEN;
+ getText.flags = GT_DEFAULT;
+ getText.lpDefaultChar = NULL;
+ getText.lpUsedDefChar = NULL;
+ SendMessage(hwndRichEdit, EM_GETTEXTEX, (WPARAM)&getText, (LPARAM) bufACP);
+ ok(!strcmp(bufACP, "morerichtext"), "expected 'morerichtext' but got '%s'\n", bufACP);
DestroyWindow(hwndRichEdit);
}
returnedCF2A.cbSize = sizeof(returnedCF2A);
SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "x");
- SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont1,(LPARAM) MAKELONG((WORD) TRUE, 0));
+ SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont1, MAKELPARAM(TRUE, 0));
SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A);
GetObjectA(testFont1, sizeof(LOGFONTA), &sentLogFont);
"EM_GETCHARFORMAT: 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, WM_SETFONT, (WPARAM)testFont2, MAKELPARAM(TRUE, 0));
SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A);
GetObjectA(testFont2, sizeof(LOGFONTA), &sentLogFont);
ok (!strcmp(sentLogFont.lfFaceName,returnedCF2A.szFaceName),
"EM_GETCHARFORMAT: 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, WM_SETFONT, (WPARAM)testFont3, MAKELPARAM(TRUE, 0));
SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A);
GetObjectA(testFont3, sizeof(LOGFONTA), &sentLogFont);
ok (!strcmp(sentLogFont.lfFaceName,returnedCF2A.szFaceName),
ZeroMemory(&sentLogFont,sizeof(sentLogFont));
returnedCF2A.cbSize = sizeof(returnedCF2A);
- SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)NULL,(LPARAM) MAKELONG((WORD) TRUE, 0));
+ SendMessage(hwndRichEdit, WM_SETFONT, 0, MAKELPARAM((WORD) TRUE, 0));
SendMessage(hwndRichEdit, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM) &returnedCF2A);
GetObjectA(NULL, sizeof(LOGFONTA), &sentLogFont);
ok (!strcmp("System",returnedCF2A.szFaceName),
/* setting font doesn't change modify flag */
SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0);
- SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont,(LPARAM) MAKELONG((WORD) TRUE, 0));
+ SendMessage(hwndRichEdit, WM_SETFONT, (WPARAM)testFont, MAKELPARAM(TRUE, 0));
result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0);
ok (result == 0,
"EM_GETMODIFY returned non-zero, instead of zero on setting font\n");
/* set char format */
SendMessage(hwndRichEdit, EM_SETMODIFY, FALSE, 0);
cf2.cbSize = sizeof(CHARFORMAT2);
- SendMessage(hwndRichEdit, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT,
- (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_GETCHARFORMAT, 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_SETCHARFORMAT, (WPARAM) SCF_ALL, (LPARAM) &cf2);
+ SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_ALL, (LPARAM)&cf2);
+ result = SendMessage(hwndRichEdit, EM_SETCHARFORMAT, SCF_ALL, (LPARAM)&cf2);
ok(result == 1, "EM_SETCHARFORMAT returned %ld instead of 1\n", result);
result = SendMessage(hwndRichEdit, EM_GETMODIFY, 0, 0);
ok (result != 0,
es.dwCookie = (DWORD_PTR)&streamText;
es.dwError = 0;
es.pfnCallback = test_EM_GETMODIFY_esCallback;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_TEXT), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, 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");
}
struct exsetsel_s {
- long min;
- long max;
- long expected_retval;
+ LONG min;
+ LONG max;
+ LRESULT 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},
+ {5, 10, 10, 5, 10, 0},
+ {15, 17, 17, 15, 17, 0},
/* test cpMax > strlen() */
- {0, 100, 18, 0, 18, 0, 1},
+ {0, 100, 18, 0, 18, 1},
/* test cpMin == cpMax */
- {5, 5, 5, 5, 5, 0, 0},
+ {5, 5, 5, 5, 5, 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},
+ {-1, 0, 5, 5, 5, 0},
+ {-1, 17, 5, 5, 5, 0},
+ {-1, 18, 5, 5, 5, 0},
/* test cpMin < 0 && cpMax < 0 */
- {-1, -1, 17, 17, 17, 0, 0},
- {-4, -5, 17, 17, 17, 0, 0},
+ {-1, -1, 17, 17, 17, 0},
+ {-4, -5, 17, 17, 17, 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},
+ {0, -1, 18, 0, 18, 1},
+ {17, -5, 18, 17, 18, 1},
+ {18, -3, 17, 17, 17, 0},
/* test if cpMin > cpMax */
- {15, 19, 18, 15, 18, 0, 1},
- {19, 15, 18, 15, 18, 0, 1}
+ {15, 19, 18, 15, 18, 1},
+ {19, 15, 18, 15, 18, 1}
};
static void check_EM_EXSETSEL(HWND hwnd, const struct exsetsel_s *setsel, int id) {
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);
- }
+ 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);
/* FIXME add more tests */
SendMessage(hwndRichEdit, EM_SETSEL, 7, 17);
- r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) NULL);
+ r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, 0);
ok(0 == r, "EM_REPLACESEL returned %d, expected 0\n", r);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
r = strcmp(buffer, "testing");
SendMessage(hwndRichEdit, WM_SETREDRAW, redraw, 0);
/* Test behavior with carriage returns and newlines */
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "RichEdit1");
ok(9 == r, "EM_REPLACESEL returned %d, expected 9\n", r);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 1, "EM_GETLINECOUNT returned %d, expected 1\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "RichEdit1\r");
ok(10 == r, "EM_REPLACESEL returned %d, expected 10\n", r);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
characters interpreted from the original lParam. Wine's builtin riched20
implements the WinXP behavior.
*/
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "RichEdit1\r\n");
ok(11 == r /* WinXP */ || 10 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 11 or 10\n", r);
string.
*/
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\r\r");
ok(2 == r, "EM_REPLACESEL returned %d, expected 4\n", r);
r = SendMessage(hwndRichEdit, EM_EXGETSEL, 0, (LPARAM)&cr);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 3, "EM_GETLINECOUNT returned %d, expected 3\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\r\r\n");
ok(3 == r /* WinXP */ || 1 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 3 or 1\n", r);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 1, "EM_GETLINECOUNT returned %d, expected 1\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\r\r\r\r\r\n\r\r\r");
ok(9 == r /* WinXP */ || 7 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 9 or 7\n", r);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 7, "EM_GETLINECOUNT returned %d, expected 7\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\r\r\n\r\n");
ok(5 == r /* WinXP */ || 2 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 5 or 2\n", r);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 2, "EM_GETLINECOUNT returned %d, expected 2\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\r\r\n\r\r");
ok(5 == r /* WinXP */ || 3 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 5 or 3\n", r);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 3, "EM_GETLINECOUNT returned %d, expected 3\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\rX\r\n\r\r");
ok(6 == r /* WinXP */ || 5 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 6 or 5\n", r);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 5, "EM_GETLINECOUNT returned %d, expected 5\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\n\n");
ok(2 == r, "EM_REPLACESEL returned %d, expected 2\n", r);
r = SendMessage(hwndRichEdit, EM_EXGETSEL, 0, (LPARAM)&cr);
r = SendMessage(hwndRichEdit, EM_GETLINECOUNT, 0, 0);
ok(r == 3, "EM_GETLINECOUNT returned %d, expected 3\n", r);
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)NULL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
r = SendMessage(hwndRichEdit, EM_REPLACESEL, 0, (LPARAM) "\n\n\n\n\r\r\r\r\n");
ok(9 == r /* WinXP */ || 7 == r /* Win98 */,
"EM_REPLACESEL returned %d, expected 9 or 7\n", r);
const char* text3 = "testing paste\r\npaste\r\ntesting paste";
HWND hwndRichEdit = new_richedit(NULL);
- /* Native riched20 won't obey WM_CHAR messages or WM_KEYDOWN/WM_KEYUP
- messages, probably because it inspects the keyboard state itself.
- Therefore, native requires this in order to obey Ctrl-<key> keystrokes.
+ /* Native riched20 inspects the keyboard state (e.g. GetKeyState)
+ * to test the state of the modifiers (Ctrl/Alt/Shift).
+ *
+ * Therefore Ctrl-<key> keystrokes need to be simulated with
+ * keybd_event or by using SetKeyboardState to set the modifiers
+ * and SendMessage to simulate the keystrokes.
*/
+ /* Sent keystrokes with keybd_event */
#define SEND_CTRL_C(hwnd) pressKeyWithModifier(hwnd, VK_CONTROL, 'C')
#define SEND_CTRL_X(hwnd) pressKeyWithModifier(hwnd, VK_CONTROL, 'X')
#define SEND_CTRL_V(hwnd) pressKeyWithModifier(hwnd, VK_CONTROL, 'V')
/* Pasted text should be visible at this step */
result = strcmp(text1_step1, buffer);
ok(result == 0,
- "test paste: strcmp = %i\n", result);
+ "test paste: strcmp = %i, text='%s'\n", result, buffer);
+
SEND_CTRL_Z(hwndRichEdit); /* Undo */
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
/* Text should be the same as before (except for \r -> \r\n conversion) */
result = strcmp(text1_after, buffer);
ok(result == 0,
- "test paste: strcmp = %i\n", result);
+ "test paste: strcmp = %i, text='%s'\n", result, buffer);
SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text2);
SendMessage(hwndRichEdit, EM_SETSEL, 8, 13);
ok(result == 0,
"test paste: strcmp = %i\n", result);
+#undef SEND_CTRL_C
+#undef SEND_CTRL_X
+#undef SEND_CTRL_V
+#undef SEND_CTRL_Z
+#undef SEND_CTRL_Y
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ /* Send WM_CHAR to simulates Ctrl-V */
+ SendMessage(hwndRichEdit, WM_CHAR, 22,
+ (MapVirtualKey('V', MAPVK_VK_TO_VSC) << 16) & 1);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ /* Shouldn't paste because pasting is handled by WM_KEYDOWN */
+ result = strcmp(buffer,"");
+ ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+
+ /* Send keystrokes with WM_KEYDOWN after setting the modifiers
+ * with SetKeyboard state. */
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ /* Simulates paste (Ctrl-V) */
+ hold_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'V',
+ (MapVirtualKey('V', MAPVK_VK_TO_VSC) << 16) & 1);
+ release_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"paste");
+ ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) text1);
+ SendMessage(hwndRichEdit, EM_SETSEL, 0, 7);
+ /* Simulates copy (Ctrl-C) */
+ hold_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'C',
+ (MapVirtualKey('C', MAPVK_VK_TO_VSC) << 16) & 1);
+ release_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ SendMessage(hwndRichEdit, WM_PASTE, 0, 0);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"testing");
+ ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+
+ /* Cut with WM_KEYDOWN to simulate Ctrl-X */
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) "cut");
+ /* Simulates select all (Ctrl-A) */
+ hold_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'A',
+ (MapVirtualKey('A', MAPVK_VK_TO_VSC) << 16) & 1);
+ /* Simulates select cut (Ctrl-X) */
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'X',
+ (MapVirtualKey('X', MAPVK_VK_TO_VSC) << 16) & 1);
+ release_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"");
+ ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, 0);
+ SendMessage(hwndRichEdit, WM_PASTE, 0, 0);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"cut\r\n");
+ todo_wine ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+ /* Simulates undo (Ctrl-Z) */
+ hold_key(VK_CONTROL);
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'Z',
+ (MapVirtualKey('Z', MAPVK_VK_TO_VSC) << 16) & 1);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"");
+ ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+ /* Simulates redo (Ctrl-Y) */
+ SendMessage(hwndRichEdit, WM_KEYDOWN, 'Y',
+ (MapVirtualKey('Y', MAPVK_VK_TO_VSC) << 16) & 1);
+ SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
+ result = strcmp(buffer,"cut\r\n");
+ todo_wine ok(result == 0,
+ "test paste: strcmp = %i, actual = '%s'\n", result, buffer);
+ release_key(VK_CONTROL);
+
DestroyWindow(hwndRichEdit);
}
static void test_EM_FORMATRANGE(void)
{
- int r;
- FORMATRANGE fr;
+ int i, tpp_x, tpp_y;
HDC hdc;
HWND hwndRichEdit = new_richedit(NULL);
-
- SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) haystack);
+ static const struct {
+ const char *string; /* The string */
+ int first; /* First 'pagebreak', 0 for don't care */
+ int second; /* Second 'pagebreak', 0 for don't care */
+ } fmtstrings[] = {
+ {"WINE wine", 0, 0},
+ {"WINE wineWine", 0, 0},
+ {"WINE\r\nwine\r\nwine", 5, 10},
+ {"WINE\r\nWINEwine\r\nWINEwine", 5, 14},
+ {"WINE\r\n\r\nwine\r\nwine", 5, 6}
+ };
hdc = GetDC(hwndRichEdit);
ok(hdc != NULL, "Could not get HDC\n");
- fr.hdc = fr.hdcTarget = hdc;
- fr.rc.top = fr.rcPage.top = fr.rc.left = fr.rcPage.left = 0;
- fr.rc.right = fr.rcPage.right = GetDeviceCaps(hdc, HORZRES);
- fr.rc.bottom = fr.rcPage.bottom = GetDeviceCaps(hdc, VERTRES);
- fr.chrg.cpMin = 0;
- fr.chrg.cpMax = 20;
+ /* Calculate the twips per pixel */
+ tpp_x = 1440 / GetDeviceCaps(hdc, LOGPIXELSX);
+ tpp_y = 1440 / GetDeviceCaps(hdc, LOGPIXELSY);
- r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) NULL);
- todo_wine {
- ok(r == 31, "EM_FORMATRANGE expect %d, got %d\n", 31, r);
- }
+ SendMessage(hwndRichEdit, EM_FORMATRANGE, FALSE, 0);
- r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) &fr);
- todo_wine {
- ok(r == 20, "EM_FORMATRANGE expect %d, got %d\n", 20, r);
- }
+ for (i = 0; i < sizeof(fmtstrings)/sizeof(fmtstrings[0]); i++)
+ {
+ FORMATRANGE fr;
+ GETTEXTLENGTHEX gtl;
+ SIZE stringsize;
+ int r, len;
- fr.chrg.cpMin = 0;
- fr.chrg.cpMax = 10;
+ SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM) fmtstrings[i].string);
- r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) &fr);
- todo_wine {
- ok(r == 10, "EM_FORMATRANGE expect %d, got %d\n", 10, r);
- }
+ gtl.flags = GTL_NUMCHARS | GTL_PRECISE;
+ gtl.codepage = CP_ACP;
+ len = SendMessageA(hwndRichEdit, EM_GETTEXTLENGTHEX, (WPARAM)>l, 0);
- r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) NULL);
- todo_wine {
- ok(r == 31, "EM_FORMATRANGE expect %d, got %d\n", 31, r);
+ /* Get some size information for the string */
+ GetTextExtentPoint32(hdc, fmtstrings[i].string, strlen(fmtstrings[i].string), &stringsize);
+
+ /* Define the box to be half the width needed and a bit larger than the height.
+ * Changes to the width means we have at least 2 pages. Changes to the height
+ * is done so we can check the changing of fr.rc.bottom.
+ */
+ fr.hdc = fr.hdcTarget = hdc;
+ fr.rc.top = fr.rcPage.top = fr.rc.left = fr.rcPage.left = 0;
+ fr.rc.right = fr.rcPage.right = (stringsize.cx / 2) * tpp_x;
+ fr.rc.bottom = fr.rcPage.bottom = (stringsize.cy + 10) * tpp_y;
+
+ r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, 0);
+ todo_wine {
+ ok(r == len, "Expected %d, got %d\n", len, r);
+ }
+
+ /* We know that the page can't hold the full string. See how many characters
+ * are on the first one
+ */
+ fr.chrg.cpMin = 0;
+ fr.chrg.cpMax = -1;
+ r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) &fr);
+ todo_wine {
+ ok(fr.rc.bottom == (stringsize.cy * tpp_y), "Expected bottom to be %d, got %d\n", (stringsize.cy * tpp_y), fr.rc.bottom);
+ }
+ if (fmtstrings[i].first)
+ todo_wine {
+ ok(r == fmtstrings[i].first, "Expected %d, got %d\n", fmtstrings[i].first, r);
+ }
+ else
+ ok(r < len, "Expected < %d, got %d\n", len, r);
+
+ /* Do another page */
+ fr.chrg.cpMin = r;
+ r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, (LPARAM) &fr);
+ if (fmtstrings[i].second)
+ todo_wine {
+ ok(r == fmtstrings[i].second, "Expected %d, got %d\n", fmtstrings[i].second, r);
+ }
+ else
+ ok (r < len, "Expected < %d, got %d\n", len, r);
+
+ /* There is at least on more page, but we don't care */
+
+ r = SendMessage(hwndRichEdit, EM_FORMATRANGE, TRUE, 0);
+ todo_wine {
+ ok(r == len, "Expected %d, got %d\n", len, r);
+ }
}
+ ReleaseDC(NULL, hdc);
DestroyWindow(hwndRichEdit);
}
const char * streamText0b = "{\\rtf1 TestSomeText\\par\\par}";
const char * streamText1 =
- "{\\rtf1\\ansi\\ansicpg1252\\deff0\\deflang12298{\\fonttbl{\\f0\\fswiss\\fprq2\\fcharset0 System;}}\r\n" \
- "\\viewkind4\\uc1\\pard\\f0\\fs17 TestSomeText\\par\r\n" \
+ "{\\rtf1\\ansi\\ansicpg1252\\deff0\\deflang12298{\\fonttbl{\\f0\\fswiss\\fprq2\\fcharset0 System;}}\r\n"
+ "\\viewkind4\\uc1\\pard\\f0\\fs17 TestSomeText\\par\r\n"
"}\r\n";
/* In richedit 2.0 mode, this should NOT be accepted, unlike 1.0 */
const char * streamText2 =
- "{{\\colortbl;\\red0\\green255\\blue102;\\red255\\green255\\blue255;" \
- "\\red170\\green255\\blue255;\\red255\\green238\\blue0;\\red51\\green255" \
- "\\blue221;\\red238\\green238\\blue238;}\\tx0 \\tx424 \\tx848 \\tx1272 " \
- "\\tx1696 \\tx2120 \\tx2544 \\tx2968 \\tx3392 \\tx3816 \\tx4240 \\tx4664 " \
- "\\tx5088 \\tx5512 \\tx5936 \\tx6360 \\tx6784 \\tx7208 \\tx7632 \\tx8056 " \
- "\\tx8480 \\tx8904 \\tx9328 \\tx9752 \\tx10176 \\tx10600 \\tx11024 " \
+ "{{\\colortbl;\\red0\\green255\\blue102;\\red255\\green255\\blue255;"
+ "\\red170\\green255\\blue255;\\red255\\green238\\blue0;\\red51\\green255"
+ "\\blue221;\\red238\\green238\\blue238;}\\tx0 \\tx424 \\tx848 \\tx1272 "
+ "\\tx1696 \\tx2120 \\tx2544 \\tx2968 \\tx3392 \\tx3816 \\tx4240 \\tx4664 "
+ "\\tx5088 \\tx5512 \\tx5936 \\tx6360 \\tx6784 \\tx7208 \\tx7632 \\tx8056 "
+ "\\tx8480 \\tx8904 \\tx9328 \\tx9752 \\tx10176 \\tx10600 \\tx11024 "
"\\tx11448 \\tx11872 \\tx12296 \\tx12720 \\tx13144 \\cf2 RichEdit1\\line }";
const char * streamText3 = "RichEdit1";
struct StringWithLength cookieForStream4;
const char * streamText4 =
- "This text just needs to be long enough to cause run to be split onto "\
- "two seperate lines and make sure the null terminating character is "\
+ "This text just needs to be long enough to cause run to be split onto "
+ "two separate lines and make sure the null terminating character is "
"handled properly.\0";
int length4 = strlen(streamText4) + 1;
cookieForStream4.buffer = (char *)streamText4;
es.dwCookie = (DWORD_PTR)&streamText0;
es.dwError = 0;
es.pfnCallback = test_EM_STREAMIN_esCallback;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 12,
es.dwCookie = (DWORD_PTR)&streamText0a;
es.dwError = 0;
es.pfnCallback = test_EM_STREAMIN_esCallback;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 12,
es.dwCookie = (DWORD_PTR)&streamText0b;
es.dwError = 0;
es.pfnCallback = test_EM_STREAMIN_esCallback;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 14,
es.dwCookie = (DWORD_PTR)&streamText1;
es.dwError = 0;
es.pfnCallback = test_EM_STREAMIN_esCallback;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 12,
es.dwCookie = (DWORD_PTR)&streamText2;
es.dwError = 0;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 0,
es.dwCookie = (DWORD_PTR)&streamText3;
es.dwError = 0;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_RTF), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_RTF, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == 0,
es.dwCookie = (DWORD_PTR)&cookieForStream4;
es.dwError = 0;
es.pfnCallback = test_EM_STREAMIN_esCallback2;
- SendMessage(hwndRichEdit, EM_STREAMIN,
- (WPARAM)(SF_TEXT), (LPARAM)&es);
+ SendMessage(hwndRichEdit, EM_STREAMIN, SF_TEXT, (LPARAM)&es);
result = SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
ok (result == length4,
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, EM_STREAMIN, SF_TEXT, (LPARAM)&es);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
result = strcmp (buffer,"test");
ok (result == 0,
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, EM_STREAMIN, SF_TEXT|SFF_SELECTION, (LPARAM)&es);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
result = strcmp (buffer,"testSome text");
ok (result == 0,
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, EM_STREAMIN, SF_TEXT|SFF_SELECTION, (LPARAM)&es);
SendMessage(hwndRichEdit, WM_GETTEXT, 1024, (LPARAM) buffer);
result = strcmp (buffer,"Sometesttext");
ok (result == 0,
char bufA[64];
WCHAR bufW[64];
HWND hwnd;
- int is_win9x, em_settextex_supported, ret;
-
- is_win9x = GetVersion() & 0x80000000;
+ int em_settextex_supported, ret;
#define set_textA(hwnd, wm_set_text, txt) \
do { \
expect_empty(hwnd, WM_GETTEXT);
expect_empty(hwnd, EM_GETTEXTEX);
- ret = SendMessageA(hwnd, WM_CHAR, (WPARAM)textW[0], 0);
+ ret = SendMessageA(hwnd, WM_CHAR, textW[0], 0);
ok(!ret, "SendMessageA(WM_CHAR) should return 0, got %d\n", ret);
expect_textA(hwnd, WM_GETTEXT, "t");
expect_textA(hwnd, EM_GETTEXTEX, "t");
expect_textW(hwnd, EM_GETTEXTEX, tW);
- ret = SendMessageA(hwnd, WM_CHAR, (WPARAM)textA[1], 0);
+ ret = SendMessageA(hwnd, WM_CHAR, textA[1], 0);
ok(!ret, "SendMessageA(WM_CHAR) should return 0, got %d\n", ret);
expect_textA(hwnd, WM_GETTEXT, "te");
expect_textA(hwnd, EM_GETTEXTEX, "te");
char buffer[64] = {0};
/* single line */
- hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP,
- 0, 0, 200, 60, 0, 0, 0, 0);
+ if (!is_win9x)
+ hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP,
+ 0, 0, 200, 60, 0, 0, 0, 0);
+ else
+ hwnd = CreateWindowExA(0, "RichEdit20A", NULL, WS_POPUP,
+ 0, 0, 200, 60, 0, 0, 0, 0);
ok(hwnd != 0, "CreateWindowExA error %u\n", GetLastError());
gtl.flags = GTL_NUMCHARS | GTL_PRECISE | GTL_USECRLF;
DestroyWindow(hwnd);
/* multi line */
- hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP | ES_MULTILINE,
- 0, 0, 200, 60, 0, 0, 0, 0);
+ if (!is_win9x)
+ hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP | ES_MULTILINE,
+ 0, 0, 200, 60, 0, 0, 0, 0);
+ else
+ hwnd = CreateWindowExA(0, "RichEdit20A", NULL, WS_POPUP | ES_MULTILINE,
+ 0, 0, 200, 60, 0, 0, 0, 0);
ok(hwnd != 0, "CreateWindowExA error %u\n", GetLastError());
gtl.flags = GTL_NUMCHARS | GTL_PRECISE | GTL_USECRLF;
static void test_eventMask(void)
{
HWND parent;
- int ret;
+ int ret, style;
WNDCLASSA cls;
const char text[] = "foo bar\n";
int eventMask;
cls.cbWndExtra = 0;
cls.hInstance = GetModuleHandleA(0);
cls.hIcon = 0;
- cls.hCursor = LoadCursorA(0, (LPSTR)IDC_ARROW);
+ cls.hCursor = LoadCursorA(0, IDC_ARROW);
cls.hbrBackground = GetStockObject(WHITE_BRUSH);
cls.lpszMenuName = NULL;
cls.lpszClassName = "EventMaskParentClass";
ok(eventMaskEditHwnd != 0, "Failed to create edit window\n");
eventMask = ENM_CHANGE | ENM_UPDATE;
- ret = SendMessage(eventMaskEditHwnd, EM_SETEVENTMASK, 0, (LPARAM) eventMask);
+ ret = SendMessage(eventMaskEditHwnd, EM_SETEVENTMASK, 0, eventMask);
ok(ret == ENM_NONE, "wrong event mask\n");
ret = SendMessage(eventMaskEditHwnd, EM_GETEVENTMASK, 0, 0);
ok(ret == eventMask, "failed to set event mask\n");
ok(queriedEventMask == (eventMask & ~ENM_CHANGE),
"wrong event mask (0x%x) during WM_COMMAND\n", queriedEventMask);
+ /* check to see if EN_CHANGE is sent when redraw is turned off */
+ SendMessage(eventMaskEditHwnd, WM_CLEAR, 0, 0);
+ ok(IsWindowVisible(eventMaskEditHwnd), "Window should be visible.\n");
+ SendMessage(eventMaskEditHwnd, WM_SETREDRAW, FALSE, 0);
+ /* redraw is disabled by making the window invisible. */
+ ok(!IsWindowVisible(eventMaskEditHwnd), "Window shouldn't be visible.\n");
+ queriedEventMask = 0; /* initialize to something other than we expect */
+ SendMessage(eventMaskEditHwnd, EM_REPLACESEL, 0, (LPARAM) text);
+ ok(queriedEventMask == (eventMask & ~ENM_CHANGE),
+ "wrong event mask (0x%x) during WM_COMMAND\n", queriedEventMask);
+ SendMessage(eventMaskEditHwnd, WM_SETREDRAW, TRUE, 0);
+ ok(IsWindowVisible(eventMaskEditHwnd), "Window should be visible.\n");
+
+ /* check to see if EN_UPDATE is sent when the editor isn't visible */
+ SendMessage(eventMaskEditHwnd, WM_CLEAR, 0, 0);
+ style = GetWindowLong(eventMaskEditHwnd, GWL_STYLE);
+ SetWindowLong(eventMaskEditHwnd, GWL_STYLE, style & ~WS_VISIBLE);
+ ok(!IsWindowVisible(eventMaskEditHwnd), "Window shouldn't be visible.\n");
+ watchForEventMask = EN_UPDATE;
+ queriedEventMask = 0; /* initialize to something other than we expect */
+ SendMessage(eventMaskEditHwnd, EM_REPLACESEL, 0, (LPARAM) text);
+ ok(queriedEventMask == 0,
+ "wrong event mask (0x%x) during WM_COMMAND\n", queriedEventMask);
+ SetWindowLong(eventMaskEditHwnd, GWL_STYLE, style);
+ ok(IsWindowVisible(eventMaskEditHwnd), "Window should be visible.\n");
+ queriedEventMask = 0; /* initialize to something other than we expect */
+ SendMessage(eventMaskEditHwnd, EM_REPLACESEL, 0, (LPARAM) text);
+ ok(queriedEventMask == eventMask,
+ "wrong event mask (0x%x) during WM_COMMAND\n", queriedEventMask);
+
+
+ DestroyWindow(parent);
}
static int received_WM_NOTIFY = 0;
static int modify_at_WM_NOTIFY = 0;
+static BOOL filter_on_WM_NOTIFY = FALSE;
static HWND hwndRichedit_WM_NOTIFY;
static LRESULT WINAPI WM_NOTIFY_ParentMsgCheckProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
received_WM_NOTIFY = 1;
modify_at_WM_NOTIFY = SendMessage(hwndRichedit_WM_NOTIFY, EM_GETMODIFY, 0, 0);
+ if (filter_on_WM_NOTIFY) return TRUE;
}
return DefWindowProcA(hwnd, message, wParam, lParam);
}
HWND parent;
WNDCLASSA cls;
CHARFORMAT2 cf2;
+ int sel_start, sel_end;
/* register class to capture WM_NOTIFY */
cls.style = 0;
cls.cbWndExtra = 0;
cls.hInstance = GetModuleHandleA(0);
cls.hIcon = 0;
- cls.hCursor = LoadCursorA(0, (LPSTR)IDC_ARROW);
+ cls.hCursor = LoadCursorA(0, IDC_ARROW);
cls.hbrBackground = GetStockObject(WHITE_BRUSH);
cls.lpszMenuName = NULL;
cls.lpszClassName = "WM_NOTIFY_ParentClass";
ME_CommitUndo, which should check whether message should be sent */
received_WM_NOTIFY = 0;
cf2.cbSize = sizeof(CHARFORMAT2);
- SendMessage(hwndRichedit_WM_NOTIFY, EM_GETCHARFORMAT, (WPARAM) SCF_DEFAULT,
- (LPARAM) &cf2);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_GETCHARFORMAT, SCF_DEFAULT, (LPARAM)&cf2);
cf2.dwMask = CFM_ITALIC | cf2.dwMask;
cf2.dwEffects = CFE_ITALIC ^ cf2.dwEffects;
SendMessage(hwndRichedit_WM_NOTIFY, EM_SETCHARFORMAT, 0, (LPARAM) &cf2);
ok(received_WM_NOTIFY == 1, "Expected WM_NOTIFY was NOT sent!\n");
ok(modify_at_WM_NOTIFY == 0, "WM_NOTIFY callback saw text flagged as modified!\n");
+ /* Test for WM_NOTIFY messages with redraw disabled. */
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_SETSEL, 0, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, WM_SETREDRAW, FALSE, 0);
+ received_WM_NOTIFY = 0;
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_REPLACESEL, FALSE, (LPARAM)"inserted");
+ ok(received_WM_NOTIFY == 1, "Expected WM_NOTIFY was NOT sent!\n");
+ SendMessage(hwndRichedit_WM_NOTIFY, WM_SETREDRAW, TRUE, 0);
+
+ /* Test filtering key events. */
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_SETSEL, 0, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_SETEVENTMASK, 0, ENM_KEYEVENTS);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ received_WM_NOTIFY = 0;
+ SendMessage(hwndRichedit_WM_NOTIFY, WM_KEYDOWN, VK_RIGHT, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 1 && sel_end == 1,
+ "selections is incorrectly at (%d,%d)\n", sel_start, sel_end);
+ filter_on_WM_NOTIFY = TRUE;
+ received_WM_NOTIFY = 0;
+ SendMessage(hwndRichedit_WM_NOTIFY, WM_KEYDOWN, VK_RIGHT, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 1 && sel_end == 1,
+ "selections is incorrectly at (%d,%d)\n", sel_start, sel_end);
+
+ /* test with owner set to NULL */
+ SetWindowLongPtr(hwndRichedit_WM_NOTIFY, GWLP_HWNDPARENT, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, WM_KEYDOWN, VK_RIGHT, 0);
+ SendMessage(hwndRichedit_WM_NOTIFY, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == 1 && sel_end == 1,
+ "selections is incorrectly at (%d,%d)\n", sel_start, sel_end);
+
DestroyWindow(hwndRichedit_WM_NOTIFY);
DestroyWindow(parent);
}
char buffer[64] = {0};
/* multi-line control inserts CR normally */
- hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP|ES_MULTILINE,
- 0, 0, 200, 60, 0, 0, 0, 0);
+ if (!is_win9x)
+ hwnd = CreateWindowExA(0, "RichEdit20W", NULL, WS_POPUP|ES_MULTILINE,
+ 0, 0, 200, 60, 0, 0, 0, 0);
+ else
+ hwnd = CreateWindowExA(0, "RichEdit20A", NULL, WS_POPUP|ES_MULTILINE,
+ 0, 0, 200, 60, 0, 0, 0, 0);
ok(hwnd != 0, "CreateWindowExA error %u\n", GetLastError());
result = SendMessage(hwnd, EM_CANUNDO, 0, 0);
simulate_typing_characters(hwnd, "one two three");
result = SendMessage(hwnd, EM_CANREDO, 0, 0);
ok (result == FALSE, "Redo buffer should have been cleared by typing.\n");
- SendMessage(hwnd, WM_KILLFOCUS, (WPARAM)NULL, 0);
- SendMessage(hwnd, WM_SETFOCUS, (WPARAM)NULL, 0);
+ SendMessage(hwnd, WM_KILLFOCUS, 0, 0);
+ SendMessage(hwnd, WM_SETFOCUS, 0, 0);
simulate_typing_characters(hwnd, " four five six");
result = SendMessage(hwnd, EM_UNDO, 0, 0);
ok (result == TRUE, "Failed to undo typed characters.\n");
SendMessage(hwnd, EM_EMPTYUNDOBUFFER, 0, 0);
result = SendMessageA(hwnd, WM_SETTEXT, 0, (LPARAM)"abcd");
ok(result == TRUE, "Failed to set the text.\n");
- SendMessage(hwnd, EM_SETSEL, (WPARAM)1, (LPARAM)1);
+ SendMessage(hwnd, EM_SETSEL, 1, 1);
SendMessage(hwnd, WM_KEYDOWN, VK_DELETE, 1);
SendMessage(hwnd, WM_KEYUP, VK_DELETE, 1);
SendMessage(hwnd, WM_KEYDOWN, VK_DELETE, 1);
DestroyWindow(hwnd);
}
+static LONG CALLBACK customWordBreakProc(WCHAR *text, int pos, int bytes, int code)
+{
+ int length;
+
+ /* MSDN lied, length is actually the number of bytes. */
+ length = bytes / sizeof(WCHAR);
+ switch(code)
+ {
+ case WB_ISDELIMITER:
+ return text[pos] == 'X';
+ case WB_LEFT:
+ case WB_MOVEWORDLEFT:
+ if (customWordBreakProc(text, pos, bytes, WB_ISDELIMITER))
+ return pos-1;
+ return min(customWordBreakProc(text, pos, bytes, WB_LEFTBREAK)-1, 0);
+ case WB_LEFTBREAK:
+ pos--;
+ while (pos > 0 && !customWordBreakProc(text, pos, bytes, WB_ISDELIMITER))
+ pos--;
+ return pos;
+ case WB_RIGHT:
+ case WB_MOVEWORDRIGHT:
+ if (customWordBreakProc(text, pos, bytes, WB_ISDELIMITER))
+ return pos+1;
+ return min(customWordBreakProc(text, pos, bytes, WB_RIGHTBREAK)+1, length);
+ case WB_RIGHTBREAK:
+ pos++;
+ while (pos < length && !customWordBreakProc(text, pos, bytes, WB_ISDELIMITER))
+ pos++;
+ return pos;
+ default:
+ ok(FALSE, "Unexpected code %d\n", code);
+ break;
+ }
+ return 0;
+}
+
#define SEND_CTRL_LEFT(hwnd) pressKeyWithModifier(hwnd, VK_CONTROL, VK_LEFT)
#define SEND_CTRL_RIGHT(hwnd) pressKeyWithModifier(hwnd, VK_CONTROL, VK_RIGHT)
HWND hwnd;
int result;
int sel_start, sel_end;
+ const WCHAR textW[] = {'o','n','e',' ','t','w','o','X','t','h','r','e','e',0};
/* multi-line control inserts CR normally */
hwnd = new_richedit(NULL);
/* one |two three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 4, "Cursur is at %d instead of %d\n", sel_start, 4);
+ ok(sel_start == 4, "Cursor is at %d instead of %d\n", sel_start, 4);
SEND_CTRL_RIGHT(hwnd);
/* one two |three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 9, "Cursur is at %d instead of %d\n", sel_start, 9);
+ ok(sel_start == 9, "Cursor is at %d instead of %d\n", sel_start, 9);
SEND_CTRL_LEFT(hwnd);
/* one |two three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 4, "Cursur is at %d instead of %d\n", sel_start, 4);
+ ok(sel_start == 4, "Cursor is at %d instead of %d\n", sel_start, 4);
SEND_CTRL_LEFT(hwnd);
/* |one two three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 0, "Cursur is at %d instead of %d\n", sel_start, 0);
+ ok(sel_start == 0, "Cursor is at %d instead of %d\n", sel_start, 0);
SendMessage(hwnd, EM_SETSEL, 8, 8);
/* one two | three */
/* one two |three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 9, "Cursur is at %d instead of %d\n", sel_start, 9);
+ ok(sel_start == 9, "Cursor is at %d instead of %d\n", sel_start, 9);
SendMessage(hwnd, EM_SETSEL, 11, 11);
/* one two th|ree */
/* one two |three */
SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
ok(sel_start == sel_end, "Selection should be empty\n");
- ok(sel_start == 9, "Cursur is at %d instead of %d\n", sel_start, 9);
+ ok(sel_start == 9, "Cursor is at %d instead of %d\n", sel_start, 9);
+
+ /* Test with a custom word break procedure that uses X as the delimiter. */
+ result = SendMessageA(hwnd, WM_SETTEXT, 0, (LPARAM)"one twoXthree");
+ ok (result == TRUE, "Failed to clear the text.\n");
+ SendMessage(hwnd, EM_SETWORDBREAKPROC, 0, (LPARAM)customWordBreakProc);
+ /* |one twoXthree */
+ SEND_CTRL_RIGHT(hwnd);
+ /* one twoX|three */
+ SendMessage(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == sel_end, "Selection should be empty\n");
+ ok(sel_start == 8, "Cursor is at %d instead of %d\n", sel_start, 8);
+
+ DestroyWindow(hwnd);
+
+ /* Make sure the behaviour is the same with a unicode richedit window,
+ * and using unicode functions. */
+ if (is_win9x)
+ {
+ skip("Cannot test with unicode richedit window\n");
+ return;
+ }
+
+ hwnd = CreateWindowW(RICHEDIT_CLASS20W, NULL,
+ ES_MULTILINE|WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+
+ /* Test with a custom word break procedure that uses X as the delimiter. */
+ result = SendMessageW(hwnd, WM_SETTEXT, 0, (LPARAM)textW);
+ ok (result == TRUE, "Failed to clear the text.\n");
+ SendMessageW(hwnd, EM_SETWORDBREAKPROC, 0, (LPARAM)customWordBreakProc);
+ /* |one twoXthree */
+ SEND_CTRL_RIGHT(hwnd);
+ /* one twoX|three */
+ SendMessageW(hwnd, EM_GETSEL, (WPARAM)&sel_start, (LPARAM)&sel_end);
+ ok(sel_start == sel_end, "Selection should be empty\n");
+ ok(sel_start == 8, "Cursor is at %d instead of %d\n", sel_start, 8);
DestroyWindow(hwnd);
}
{
HWND hwnd;
int result;
+ RECT rcClient;
POINTL point;
point.x = 0;
- point.y = 50;
+ point.y = 40;
/* multi-line control inserts CR normally */
hwnd = new_richedit(NULL);
result = SendMessageA(hwnd, WM_SETTEXT, 0,
- (LPARAM)"one two three four five six seven");
+ (LPARAM)"one two three four five six seven\reight");
+
+ GetClientRect(hwnd, &rcClient);
+
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ ok(result == 34, "expected character index of 34 but got %d\n", result);
+
+ /* Test with points outside the bounds of the richedit control. */
+ point.x = -1;
+ point.y = 40;
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ todo_wine ok(result == 34, "expected character index of 34 but got %d\n", result);
+
+ point.x = 1000;
+ point.y = 0;
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ todo_wine ok(result == 33, "expected character index of 33 but got %d\n", result);
+ point.x = 1000;
+ point.y = 40;
result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
- ok(result == 0, "expected character index of 0 but got %d\n", result);
+ todo_wine ok(result == 39, "expected character index of 39 but got %d\n", result);
+
+ point.x = 1000;
+ point.y = -1;
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ todo_wine ok(result == 0, "expected character index of 0 but got %d\n", result);
+
+ point.x = 1000;
+ point.y = rcClient.bottom + 1;
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ todo_wine ok(result == 34, "expected character index of 34 but got %d\n", result);
+
+ point.x = 1000;
+ point.y = rcClient.bottom;
+ result = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM)&point);
+ todo_wine ok(result == 39, "expected character index of 39 but got %d\n", result);
DestroyWindow(hwnd);
}
-START_TEST( editor )
+static void test_word_wrap(void)
{
- MSG msg;
- time_t end;
+ HWND hwnd;
+ POINTL point = {0, 60}; /* This point must be below the first line */
+ const char *text = "Must be long enough to test line wrapping";
+ DWORD dwCommonStyle = WS_VISIBLE|WS_POPUP|WS_VSCROLL|ES_MULTILINE;
+ int res, pos, lines;
+
+ /* Test the effect of WS_HSCROLL and ES_AUTOHSCROLL styles on wrapping
+ * when specified on window creation and set later. */
+ hwnd = CreateWindow(RICHEDIT_CLASS, NULL, dwCommonStyle,
+ 0, 0, 200, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "error: %d\n", (int) GetLastError());
+ res = SendMessage(hwnd, WM_SETTEXT, 0, (LPARAM) text);
+ ok(res, "WM_SETTEXT failed.\n");
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(pos, "pos=%d indicating no word wrap when it is expected.\n", pos);
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines > 1, "Line was expected to wrap (lines=%d).\n", lines);
+
+ SetWindowLongW(hwnd, GWL_STYLE, dwCommonStyle|WS_HSCROLL|ES_AUTOHSCROLL);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(pos, "pos=%d indicating no word wrap when it is expected.\n", pos);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindow(RICHEDIT_CLASS, NULL, dwCommonStyle|WS_HSCROLL,
+ 0, 0, 200, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "error: %d\n", (int) GetLastError());
+
+ res = SendMessage(hwnd, WM_SETTEXT, 0, (LPARAM) text);
+ ok(res, "WM_SETTEXT failed.\n");
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines == 1, "Line wasn't expected to wrap (lines=%d).\n", lines);
+
+ SetWindowLongW(hwnd, GWL_STYLE, dwCommonStyle);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindow(RICHEDIT_CLASS, NULL, dwCommonStyle|ES_AUTOHSCROLL,
+ 0, 0, 200, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "error: %d\n", (int) GetLastError());
+ res = SendMessage(hwnd, WM_SETTEXT, 0, (LPARAM) text);
+ ok(res, "WM_SETTEXT failed.\n");
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+
+ SetWindowLongW(hwnd, GWL_STYLE, dwCommonStyle);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindow(RICHEDIT_CLASS, NULL,
+ dwCommonStyle|WS_HSCROLL|ES_AUTOHSCROLL,
+ 0, 0, 200, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "error: %d\n", (int) GetLastError());
+ res = SendMessage(hwnd, WM_SETTEXT, 0, (LPARAM) text);
+ ok(res, "WM_SETTEXT failed.\n");
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+
+ SetWindowLongW(hwnd, GWL_STYLE, dwCommonStyle);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+
+ /* Test the effect of EM_SETTARGETDEVICE on word wrap. */
+ res = SendMessage(hwnd, EM_SETTARGETDEVICE, 0, 1);
+ ok(res, "EM_SETTARGETDEVICE failed (returned %d).\n", res);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(!pos, "pos=%d indicating word wrap when none is expected.\n", pos);
+
+ res = SendMessage(hwnd, EM_SETTARGETDEVICE, 0, 0);
+ ok(res, "EM_SETTARGETDEVICE failed (returned %d).\n", res);
+ pos = SendMessage(hwnd, EM_CHARFROMPOS, 0, (LPARAM) &point);
+ ok(pos, "pos=%d indicating no word wrap when it is expected.\n", pos);
+ DestroyWindow(hwnd);
+
+ /* Test to see if wrapping happens with redraw disabled. */
+ hwnd = CreateWindow(RICHEDIT_CLASS, NULL, dwCommonStyle,
+ 0, 0, 400, 80, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "error: %d\n", (int) GetLastError());
+ SendMessage(hwnd, WM_SETREDRAW, FALSE, 0);
+ res = SendMessage(hwnd, EM_REPLACESEL, FALSE, (LPARAM) text);
+ ok(res, "EM_REPLACESEL failed.\n");
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines == 1, "Line wasn't expected to wrap (lines=%d).\n", lines);
+ MoveWindow(hwnd, 0, 0, 200, 80, FALSE);
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines > 1, "Line was expected to wrap (lines=%d).\n", lines);
+
+ SendMessage(hwnd, WM_SETREDRAW, TRUE, 0);
+ DestroyWindow(hwnd);
+}
+
+static void test_autoscroll(void)
+{
+ HWND hwnd = new_richedit(NULL);
+ int lines, ret, redraw;
+ POINT pt;
+
+ for (redraw = 0; redraw <= 1; redraw++) {
+ trace("testing with WM_SETREDRAW=%d\n", redraw);
+ SendMessage(hwnd, WM_SETREDRAW, redraw, 0);
+ SendMessage(hwnd, EM_REPLACESEL, 0, (LPARAM)"1\n2\n3\n4\n5\n6\n7\n8");
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines == 8, "%d lines instead of 8\n", lines);
+ ret = SendMessage(hwnd, EM_GETSCROLLPOS, 0, (LPARAM)&pt);
+ ok(ret == 1, "EM_GETSCROLLPOS returned %d instead of 1\n", ret);
+ ok(pt.y != 0, "Didn't scroll down after replacing text.\n");
+ ret = GetWindowLong(hwnd, GWL_STYLE);
+ ok(ret & WS_VSCROLL, "Scrollbar was not shown yet (style=%x).\n", (UINT)ret);
+
+ SendMessage(hwnd, WM_SETTEXT, 0, 0);
+ lines = SendMessage(hwnd, EM_GETLINECOUNT, 0, 0);
+ ok(lines == 1, "%d lines instead of 1\n", lines);
+ ret = SendMessage(hwnd, EM_GETSCROLLPOS, 0, (LPARAM)&pt);
+ ok(ret == 1, "EM_GETSCROLLPOS returned %d instead of 1\n", ret);
+ ok(pt.y == 0, "y scroll position is %d after clearing text.\n", pt.y);
+ ret = GetWindowLong(hwnd, GWL_STYLE);
+ ok(!(ret & WS_VSCROLL), "Scrollbar is still shown (style=%x).\n", (UINT)ret);
+ }
+
+ SendMessage(hwnd, WM_SETREDRAW, TRUE, 0);
+ DestroyWindow(hwnd);
+
+ /* The WS_VSCROLL and WS_HSCROLL styles implicitly set
+ * auto vertical/horizontal scrolling options. */
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP|ES_MULTILINE|WS_VSCROLL|WS_HSCROLL,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ ret = SendMessage(hwnd, EM_GETOPTIONS, 0, 0);
+ ok(ret & ECO_AUTOVSCROLL, "ECO_AUTOVSCROLL isn't set.\n");
+ ok(ret & ECO_AUTOHSCROLL, "ECO_AUTOHSCROLL isn't set.\n");
+ ret = GetWindowLong(hwnd, GWL_STYLE);
+ ok(!(ret & ES_AUTOVSCROLL), "ES_AUTOVSCROLL is set.\n");
+ ok(!(ret & ES_AUTOHSCROLL), "ES_AUTOHSCROLL is set.\n");
+ DestroyWindow(hwnd);
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP|ES_MULTILINE,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ ret = SendMessage(hwnd, EM_GETOPTIONS, 0, 0);
+ ok(!(ret & ECO_AUTOVSCROLL), "ECO_AUTOVSCROLL is set.\n");
+ ok(!(ret & ECO_AUTOHSCROLL), "ECO_AUTOHSCROLL is set.\n");
+ ret = GetWindowLong(hwnd, GWL_STYLE);
+ ok(!(ret & ES_AUTOVSCROLL), "ES_AUTOVSCROLL is set.\n");
+ ok(!(ret & ES_AUTOHSCROLL), "ES_AUTOHSCROLL is set.\n");
+ DestroyWindow(hwnd);
+}
+
+
+static void test_format_rect(void)
+{
+ HWND hwnd;
+ RECT rc, expected, clientRect;
+ int n;
+ DWORD options;
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+
+ GetClientRect(hwnd, &clientRect);
+
+ expected = clientRect;
+ expected.left += 1;
+ expected.right -= 1;
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ for (n = -3; n <= 3; n++)
+ {
+ rc = clientRect;
+ rc.top += n;
+ rc.left += n;
+ rc.bottom -= n;
+ rc.right -= n;
+ SendMessageA(hwnd, EM_SETRECT, 0, (LPARAM)&rc);
+
+ expected = rc;
+ expected.top = max(0, rc.top);
+ expected.left = max(0, rc.left);
+ expected.bottom = min(clientRect.bottom, rc.bottom);
+ expected.right = min(clientRect.right, rc.right);
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "[n=%d] rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ n, rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+ }
+
+ rc = clientRect;
+ SendMessageA(hwnd, EM_SETRECT, 0, (LPARAM)&rc);
+ expected = clientRect;
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ /* Adding the selectionbar adds the selectionbar width to the left side. */
+ SendMessageA(hwnd, EM_SETOPTIONS, ECOOP_OR, ECO_SELECTIONBAR);
+ options = SendMessageA(hwnd, EM_GETOPTIONS, 0, 0);
+ ok(options & ECO_SELECTIONBAR, "EM_SETOPTIONS failed to add selectionbar.\n");
+ expected.left += 8; /* selection bar width */
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ rc = clientRect;
+ SendMessageA(hwnd, EM_SETRECT, 0, (LPARAM)&rc);
+ expected = clientRect;
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ /* Removing the selectionbar subtracts the selectionbar width from the left side,
+ * even if the left side is already 0. */
+ SendMessageA(hwnd, EM_SETOPTIONS, ECOOP_AND, ~ECO_SELECTIONBAR);
+ options = SendMessageA(hwnd, EM_GETOPTIONS, 0, 0);
+ ok(!(options & ECO_SELECTIONBAR), "EM_SETOPTIONS failed to remove selectionbar.\n");
+ expected.left -= 8; /* selection bar width */
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ /* Set the absolute value of the formatting rectangle. */
+ rc = clientRect;
+ SendMessageA(hwnd, EM_SETRECT, 0, (LPARAM)&rc);
+ expected = clientRect;
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "[n=%d] rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ n, rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ /* MSDN documents the EM_SETRECT message as using the rectangle provided in
+ * LPARAM as being a relative offset when the WPARAM value is 1, but these
+ * tests show that this isn't true. */
+ rc.top = 15;
+ rc.left = 15;
+ rc.bottom = clientRect.bottom - 15;
+ rc.right = clientRect.right - 15;
+ expected = rc;
+ SendMessageA(hwnd, EM_SETRECT, 1, (LPARAM)&rc);
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ /* For some reason it does not limit the values to the client rect with
+ * a WPARAM value of 1. */
+ rc.top = -15;
+ rc.left = -15;
+ rc.bottom = clientRect.bottom + 15;
+ rc.right = clientRect.right + 15;
+ expected = rc;
+ SendMessageA(hwnd, EM_SETRECT, 1, (LPARAM)&rc);
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ DestroyWindow(hwnd);
+
+ /* The extended window style affects the formatting rectangle. */
+ hwnd = CreateWindowEx(WS_EX_CLIENTEDGE, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP|WS_HSCROLL|WS_VSCROLL|WS_VISIBLE,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+
+ GetClientRect(hwnd, &clientRect);
+
+ expected = clientRect;
+ expected.left += 1;
+ expected.top += 1;
+ expected.right -= 1;
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ rc = clientRect;
+ rc.top += 5;
+ rc.left += 5;
+ rc.bottom -= 5;
+ rc.right -= 5;
+ expected = rc;
+ expected.top -= 1;
+ expected.left -= 1;
+ expected.right += 1;
+ SendMessageA(hwnd, EM_SETRECT, 0, (LPARAM)&rc);
+ SendMessageA(hwnd, EM_GETRECT, 0, (LPARAM)&rc);
+ ok(rc.top == expected.top && rc.left == expected.left &&
+ rc.bottom == expected.bottom && rc.right == expected.right,
+ "rect a(t=%d, l=%d, b=%d, r=%d) != e(t=%d, l=%d, b=%d, r=%d)\n",
+ rc.top, rc.left, rc.bottom, rc.right,
+ expected.top, expected.left, expected.bottom, expected.right);
+
+ DestroyWindow(hwnd);
+}
+
+static void test_WM_GETDLGCODE(void)
+{
+ HWND hwnd;
+ UINT res, expected;
+ MSG msg;
+
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|ES_WANTRETURN|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, 0);
+ expected = expected | DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ msg.message = WM_KEYDOWN;
+ msg.wParam = VK_RETURN;
+ msg.lParam = MapVirtualKey(VK_RETURN, MAPVK_VK_TO_VSC) | 0x0001;
+ msg.pt.x = 0;
+ msg.pt.y = 0;
+ msg.time = GetTickCount();
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|ES_WANTRETURN|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = expected | DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_WANTRETURN|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ msg.wParam = VK_TAB;
+ msg.lParam = MapVirtualKey(VK_TAB, MAPVK_VK_TO_VSC) | 0x0001;
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hold_key(VK_CONTROL);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ release_key(VK_CONTROL);
+
+ msg.wParam = 'a';
+ msg.lParam = MapVirtualKey('a', MAPVK_VK_TO_VSC) | 0x0001;
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ msg.message = WM_CHAR;
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ ES_MULTILINE|WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL|DLGC_WANTMESSAGE;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+
+ hwnd = CreateWindowEx(0, RICHEDIT_CLASS, NULL,
+ WS_POPUP,
+ 0, 0, 200, 60, NULL, NULL, hmoduleRichEdit, NULL);
+ ok(hwnd != NULL, "class: %s, error: %d\n", RICHEDIT_CLASS, (int) GetLastError());
+ msg.hwnd = hwnd;
+ res = SendMessage(hwnd, WM_GETDLGCODE, VK_RETURN, (LPARAM)&msg);
+ expected = DLGC_WANTCHARS|DLGC_WANTTAB|DLGC_WANTARROWS|DLGC_HASSETSEL;
+ ok(res == expected, "WM_GETDLGCODE returned %x but expected %x\n",
+ res, expected);
+ DestroyWindow(hwnd);
+}
+
+static void test_zoom(void)
+{
+ HWND hwnd;
+ UINT ret;
+ RECT rc;
+ POINT pt;
+ int numerator, denominator;
+
+ hwnd = new_richedit(NULL);
+ GetClientRect(hwnd, &rc);
+ pt.x = (rc.right - rc.left) / 2;
+ pt.y = (rc.bottom - rc.top) / 2;
+ ClientToScreen(hwnd, &pt);
+
+ /* Test initial zoom value */
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 0, "Numerator should be initialized to 0 (got %d).\n", numerator);
+ ok(denominator == 0, "Denominator should be initialized to 0 (got %d).\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ /* test scroll wheel */
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 110, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ /* Test how much the mouse wheel can zoom in and out. */
+ ret = SendMessage(hwnd, EM_SETZOOM, 490, 100);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 500, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 491, 100);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 491, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 20, 100);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, -WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 10, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 19, 100);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, -WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 19, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ /* Test how WM_SCROLLWHEEL treats our custom denominator. */
+ ret = SendMessage(hwnd, EM_SETZOOM, 50, 13);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ hold_key(VK_CONTROL);
+ ret = SendMessage(hwnd, WM_MOUSEWHEEL, MAKEWPARAM(MK_CONTROL, WHEEL_DELTA),
+ MAKELPARAM(pt.x, pt.y));
+ ok(!ret, "WM_MOUSEWHEEL failed (%d).\n", ret);
+ release_key(VK_CONTROL);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 394, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 100, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ /* Test bounds checking on EM_SETZOOM */
+ ret = SendMessage(hwnd, EM_SETZOOM, 2, 127);
+ ok(ret == TRUE, "EM_SETZOOM rejected valid values (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 127, 2);
+ ok(ret == TRUE, "EM_SETZOOM rejected valid values (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 2, 128);
+ ok(ret == FALSE, "EM_SETZOOM accepted invalid values (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 127, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 2, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_SETZOOM, 128, 2);
+ ok(ret == FALSE, "EM_SETZOOM accepted invalid values (%d).\n", ret);
+
+ /* See if negative numbers are accepted. */
+ ret = SendMessage(hwnd, EM_SETZOOM, -100, -100);
+ ok(ret == FALSE, "EM_SETZOOM accepted invalid values (%d).\n", ret);
+
+ /* See if negative numbers are accepted. */
+ ret = SendMessage(hwnd, EM_SETZOOM, 0, 100);
+ ok(ret == FALSE, "EM_SETZOOM failed (%d).\n", ret);
+
+ ret = SendMessage(hwnd, EM_GETZOOM, (WPARAM)&numerator, (LPARAM)&denominator);
+ ok(numerator == 127, "incorrect numerator is %d\n", numerator);
+ ok(denominator == 2, "incorrect denominator is %d\n", denominator);
+ ok(ret == TRUE, "EM_GETZOOM failed (%d).\n", ret);
+
+ /* Reset the zoom value */
+ ret = SendMessage(hwnd, EM_SETZOOM, 0, 0);
+ ok(ret == TRUE, "EM_SETZOOM failed (%d).\n", ret);
+
+ DestroyWindow(hwnd);
+}
+
+struct dialog_mode_messages
+{
+ int wm_getdefid, wm_close, wm_nextdlgctl;
+};
+
+static struct dialog_mode_messages dm_messages;
+
+#define test_dm_messages(wmclose, wmgetdefid, wmnextdlgctl) \
+ ok(dm_messages.wm_close == wmclose, "expected %d WM_CLOSE message, " \
+ "got %d\n", wmclose, dm_messages.wm_close); \
+ ok(dm_messages.wm_getdefid == wmgetdefid, "expected %d WM_GETDIFID message, " \
+ "got %d\n", wmgetdefid, dm_messages.wm_getdefid);\
+ ok(dm_messages.wm_nextdlgctl == wmnextdlgctl, "expected %d WM_NEXTDLGCTL message, " \
+ "got %d\n", wmnextdlgctl, dm_messages.wm_nextdlgctl)
+
+static LRESULT CALLBACK dialog_mode_wnd_proc(HWND hwnd, UINT iMsg, WPARAM wParam, LPARAM lParam)
+{
+ switch (iMsg)
+ {
+ case DM_GETDEFID:
+ dm_messages.wm_getdefid++;
+ return MAKELONG(ID_RICHEDITTESTDBUTTON, DC_HASDEFID);
+ case WM_NEXTDLGCTL:
+ dm_messages.wm_nextdlgctl++;
+ break;
+ case WM_CLOSE:
+ dm_messages.wm_close++;
+ break;
+ }
+
+ return DefWindowProc(hwnd, iMsg, wParam, lParam);
+}
+
+static void test_dialogmode(void)
+{
+ HWND hwRichEdit, hwParent, hwButton;
+ MSG msg= {0};
+ int lcount, r;
+ WNDCLASSA cls;
+
+ cls.style = 0;
+ cls.lpfnWndProc = dialog_mode_wnd_proc;
+ cls.cbClsExtra = 0;
+ cls.cbWndExtra = 0;
+ cls.hInstance = GetModuleHandleA(0);
+ cls.hIcon = 0;
+ cls.hCursor = LoadCursorA(0, IDC_ARROW);
+ cls.hbrBackground = GetStockObject(WHITE_BRUSH);
+ cls.lpszMenuName = NULL;
+ cls.lpszClassName = "DialogModeParentClass";
+ if(!RegisterClassA(&cls)) assert(0);
+
+ hwParent = CreateWindow("DialogModeParentClass", NULL, WS_OVERLAPPEDWINDOW,
+ CW_USEDEFAULT, 0, 200, 120, NULL, NULL, GetModuleHandleA(0), NULL);
+
+ /* Test richedit(ES_MULTILINE) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE, hwParent);
+
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, 0);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(3 == lcount, "expected 3, got %d\n", lcount);
+
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(3 == lcount, "expected 3, got %d\n", lcount);
+
+ DestroyWindow(hwRichEdit);
+
+ /* Test standalone richedit(ES_MULTILINE) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE, NULL);
+
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ DestroyWindow(hwRichEdit);
+
+ /* Check a destination for messages */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE, hwParent);
+
+ SetWindowLong(hwRichEdit, GWL_STYLE, GetWindowLong(hwRichEdit, GWL_STYLE)& ~WS_POPUP);
+ SetParent( hwRichEdit, NULL);
+
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 1, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 1);
+
+ DestroyWindow(hwRichEdit);
+
+ /* Check messages from richedit(ES_MULTILINE) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE, hwParent);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 1, 0);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 1);
+
+ hwButton = CreateWindow("BUTTON", "OK", WS_VISIBLE|WS_CHILD|BS_PUSHBUTTON,
+ 100, 100, 50, 20, hwParent, (HMENU)ID_RICHEDITTESTDBUTTON, GetModuleHandleA(0), NULL);
+ ok(hwButton!=NULL, "CreateWindow failed with error code %d\n", GetLastError());
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 1, 1);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ DestroyWindow(hwButton);
+ DestroyWindow(hwRichEdit);
+
+ /* Check messages from richedit(ES_MULTILINE|ES_WANTRETURN) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_MULTILINE|ES_WANTRETURN, hwParent);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(2 == lcount, "expected 2, got %d\n", lcount);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8f == r, "expected 0x8f, got 0x%x\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(3 == lcount, "expected 3, got %d\n", lcount);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 1);
+
+ hwButton = CreateWindow("BUTTON", "OK", WS_VISIBLE|WS_CHILD|BS_PUSHBUTTON,
+ 100, 100, 50, 20, hwParent, (HMENU)ID_RICHEDITTESTDBUTTON, GetModuleHandleA(0), NULL);
+ ok(hwButton!=NULL, "CreateWindow failed with error code %d\n", GetLastError());
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ lcount = SendMessage(hwRichEdit, EM_GETLINECOUNT, 0, 0);
+ ok(4 == lcount, "expected 4, got %d\n", lcount);
+
+ DestroyWindow(hwButton);
+ DestroyWindow(hwRichEdit);
+
+ /* Check messages from richedit(0) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, 0, hwParent);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8b == r, "expected 0x8b, got 0x%x\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 1, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_ESCAPE, 0x10001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_TAB, 0xf0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 1);
+
+ hwButton = CreateWindow("BUTTON", "OK", WS_VISIBLE|WS_CHILD|BS_PUSHBUTTON,
+ 100, 100, 50, 20, hwParent, (HMENU)ID_RICHEDITTESTDBUTTON, GetModuleHandleA(0), NULL);
+ ok(hwButton!=NULL, "CreateWindow failed with error code %d\n", GetLastError());
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 1, 1);
+
+ DestroyWindow(hwRichEdit);
+
+ /* Check messages from richedit(ES_WANTRETURN) */
+
+ hwRichEdit = new_window(RICHEDIT_CLASS, ES_WANTRETURN, hwParent);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_GETDLGCODE, 0, (LPARAM)&msg);
+ ok(0x8b == r, "expected 0x8b, got 0x%x\n", r);
+ test_dm_messages(0, 0, 0);
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ hwButton = CreateWindow("BUTTON", "OK", WS_VISIBLE|WS_CHILD|BS_PUSHBUTTON,
+ 100, 100, 50, 20, hwParent, (HMENU)ID_RICHEDITTESTDBUTTON, GetModuleHandleA(0), NULL);
+ ok(hwButton!=NULL, "CreateWindow failed with error code %d\n", GetLastError());
+
+ memset(&dm_messages, 0, sizeof(dm_messages));
+ r = SendMessage(hwRichEdit, WM_KEYDOWN, VK_RETURN, 0x1c0001);
+ ok(0 == r, "expected 0, got %d\n", r);
+ test_dm_messages(0, 0, 0);
+
+ DestroyWindow(hwRichEdit);
+ DestroyWindow(hwParent);
+}
+
+START_TEST( editor )
+{
/* 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());
+
+ is_win9x = GetVersion() & 0x80000000;
+
test_WM_CHAR();
test_EM_FINDTEXT();
test_EM_GETLINE();
test_EM_POSFROMCHAR();
test_EM_SCROLLCARET();
test_EM_SCROLL();
+ test_scrollbar_visibility();
test_WM_SETTEXT();
test_EM_LINELENGTH();
test_EM_SETCHARFORMAT();
test_WM_PASTE();
test_EM_STREAMIN();
test_EM_STREAMOUT();
+ test_EM_STREAMOUT_FONTTBL();
test_EM_StreamIn_Undo();
test_EM_FORMATRANGE();
test_unicode_conversions();
test_undo_coalescing();
test_word_movement();
test_EM_CHARFROMPOS();
+ test_SETPARAFORMAT();
+ test_word_wrap();
+ test_autoscroll();
+ test_format_rect();
+ test_WM_GETDLGCODE();
+ test_zoom();
+ test_dialogmode();
/* 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);
- }
- }
+ keep_responsive(30);
}
OleFlushClipboard();