Commit 61f189ce authored by Dylan Smith's avatar Dylan Smith Committed by Alexandre Julliard

richedit: Handle starting in EOL sequence in EM_GETTEXTRANGE.

EM_GETTEXTRANGE allows the start character offset and end characters offset to be used to specify the range of text to retrieve. If the start offset is in the middle of an end of paragraph run (i.e. \r\n), then it should only retrieve the characters after the specified character offset.
parent 5bf9e73d
...@@ -4568,104 +4568,69 @@ int ME_CountParagraphsBetween(ME_TextEditor *editor, int from, int to) ...@@ -4568,104 +4568,69 @@ int ME_CountParagraphsBetween(ME_TextEditor *editor, int from, int to)
} }
int ME_GetTextW(ME_TextEditor *editor, WCHAR *buffer, int nStart, int nChars, int bCRLF) int ME_GetTextW(ME_TextEditor *editor, WCHAR *buffer, int nStart,
int nChars, int bCRLF)
{ {
ME_DisplayItem *item = ME_FindItemAtOffset(editor, diRun, nStart, &nStart); ME_DisplayItem *pRun;
int nWritten = 0; int nOffset, nWritten = 0;
WCHAR *pStart = buffer; WCHAR *pStart = buffer;
if (!item) { ME_RunOfsFromCharOfs(editor, nStart, &pRun, &nOffset);
*buffer = 0; /* Get actual offset within run (ME_RunOfsFromCharOfs sets to 0 if MERF_ENDPARA) */
return 0; nOffset = nStart - ME_GetParagraph(pRun)->member.para.nCharOfs - pRun->member.run.nCharOfs;
}
/* bCRLF flag is only honored in 2.0 and up. 1.0 must always return text verbatim */ /* bCRLF flag is only honored in 2.0 and up. 1.0 must always return text verbatim */
if (editor->bEmulateVersion10) bCRLF = 0; if (editor->bEmulateVersion10) bCRLF = 0;
if (nStart) while (nChars && pRun)
{ {
int nLen = ME_StrLen(item->member.run.strText) - nStart; int nLen;
if (nLen > nChars)
nLen = nChars;
CopyMemory(buffer, item->member.run.strText->szData + nStart, sizeof(WCHAR)*nLen);
nChars -= nLen;
nWritten += nLen;
buffer += nLen;
if (!nChars) {
*buffer = 0;
return nWritten;
}
nStart = 0;
item = ME_FindItemFwd(item, diRun);
}
while(nChars && item)
{
int nLen = ME_StrLen(item->member.run.strText);
if (item->member.run.nFlags & MERF_ENDPARA)
nLen = item->member.run.nCR + item->member.run.nLF;
if (nLen > nChars)
nLen = nChars;
if (item->member.run.nFlags & MERF_ENDCELL && if (pRun->member.run.nFlags & MERF_ENDCELL &&
item->member.run.nFlags & MERF_ENDPARA) pRun->member.run.nFlags & MERF_ENDPARA)
{ {
*buffer = '\t'; *buffer = '\t';
} nLen = 1;
else if (item->member.run.nFlags & MERF_ENDPARA) } else if (pRun->member.run.nFlags & MERF_ENDPARA) {
{ if (!ME_FindItemFwd(pRun, diRun)) {
if (!ME_FindItemFwd(item, diRun))
/* No '\r' is appended to the last paragraph. */ /* No '\r' is appended to the last paragraph. */
nLen = 0; nLen = 0;
else if (bCRLF && nChars == 1) { } else if (bCRLF && nChars == 1) {
nLen = 0; nLen = 0;
nChars = 0; nChars = 0;
} else { } else {
if (bCRLF) int numCR, numLF;
{
/* richedit 2.0 case - actual line-break is \r but should report \r\n */
if (ME_GetParagraph(item)->member.para.nFlags & (MEPF_ROWSTART|MEPF_ROWEND))
assert(nLen == 2);
else
assert(nLen == 1);
*buffer++ = '\r';
*buffer = '\n'; /* Later updated by nLen==1 at the end of the loop */
if (nLen == 1)
nWritten++;
else
buffer--;
}
else
{
int i, j; int i, j;
/* richedit 2.0 verbatim has only \r. richedit 1.0 should honor encodings */ if (bCRLF)
i = 0;
while (nChars - i > 0 && i < item->member.run.nCR)
{
buffer[i] = '\r'; i++;
}
j = 0;
while (nChars - i - j > 0 && j < item->member.run.nLF)
{ {
buffer[i+j] = '\n'; j++; numCR = 1;
} numLF = 1;
} else {
numCR = pRun->member.run.nCR;
numLF = pRun->member.run.nLF;
} }
numCR -= nOffset;
nLen = min(nChars, numCR + numLF);
for (i = 0; i < nLen && i < numCR; i++)
buffer[i] = '\r';
for (j = 0; i + j < nLen && j < numLF; j++)
buffer[i+j] = '\n';
} }
} else {
nLen = min(nChars, ME_StrLen(pRun->member.run.strText) - nOffset);
CopyMemory(buffer, pRun->member.run.strText->szData + nOffset,
sizeof(WCHAR) * nLen);
} }
else
CopyMemory(buffer, item->member.run.strText->szData, sizeof(WCHAR)*nLen);
nChars -= nLen; nChars -= nLen;
nWritten += nLen; nWritten += nLen;
buffer += nLen; buffer += nLen;
nOffset = 0;
if (!nChars) pRun = ME_FindItemFwd(pRun, diRun);
{
TRACE("nWritten=%d, actual=%d\n", nWritten, buffer-pStart);
*buffer = 0;
return nWritten;
}
item = ME_FindItemFwd(item, diRun);
} }
*buffer = 0; *buffer = 0;
TRACE("nWritten=%d, actual=%d\n", nWritten, buffer-pStart); TRACE("nWritten=%d, actual=%d\n", nWritten, buffer-pStart);
......
...@@ -487,7 +487,7 @@ static void test_EM_GETTEXTRANGE(void) ...@@ -487,7 +487,7 @@ static void test_EM_GETTEXTRANGE(void)
textRange.chrg.cpMax = 12; textRange.chrg.cpMax = 12;
result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange); result = SendMessage(hwndRichEdit, EM_GETTEXTRANGE, 0, (LPARAM)&textRange);
ok(result == 4, "EM_GETTEXTRANGE returned %ld\n", result); ok(result == 4, "EM_GETTEXTRANGE returned %ld\n", result);
todo_wine ok(!strcmp(expect2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer); ok(!strcmp(expect2, buffer), "EM_GETTEXTRANGE filled %s\n", buffer);
SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text3); SendMessage(hwndRichEdit, WM_SETTEXT, 0, (LPARAM)text3);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment