diff --git a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Display.java b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Display.java index 70576d85ce..64b2bbae36 100644 --- a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Display.java +++ b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Display.java @@ -5152,16 +5152,90 @@ String wrapText (String text, long handle, int width) { } static String withCrLf (String string) { - if (string == null) { + + /* If the string is empty, return the string. */ + int length = string.length (); + if (length == 0) return string; + + /* + * Check for an LF or CR/LF and assume the rest of + * the string is formated that way. This will not + * work if the string contains mixed delimiters. + */ + int i = string.indexOf ('\n', 0); + if (i == -1) return string; + if (i > 0 && string.charAt (i - 1) == '\r') { return string; } - // Replace \r\n, \r, or \n with \r\n - return string.replaceAll("(\r\n|\r|\n)", "\r\n"); + + /* + * The string is formatted with LF. Compute the + * number of lines and the size of the buffer + * needed to hold the result + */ + i++; + int count = 1; + while (i < length) { + if ((i = string.indexOf ('\n', i)) == -1) break; + count++; i++; + } + count += length; + + /* Create a new string with the CR/LF line terminator. */ + i = 0; + StringBuilder result = new StringBuilder (count); + while (i < length) { + int j = string.indexOf ('\n', i); + if (j > 0 && string.charAt(j - 1) == '\r') { + result.append(string.substring(i, j + 1)); + i = j + 1; + } else { + if (j == -1) j = length; + result.append (string.substring (i, j)); + if ((i = j) < length) { + result.append ("\r\n"); //$NON-NLS-1$ + i++; + } + } + } + return result.toString (); } static char [] withCrLf (char [] string) { - String withCrLf = withCrLf(new String(string)); - return withCrLf.toCharArray(); + /* If the string is empty, return the string. */ + int length = string.length; + if (length == 0) return string; + + /* + * Check for an LF or CR/LF and assume the rest of + * the string is formated that way. This will not + * work if the string contains mixed delimiters. + * Also, compute the number of lines. + */ + int count = 0; + for (int i = 0; i < string.length; i++) { + if (string [i] == '\n') { + count++; + if (count == 1 && i > 0 && string [i - 1] == '\r') return string; + } + } + if (count == 0) return string; + + /* + * The string is formatted with LF. + */ + count += length; + + /* Create a new string with the CR/LF line terminator. */ + char [] result = new char [count]; + for (int i = 0, j = 0; i < length && j < count; i++) { + if (string [i] == '\n') { + result [j++] = '\r'; + } + result [j++] = string [i]; + } + + return result; } static boolean isActivateShellOnForceFocus() { diff --git a/tests/org.eclipse.swt.tests.win32/JUnit Tests/org/eclipse/swt/tests/win32/widgets/Test_org_eclipse_swt_widgets_Display.java b/tests/org.eclipse.swt.tests.win32/JUnit Tests/org/eclipse/swt/tests/win32/widgets/Test_org_eclipse_swt_widgets_Display.java index a3c9ff77fc..4b3c4e0f1f 100644 --- a/tests/org.eclipse.swt.tests.win32/JUnit Tests/org/eclipse/swt/tests/win32/widgets/Test_org_eclipse_swt_widgets_Display.java +++ b/tests/org.eclipse.swt.tests.win32/JUnit Tests/org/eclipse/swt/tests/win32/widgets/Test_org_eclipse_swt_widgets_Display.java @@ -60,5 +60,8 @@ public void test_mixedLfAndCrfl() { text.setText("First Line \n second line \r\n third line"); assertEquals("First Line \r\n second line \r\n third line", text.getText()); + + text.setText("First Line \n second line \r\n third line\n"); + assertEquals("First Line \r\n second line \r\n third line\r\n", text.getText()); } }