From 55dc1f37ab6f77cf16e3678519fb0667fe927a1a Mon Sep 17 00:00:00 2001 From: Deepika Udayagiri Date: Mon, 12 Aug 2024 14:24:06 +0530 Subject: [PATCH] Corrected TextLayout to produce right number of lines when '\r\n' sequence comes. Added some junit tests as well. Updated getStyle() and setStyle() accordingly. Added failing and success junits again. Fixes #184 --- .../org/eclipse/swt/graphics/TextLayout.java | 44 +++++++--- ...t_org_eclipse_swt_graphics_TextLayout.java | 81 +++++++++++++++++++ .../Issue184_CarriageReturnHandled.java | 69 ++++++++++++++++ 3 files changed, 183 insertions(+), 11 deletions(-) create mode 100644 tests/org.eclipse.swt.tests/ManualTests/org/eclipse/swt/tests/manual/Issue184_CarriageReturnHandled.java diff --git a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/graphics/TextLayout.java b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/graphics/TextLayout.java index 0db4dc40e3a..9f43d7edfaa 100644 --- a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/graphics/TextLayout.java +++ b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/graphics/TextLayout.java @@ -42,7 +42,7 @@ */ public final class TextLayout extends Resource { Font font; - String text, segmentsText; + String text, segmentsText, originalText; int lineSpacingInPoints; int ascent, descent; int alignment; @@ -309,7 +309,7 @@ public TextLayout (Device device) { styles[0] = new StyleItem(); styles[1] = new StyleItem(); stylesCount = 2; - text = ""; //$NON-NLS-1$ + text = originalText = ""; //$NON-NLS-1$ long[] ppv = new long[1]; OS.OleInitialize(0); if (COM.CoCreateInstance(COM.CLSID_CMultiLanguage, 0, COM.CLSCTX_INPROC_SERVER, COM.IID_IMLangFontLink2, ppv) == OS.S_OK) { @@ -2724,8 +2724,16 @@ private int getScaledVerticalIndent() { */ public TextStyle getStyle (int offset) { checkLayout(); - int length = text.length(); + int length = originalText.length(); if (!(0 <= offset && offset < length)) SWT.error(SWT.ERROR_INVALID_RANGE); + int crCount = 0, subStringLength = originalText.subSequence(0, offset+1).length(); + for (int crIndex = 0; crIndex < subStringLength; crIndex++) { + if (originalText.charAt(crIndex) == '\r') { + crCount++; + } + } + offset = offset - crCount; + for (int i=1; i offset) { return styles[i - 1].style; @@ -2956,15 +2964,11 @@ StyleItem[] merge (long items, int itemCount) { linkBefore = false; } char ch = segmentsText.charAt(start); - switch (ch) { - case '\r': - case '\n': - item.lineBreak = true; - break; - case '\t': - item.tab = true; - break; + if (ch == '\r' && start + 1 < end) { + ch = segmentsText.charAt(start + 1); } + item.lineBreak = ch == '\n'; + item.tab = ch == '\t'; if (itemLimit == -1) { nextItemIndex = itemIndex + 1; OS.MoveMemory(scriptItem, items + nextItemIndex * SCRIPT_ITEM.sizeof, SCRIPT_ITEM.sizeof); @@ -3455,6 +3459,22 @@ public void setStyle (TextStyle style, int start, int end) { int length = text.length(); if (length == 0) return; if (start > end) return; + + int startCount = 0; + int endCount = 0; + for (int crIndex = originalText.indexOf('\r', 0); crIndex >= 0; crIndex = originalText.indexOf('\r', crIndex + 1)) { + if (crIndex < start) { + ++startCount; + } else if (crIndex <= end) { + ++endCount; + } else { + break; + } + } + endCount = endCount + startCount; + start -= startCount; + end -= endCount; + start = Math.min(Math.max(0, start), length - 1); end = Math.min(Math.max(0, end), length - 1); int low = -1; @@ -3568,6 +3588,8 @@ public void setTabs (int[] tabs) { */ public void setText (String text) { checkLayout(); + this.originalText = text; + text = text.replace("\r", ""); if (text == null) SWT.error(SWT.ERROR_NULL_ARGUMENT); if (text.equals(this.text)) return; freeRuns(); diff --git a/tests/org.eclipse.swt.tests/JUnit Tests/org/eclipse/swt/tests/junit/Test_org_eclipse_swt_graphics_TextLayout.java b/tests/org.eclipse.swt.tests/JUnit Tests/org/eclipse/swt/tests/junit/Test_org_eclipse_swt_graphics_TextLayout.java index 9fb04790187..47870277835 100644 --- a/tests/org.eclipse.swt.tests/JUnit Tests/org/eclipse/swt/tests/junit/Test_org_eclipse_swt_graphics_TextLayout.java +++ b/tests/org.eclipse.swt.tests/JUnit Tests/org/eclipse/swt/tests/junit/Test_org_eclipse_swt_graphics_TextLayout.java @@ -547,6 +547,60 @@ public void test_setStyle() { assertNull(layout.getStyle(4)); assertNull(layout.getStyle(5)); layout.dispose(); + + layout = new TextLayout (display); + layout.setText("\rabc\ndef\r\rghi\n\njkl\r\nmno\r\n\r\npqr\r"); + layout.setStyle(s1, 0, 5);//abcd + layout.setStyle (s2, 7, 10);//fg + layout.setStyle (s3, 11, 18);//hijkl + layout.setStyle (s4, 19, 24);//mno + assertEquals(s1, layout.getStyle(0)); + assertEquals(s1, layout.getStyle(1)); + assertEquals(s1, layout.getStyle(3)); + assertEquals(s1, layout.getStyle(4)); + assertEquals(s1, layout.getStyle(5)); + + assertEquals(s2, layout.getStyle(7)); + assertEquals(s2, layout.getStyle(8)); + assertEquals(s2, layout.getStyle(9)); + assertEquals(s2, layout.getStyle(10)); + + assertEquals(s3, layout.getStyle(12)); + assertEquals(s3, layout.getStyle(13)); + assertEquals(s3, layout.getStyle(14)); + assertEquals(s3, layout.getStyle(15)); + + assertEquals(s4, layout.getStyle(19)); + assertEquals(s4, layout.getStyle(20)); + assertEquals(s4, layout.getStyle(22)); + assertEquals(s4, layout.getStyle(23)); + assertEquals(s4, layout.getStyle(24)); + + layout.setStyle (null, 0, 3);//abc + assertNull(layout.getStyle(0)); + + layout.setStyle(s1, 0, 9);//abcdef + assertEquals(s1, layout.getStyle(8)); + + layout.setStyle (s2, 1, 24);//abcdefghijklmno + assertEquals(s2, layout.getStyle(7)); + assertEquals(s2, layout.getStyle(8)); + assertEquals(s2, layout.getStyle(14)); + assertEquals(s2, layout.getStyle(18)); + assertEquals(s2, layout.getStyle(22)); + assertEquals(s2, layout.getStyle(23)); + assertEquals(s2, layout.getStyle(24)); + + layout.setStyle (s3, 0, 30);//abcdefghijklmnopqr + assertEquals(s3, layout.getStyle(0)); + assertEquals(s3, layout.getStyle(1)); + assertEquals(s3, layout.getStyle(30)); + + layout.setStyle (s4, 1, 30);//abcdefghijklmnopqr + assertEquals(s4, layout.getStyle(1)); + assertEquals(s4, layout.getStyle(29)); + assertEquals(s4, layout.getStyle(30)); + layout.dispose(); } @Test @@ -1219,4 +1273,31 @@ public void test_Bug579335_win32_StyledText_LongLine() { font.dispose(); } } + +@Test +public void test_getLineCount() { + // Skipping this test for GTK and Cocoa as this pr #1320 applies only for + // windows platform. + // Without pr #1320(master) : Number of lines: 11 (for Cocoa it is : 10) + // Expected : Number of lines: 7 (for Cocoa it is : 10) + TextLayout layout = new TextLayout(display); + try { + String text = "\rabc\ndef\r\rghi\n\njkl\r\nmno\r\n\r\npqr\r"; + layout.setText(text); + int lineCount = layout.getLineCount(); + + int expected = 0; + if (SwtTestUtil.isWindows) { + expected = 7; + } else if (SwtTestUtil.isCocoa) { + expected = 10; + } else if (SwtTestUtil.isGTK) { + expected = 11; + } + + assertEquals(expected, lineCount); + } finally { + layout.dispose(); + } +} } diff --git a/tests/org.eclipse.swt.tests/ManualTests/org/eclipse/swt/tests/manual/Issue184_CarriageReturnHandled.java b/tests/org.eclipse.swt.tests/ManualTests/org/eclipse/swt/tests/manual/Issue184_CarriageReturnHandled.java new file mode 100644 index 00000000000..12d494795d6 --- /dev/null +++ b/tests/org.eclipse.swt.tests/ManualTests/org/eclipse/swt/tests/manual/Issue184_CarriageReturnHandled.java @@ -0,0 +1,69 @@ +/******************************************************************************* + * Copyright (c) 2024 IBM and others. + * + * This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + *******************************************************************************/ +package org.eclipse.swt.tests.manual; + +import org.eclipse.swt.*; +import org.eclipse.swt.graphics.*; +import org.eclipse.swt.widgets.*; + +public class Issue184_CarriageReturnHandled { + + private static FontData[] getFontData(Font font, int style) { + FontData[] fontDatas = font.getFontData(); + for (FontData fontData : fontDatas) { + fontData.setStyle(style); + } + return fontDatas; + } + + public static void main(String[] args) { + Display display = new Display(); + final Shell shell = new Shell(display, SWT.SHELL_TRIM | SWT.DOUBLE_BUFFERED); + shell.setText("Underline, Strike Out"); + String text1 = "\r\nde\nep\rika\r\rudaya\n\ngiri\r"; + + FontData[] fontData = getFontData(shell.getFont(), SWT.BOLD); + Font font = new Font(shell.getDisplay(), fontData); + + FontData[] fontData1 = getFontData(shell.getFont(), SWT.ITALIC | SWT.BOLD); + Font font1 = new Font(shell.getDisplay(), fontData1); + + FontData[] fontData2 = getFontData(shell.getFont(), SWT.BOLD); + Font font2 = new Font(shell.getDisplay(), fontData2); + + final TextLayout layout = new TextLayout(display); + layout.setText(text1); + + TextStyle style1 = new TextStyle(font, null, null); + layout.setStyle(style1, 3, 7); // eep in bold + + TextStyle style2 = new TextStyle(font1, null, null); + layout.setStyle(style2, 12, 18); // udaya in bold + + TextStyle style3 = new TextStyle(font2, null, null); + layout.setStyle(style3, 21, 24); // iri in bold + + shell.addListener(SWT.Paint, event -> { + Point point = new Point(10, 10); + int width = shell.getClientArea().width - 2 * point.x; + layout.setWidth(width); + layout.draw(event.gc, point.x, point.y); + }); + shell.setSize(400, 300); + shell.open(); + while (!shell.isDisposed()) { + if (!display.readAndDispatch()) + display.sleep(); + } + layout.dispose(); + display.dispose(); + } +}