8326734: text-decoration applied to <span> lost when mixed with <u> or <s>
8325620: HTMLReader uses ConvertAction instead of specified CharacterAction for <b>, <i>, <u> Reviewed-by: honkar, prr
This commit is contained in:
parent
c2cca2ab44
commit
cd3e4c0366
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1998, 2023, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1998, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -843,6 +843,18 @@ public class CSS implements Serializable {
|
|||||||
return r != null ? r : conv.parseCssValue(key.getDefaultValue());
|
return r != null ? r : conv.parseCssValue(key.getDefaultValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Object mergeTextDecoration(String value) {
|
||||||
|
boolean underline = value.contains("underline");
|
||||||
|
boolean strikeThrough = value.contains("line-through");
|
||||||
|
if (!underline && !strikeThrough) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
String newValue = underline && strikeThrough
|
||||||
|
? "underline,line-through"
|
||||||
|
: (underline ? "underline" : "line-through");
|
||||||
|
return new StringValue().parseCssValue(newValue);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Maps from a StyleConstants to a CSS Attribute.
|
* Maps from a StyleConstants to a CSS Attribute.
|
||||||
*/
|
*/
|
||||||
|
@ -2499,7 +2499,7 @@ public class HTMLDocument extends DefaultStyledDocument {
|
|||||||
tagMap.put(HTML.Tag.SCRIPT, ha);
|
tagMap.put(HTML.Tag.SCRIPT, ha);
|
||||||
tagMap.put(HTML.Tag.SELECT, fa);
|
tagMap.put(HTML.Tag.SELECT, fa);
|
||||||
tagMap.put(HTML.Tag.SMALL, ca);
|
tagMap.put(HTML.Tag.SMALL, ca);
|
||||||
tagMap.put(HTML.Tag.SPAN, ca);
|
tagMap.put(HTML.Tag.SPAN, new ConvertSpanAction());
|
||||||
tagMap.put(HTML.Tag.STRIKE, conv);
|
tagMap.put(HTML.Tag.STRIKE, conv);
|
||||||
tagMap.put(HTML.Tag.S, conv);
|
tagMap.put(HTML.Tag.S, conv);
|
||||||
tagMap.put(HTML.Tag.STRONG, ca);
|
tagMap.put(HTML.Tag.STRONG, ca);
|
||||||
@ -3423,11 +3423,43 @@ public class HTMLDocument extends DefaultStyledDocument {
|
|||||||
if (styleAttributes != null) {
|
if (styleAttributes != null) {
|
||||||
charAttr.addAttributes(styleAttributes);
|
charAttr.addAttributes(styleAttributes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
convertAttributes(t, attr);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void end(HTML.Tag t) {
|
public void end(HTML.Tag t) {
|
||||||
popCharacterStyle();
|
popCharacterStyle();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts HTML tags to CSS attributes.
|
||||||
|
* @param t the current HTML tag
|
||||||
|
* @param attr the attributes of the HTML tag
|
||||||
|
*/
|
||||||
|
void convertAttributes(HTML.Tag t, MutableAttributeSet attr) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final class ConvertSpanAction extends CharacterAction {
|
||||||
|
@Override
|
||||||
|
void convertAttributes(HTML.Tag t, MutableAttributeSet attr) {
|
||||||
|
Object newDecoration = attr.getAttribute(CSS.Attribute.TEXT_DECORATION);
|
||||||
|
Object previousDecoration =
|
||||||
|
charAttrStack.peek()
|
||||||
|
.getAttribute(CSS.Attribute.TEXT_DECORATION);
|
||||||
|
|
||||||
|
if (newDecoration != null
|
||||||
|
&& !"none".equals(newDecoration.toString())
|
||||||
|
&& previousDecoration != null
|
||||||
|
&& !"none".equals(previousDecoration.toString())) {
|
||||||
|
StyleSheet sheet = getStyleSheet();
|
||||||
|
sheet.addCSSAttribute(charAttr,
|
||||||
|
CSS.Attribute.TEXT_DECORATION,
|
||||||
|
CSS.mergeTextDecoration(newDecoration + ","
|
||||||
|
+ previousDecoration)
|
||||||
|
.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -3435,35 +3467,9 @@ public class HTMLDocument extends DefaultStyledDocument {
|
|||||||
* mappings that have a corresponding StyleConstants
|
* mappings that have a corresponding StyleConstants
|
||||||
* and CSS mapping. The conversion is to CSS attributes.
|
* and CSS mapping. The conversion is to CSS attributes.
|
||||||
*/
|
*/
|
||||||
class ConvertAction extends TagAction {
|
final class ConvertAction extends CharacterAction {
|
||||||
|
@Override
|
||||||
public void start(HTML.Tag t, MutableAttributeSet attr) {
|
void convertAttributes(HTML.Tag t, MutableAttributeSet attr) {
|
||||||
pushCharacterStyle();
|
|
||||||
if (!foundInsertTag) {
|
|
||||||
// Note that the third argument should really be based off
|
|
||||||
// inParagraph and impliedP. If we're wrong (that is
|
|
||||||
// insertTagDepthDelta shouldn't be changed), we'll end up
|
|
||||||
// removing an extra EndSpec, which won't matter anyway.
|
|
||||||
boolean insert = canInsertTag(t, attr, false);
|
|
||||||
if (foundInsertTag) {
|
|
||||||
if (!inParagraph) {
|
|
||||||
inParagraph = impliedP = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!insert) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (attr.isDefined(IMPLIED)) {
|
|
||||||
attr.removeAttribute(IMPLIED);
|
|
||||||
}
|
|
||||||
if (styleAttributes != null) {
|
|
||||||
charAttr.addAttributes(styleAttributes);
|
|
||||||
}
|
|
||||||
// We also need to add attr, otherwise we lose custom
|
|
||||||
// attributes, including class/id for style lookups, and
|
|
||||||
// further confuse style lookup (doesn't have tag).
|
|
||||||
charAttr.addAttribute(t, attr.copyAttributes());
|
|
||||||
StyleSheet sheet = getStyleSheet();
|
StyleSheet sheet = getStyleSheet();
|
||||||
if (t == HTML.Tag.B) {
|
if (t == HTML.Tag.B) {
|
||||||
sheet.addCSSAttribute(charAttr, CSS.Attribute.FONT_WEIGHT, "bold");
|
sheet.addCSSAttribute(charAttr, CSS.Attribute.FONT_WEIGHT, "bold");
|
||||||
@ -3504,11 +3510,6 @@ public class HTMLDocument extends DefaultStyledDocument {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void end(HTML.Tag t) {
|
|
||||||
popCharacterStyle();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class AnchorAction extends CharacterAction {
|
class AnchorAction extends CharacterAction {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2001, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2001, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -24,9 +24,16 @@
|
|||||||
*/
|
*/
|
||||||
package javax.swing.text.html;
|
package javax.swing.text.html;
|
||||||
|
|
||||||
import javax.swing.text.*;
|
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
import java.util.*;
|
import java.util.Arrays;
|
||||||
|
import java.util.Enumeration;
|
||||||
|
import java.util.NoSuchElementException;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import javax.swing.text.AttributeSet;
|
||||||
|
import javax.swing.text.MutableAttributeSet;
|
||||||
|
import javax.swing.text.SimpleAttributeSet;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An implementation of <code>AttributeSet</code> that can multiplex
|
* An implementation of <code>AttributeSet</code> that can multiplex
|
||||||
@ -196,8 +203,9 @@ class MuxingAttributeSet implements AttributeSet, Serializable {
|
|||||||
* @see AttributeSet#getAttribute
|
* @see AttributeSet#getAttribute
|
||||||
*/
|
*/
|
||||||
public Object getAttribute(Object key) {
|
public Object getAttribute(Object key) {
|
||||||
AttributeSet[] as = getAttributes();
|
final AttributeSet[] as = getAttributes();
|
||||||
int n = as.length;
|
final int n = as.length;
|
||||||
|
if (key != CSS.Attribute.TEXT_DECORATION) {
|
||||||
for (int i = 0; i < n; i++) {
|
for (int i = 0; i < n; i++) {
|
||||||
Object o = as[i].getAttribute(key);
|
Object o = as[i].getAttribute(key);
|
||||||
if (o != null) {
|
if (o != null) {
|
||||||
@ -207,6 +215,14 @@ class MuxingAttributeSet implements AttributeSet, Serializable {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String values = Arrays.stream(as)
|
||||||
|
.map(a -> a.getAttribute(key))
|
||||||
|
.filter(Objects::nonNull)
|
||||||
|
.map(Object::toString)
|
||||||
|
.collect(Collectors.joining(","));
|
||||||
|
return CSS.mergeTextDecoration(values);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the names of all attributes.
|
* Gets the names of all attributes.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -24,17 +24,53 @@
|
|||||||
*/
|
*/
|
||||||
package javax.swing.text.html;
|
package javax.swing.text.html;
|
||||||
|
|
||||||
import sun.swing.SwingUtilities2;
|
import java.awt.Color;
|
||||||
import java.util.*;
|
import java.awt.Component;
|
||||||
import java.awt.*;
|
import java.awt.Container;
|
||||||
import java.io.*;
|
import java.awt.Font;
|
||||||
import java.net.*;
|
import java.awt.FontMetrics;
|
||||||
|
import java.awt.Graphics;
|
||||||
|
import java.awt.Graphics2D;
|
||||||
|
import java.awt.Insets;
|
||||||
|
import java.awt.Rectangle;
|
||||||
|
import java.awt.RenderingHints;
|
||||||
|
import java.awt.Shape;
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.io.StringReader;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.EmptyStackException;
|
||||||
|
import java.util.Enumeration;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Hashtable;
|
||||||
|
import java.util.Stack;
|
||||||
|
import java.util.StringTokenizer;
|
||||||
|
import java.util.Vector;
|
||||||
|
|
||||||
import javax.swing.Icon;
|
import javax.swing.Icon;
|
||||||
import javax.swing.ImageIcon;
|
import javax.swing.ImageIcon;
|
||||||
import javax.swing.UIManager;
|
import javax.swing.UIManager;
|
||||||
import javax.swing.border.*;
|
import javax.swing.border.BevelBorder;
|
||||||
|
import javax.swing.border.Border;
|
||||||
import javax.swing.event.ChangeListener;
|
import javax.swing.event.ChangeListener;
|
||||||
import javax.swing.text.*;
|
import javax.swing.text.AttributeSet;
|
||||||
|
import javax.swing.text.Document;
|
||||||
|
import javax.swing.text.Element;
|
||||||
|
import javax.swing.text.MutableAttributeSet;
|
||||||
|
import javax.swing.text.SimpleAttributeSet;
|
||||||
|
import javax.swing.text.Style;
|
||||||
|
import javax.swing.text.StyleConstants;
|
||||||
|
import javax.swing.text.StyleContext;
|
||||||
|
import javax.swing.text.StyledDocument;
|
||||||
|
import javax.swing.text.View;
|
||||||
|
|
||||||
|
import sun.swing.SwingUtilities2;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Support for defining the visual characteristics of
|
* Support for defining the visual characteristics of
|
||||||
@ -2817,10 +2853,31 @@ public class StyleSheet extends StyleContext {
|
|||||||
return doGetAttribute(key);
|
return doGetAttribute(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Merges the current value of the 'text-decoration' property
|
||||||
|
* with the value from parent.
|
||||||
|
*/
|
||||||
|
private Object getTextDecoration(Object value) {
|
||||||
|
AttributeSet parent = getResolveParent();
|
||||||
|
if (parent == null) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
Object parentValue = parent.getAttribute(CSS.Attribute.TEXT_DECORATION);
|
||||||
|
return parentValue == null
|
||||||
|
? value
|
||||||
|
: CSS.mergeTextDecoration(value + "," + parentValue);
|
||||||
|
}
|
||||||
|
|
||||||
Object doGetAttribute(Object key) {
|
Object doGetAttribute(Object key) {
|
||||||
Object retValue = super.getAttribute(key);
|
Object retValue = super.getAttribute(key);
|
||||||
if (retValue != null) {
|
if (retValue != null) {
|
||||||
|
if (key != CSS.Attribute.TEXT_DECORATION) {
|
||||||
return retValue;
|
return retValue;
|
||||||
|
} else {
|
||||||
|
// Merge current value with parent
|
||||||
|
return getTextDecoration(retValue);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (key == CSS.Attribute.FONT_SIZE) {
|
if (key == CSS.Attribute.FONT_SIZE) {
|
||||||
|
135
test/jdk/javax/swing/text/html/HTMLDocument/HTMLStrikeOnly.java
Normal file
135
test/jdk/javax/swing/text/html/HTMLDocument/HTMLStrikeOnly.java
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.awt.Dimension;
|
||||||
|
import java.awt.Graphics;
|
||||||
|
import java.awt.image.BufferedImage;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.imageio.ImageIO;
|
||||||
|
import javax.swing.JEditorPane;
|
||||||
|
import javax.swing.text.View;
|
||||||
|
import javax.swing.text.html.CSS;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8326734
|
||||||
|
* @summary Tests different combinations of setting 'line-through'
|
||||||
|
* @run main HTMLStrikeOnly
|
||||||
|
*/
|
||||||
|
public class HTMLStrikeOnly {
|
||||||
|
private static final String HTML = """
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>line-through</title>
|
||||||
|
<style>
|
||||||
|
.lineThrough { text-decoration: line-through }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p><s><span style='text-decoration: line-through'>line-through?</span></s></p>
|
||||||
|
<p><strike><span style='text-decoration: line-through'>line-through?</span></strike></p>
|
||||||
|
<p><span style='text-decoration: line-through'><s>line-through?</s></span></p>
|
||||||
|
<p><span style='text-decoration: line-through'><strike>line-through?</strike></span></p>
|
||||||
|
|
||||||
|
<p><s><span class="lineThrough">line-through?</span></s></p>
|
||||||
|
<p><strike><span class="lineThrough">line-through?</span></strike></p>
|
||||||
|
<p><span class="lineThrough"><s>line-through?</s></span></p>
|
||||||
|
<p><span class="lineThrough"><strike>line-through?</strike></span></p>
|
||||||
|
|
||||||
|
<p style='text-decoration: line-through'><s>line-through?</s></p>
|
||||||
|
<p style='text-decoration: line-through'><strike>line-through?</strike></p>
|
||||||
|
<p style='text-decoration: line-through'><span style='text-decoration: line-through'>line-through?</span></p>
|
||||||
|
|
||||||
|
<p class="lineThrough"><s>line-through</s></p>
|
||||||
|
<p class="lineThrough"><strike>line-through</strike></p>
|
||||||
|
<p class="lineThrough"><span style='text-decoration: line-through'>line-through</span></p>
|
||||||
|
<p class="lineThrough"><span class="lineThrough">line-through</span></p>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
""";
|
||||||
|
public static void main(String[] args) {
|
||||||
|
final JEditorPane html = new JEditorPane("text/html", HTML);
|
||||||
|
html.setEditable(false);
|
||||||
|
|
||||||
|
final Dimension size = html.getPreferredSize();
|
||||||
|
html.setSize(size);
|
||||||
|
|
||||||
|
BufferedImage image = new BufferedImage(size.width, size.height,
|
||||||
|
BufferedImage.TYPE_INT_RGB);
|
||||||
|
Graphics g = image.createGraphics();
|
||||||
|
// Paint the editor pane to ensure all views are created
|
||||||
|
html.paint(g);
|
||||||
|
g.dispose();
|
||||||
|
|
||||||
|
int errorCount = 0;
|
||||||
|
String firstError = null;
|
||||||
|
|
||||||
|
System.out.println("----- Views -----");
|
||||||
|
final View bodyView = html.getUI()
|
||||||
|
.getRootView(html)
|
||||||
|
.getView(1)
|
||||||
|
.getView(1);
|
||||||
|
for (int i = 0; i < bodyView.getViewCount(); i++) {
|
||||||
|
View pView = bodyView.getView(i);
|
||||||
|
View contentView = getContentView(pView);
|
||||||
|
|
||||||
|
String decoration =
|
||||||
|
contentView.getAttributes()
|
||||||
|
.getAttribute(CSS.Attribute.TEXT_DECORATION)
|
||||||
|
.toString();
|
||||||
|
|
||||||
|
System.out.println(i + ": " + decoration);
|
||||||
|
if (!decoration.contains("line-through")
|
||||||
|
|| decoration.contains("underline")) {
|
||||||
|
errorCount++;
|
||||||
|
if (firstError == null) {
|
||||||
|
firstError = "Line " + i + ": " + decoration;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errorCount > 0) {
|
||||||
|
saveImage(image);
|
||||||
|
throw new RuntimeException(errorCount + " error(s) found, "
|
||||||
|
+ "the first one: " + firstError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static View getContentView(View parent) {
|
||||||
|
View view = parent.getView(0);
|
||||||
|
return view.getViewCount() > 0
|
||||||
|
? getContentView(view)
|
||||||
|
: view;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void saveImage(BufferedImage image) {
|
||||||
|
try {
|
||||||
|
ImageIO.write(image, "png",
|
||||||
|
new File("html.png"));
|
||||||
|
} catch (IOException ignored) { }
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,157 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.awt.Dimension;
|
||||||
|
import java.awt.Graphics;
|
||||||
|
import java.awt.image.BufferedImage;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.imageio.ImageIO;
|
||||||
|
import javax.swing.JEditorPane;
|
||||||
|
import javax.swing.text.View;
|
||||||
|
import javax.swing.text.html.CSS;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8323801 8326734
|
||||||
|
* @summary Tests different combination of 'underline' and 'line-through';
|
||||||
|
* the text should render with both 'underline' and 'line-through'.
|
||||||
|
* @run main HTMLTextDecoration
|
||||||
|
*/
|
||||||
|
public final class HTMLTextDecoration {
|
||||||
|
private static final String HTML = """
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>underline + line-through text</title>
|
||||||
|
<style>
|
||||||
|
.underline { text-decoration: underline }
|
||||||
|
.lineThrough { text-decoration: line-through }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p><u><span style='text-decoration: line-through'>underline + line-through?</span></u></p>
|
||||||
|
<p><s><span style='text-decoration: underline'>underline + line-through?</span></s></p>
|
||||||
|
<p><strike><span style='text-decoration: underline'>underline + line-through?</span></strike></p>
|
||||||
|
|
||||||
|
<p><span style='text-decoration: line-through'><u>underline + line-through?</u></span></p>
|
||||||
|
<p><span style='text-decoration: underline'><s>underline + line-through?</s></span></p>
|
||||||
|
<p><span style='text-decoration: underline'><strike>underline + line-through?</strike></span></p>
|
||||||
|
|
||||||
|
<p><span style='text-decoration: line-through'><span style='text-decoration: underline'>underline + line-through?</span></span></p>
|
||||||
|
<p><span style='text-decoration: underline'><span style='text-decoration: line-through'>underline + line-through?</span></span></p>
|
||||||
|
|
||||||
|
<p style='text-decoration: line-through'><u>underline + line-through?</u></p>
|
||||||
|
<p style='text-decoration: underline'><s>underline + line-through?</s></p>
|
||||||
|
<p style='text-decoration: underline'><strike>underline + line-through?</strike></p>
|
||||||
|
|
||||||
|
<p style='text-decoration: line-through'><span style='text-decoration: underline'>underline + line-through?</span></p>
|
||||||
|
<p style='text-decoration: underline'><span style='text-decoration: line-through'>underline + line-through?</span></p>
|
||||||
|
|
||||||
|
<p class="underline"><span class="lineThrough">underline + line-through?</span></p>
|
||||||
|
<p class="underline"><s>underline + line-through?</s></p>
|
||||||
|
<p class="underline"><strike>underline + line-through?</strike></p>
|
||||||
|
|
||||||
|
<p class="lineThrough"><span class="underline">underline + line-through?</span></p>
|
||||||
|
<p class="lineThrough"><u>underline + line-through?</u></p>
|
||||||
|
|
||||||
|
<div class="underline"><span class="lineThrough">underline + line-through?</span></div>
|
||||||
|
<div class="underline"><s>underline + line-through?</s></div>
|
||||||
|
<div class="underline"><strike>underline + line-through?</strike></div>
|
||||||
|
|
||||||
|
<div class="lineThrough"><span class="underline">underline + line-through?</span></div>
|
||||||
|
<div class="lineThrough"><u>underline + line-through?</u></div>
|
||||||
|
|
||||||
|
<div class="underline"><p class="lineThrough">underline + line-through?</p></div>
|
||||||
|
<div class="lineThrough"><p class="underline">underline + line-through?</p></div>
|
||||||
|
|
||||||
|
<div class="underline"><div class="lineThrough">underline + line-through?</div></div>
|
||||||
|
<div class="lineThrough"><div class="underline">underline + line-through?</div></div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
""";
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
final JEditorPane html = new JEditorPane("text/html", HTML);
|
||||||
|
html.setEditable(false);
|
||||||
|
|
||||||
|
final Dimension size = html.getPreferredSize();
|
||||||
|
html.setSize(size);
|
||||||
|
|
||||||
|
BufferedImage image = new BufferedImage(size.width, size.height,
|
||||||
|
BufferedImage.TYPE_INT_RGB);
|
||||||
|
Graphics g = image.createGraphics();
|
||||||
|
// Paint the editor pane to ensure all views are created
|
||||||
|
html.paint(g);
|
||||||
|
g.dispose();
|
||||||
|
|
||||||
|
int errorCount = 0;
|
||||||
|
String firstError = null;
|
||||||
|
|
||||||
|
System.out.println("----- Views -----");
|
||||||
|
final View bodyView = html.getUI()
|
||||||
|
.getRootView(html)
|
||||||
|
.getView(1)
|
||||||
|
.getView(1);
|
||||||
|
for (int i = 0; i < bodyView.getViewCount(); i++) {
|
||||||
|
View pView = bodyView.getView(i);
|
||||||
|
View contentView = getContentView(pView);
|
||||||
|
|
||||||
|
String decoration =
|
||||||
|
contentView.getAttributes()
|
||||||
|
.getAttribute(CSS.Attribute.TEXT_DECORATION)
|
||||||
|
.toString();
|
||||||
|
|
||||||
|
System.out.println(i + ": " + decoration);
|
||||||
|
if (!decoration.contains("underline")
|
||||||
|
|| !decoration.contains("line-through")) {
|
||||||
|
errorCount++;
|
||||||
|
if (firstError == null) {
|
||||||
|
firstError = "Line " + i + ": " + decoration;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errorCount > 0) {
|
||||||
|
saveImage(image);
|
||||||
|
throw new RuntimeException(errorCount + " error(s) found, "
|
||||||
|
+ "the first one: " + firstError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static View getContentView(View parent) {
|
||||||
|
View view = parent.getView(0);
|
||||||
|
return view.getViewCount() > 0
|
||||||
|
? getContentView(view)
|
||||||
|
: view;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void saveImage(BufferedImage image) {
|
||||||
|
try {
|
||||||
|
ImageIO.write(image, "png",
|
||||||
|
new File("html.png"));
|
||||||
|
} catch (IOException ignored) { }
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,129 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.awt.Dimension;
|
||||||
|
import java.awt.Graphics;
|
||||||
|
import java.awt.image.BufferedImage;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.imageio.ImageIO;
|
||||||
|
import javax.swing.JEditorPane;
|
||||||
|
import javax.swing.text.View;
|
||||||
|
import javax.swing.text.html.CSS;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8326734
|
||||||
|
* @summary Tests different combinations of setting 'underline'
|
||||||
|
* @run main HTMLUnderlineOnly
|
||||||
|
*/
|
||||||
|
public class HTMLUnderlineOnly {
|
||||||
|
private static final String HTML = """
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>underline</title>
|
||||||
|
<style>
|
||||||
|
.underline { text-decoration: underline }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p><u><span style='text-decoration: underline'>underline?</span></u></p>
|
||||||
|
<p><span style='text-decoration: underline'><u>underline?</u></span></p>
|
||||||
|
|
||||||
|
<p><u><span class="underline">underline?</span></u></p>
|
||||||
|
<p><span class="underline"><u>underline?</u></span></p>
|
||||||
|
|
||||||
|
<p style='text-decoration: underline'><u>underline?</u></p>
|
||||||
|
<p style='text-decoration: underline'><span style='text-decoration: underline'>underline?</span></p>
|
||||||
|
|
||||||
|
<p class="underline"><u>underline</u></p>
|
||||||
|
<p class="underline"><span style='text-decoration: underline'>underline</span></p>
|
||||||
|
<p class="underline"><span class="underline">underline</span></p>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
""";
|
||||||
|
public static void main(String[] args) {
|
||||||
|
final JEditorPane html = new JEditorPane("text/html", HTML);
|
||||||
|
html.setEditable(false);
|
||||||
|
|
||||||
|
final Dimension size = html.getPreferredSize();
|
||||||
|
html.setSize(size);
|
||||||
|
|
||||||
|
BufferedImage image = new BufferedImage(size.width, size.height,
|
||||||
|
BufferedImage.TYPE_INT_RGB);
|
||||||
|
Graphics g = image.createGraphics();
|
||||||
|
// Paint the editor pane to ensure all views are created
|
||||||
|
html.paint(g);
|
||||||
|
g.dispose();
|
||||||
|
|
||||||
|
int errorCount = 0;
|
||||||
|
String firstError = null;
|
||||||
|
|
||||||
|
System.out.println("----- Views -----");
|
||||||
|
final View bodyView = html.getUI()
|
||||||
|
.getRootView(html)
|
||||||
|
.getView(1)
|
||||||
|
.getView(1);
|
||||||
|
for (int i = 0; i < bodyView.getViewCount(); i++) {
|
||||||
|
View pView = bodyView.getView(i);
|
||||||
|
View contentView = getContentView(pView);
|
||||||
|
|
||||||
|
String decoration =
|
||||||
|
contentView.getAttributes()
|
||||||
|
.getAttribute(CSS.Attribute.TEXT_DECORATION)
|
||||||
|
.toString();
|
||||||
|
|
||||||
|
System.out.println(i + ": " + decoration);
|
||||||
|
if (!decoration.contains("underline")
|
||||||
|
|| decoration.contains("line-through")) {
|
||||||
|
errorCount++;
|
||||||
|
if (firstError == null) {
|
||||||
|
firstError = "Line " + i + ": " + decoration;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errorCount > 0) {
|
||||||
|
saveImage(image);
|
||||||
|
throw new RuntimeException(errorCount + " error(s) found, "
|
||||||
|
+ "the first one: " + firstError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static View getContentView(View parent) {
|
||||||
|
View view = parent.getView(0);
|
||||||
|
return view.getViewCount() > 0
|
||||||
|
? getContentView(view)
|
||||||
|
: view;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void saveImage(BufferedImage image) {
|
||||||
|
try {
|
||||||
|
ImageIO.write(image, "png",
|
||||||
|
new File("html.png"));
|
||||||
|
} catch (IOException ignored) { }
|
||||||
|
}
|
||||||
|
}
|
@ -31,7 +31,7 @@ import javax.swing.text.html.HTMLEditorKit;
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* @test
|
* @test
|
||||||
* @bug 8323801
|
* @bug 8323801 8326734
|
||||||
* @summary Tests that '<u><s>' produce underlined and struck-through text
|
* @summary Tests that '<u><s>' produce underlined and struck-through text
|
||||||
*/
|
*/
|
||||||
public final class HTMLUnderlineStrike {
|
public final class HTMLUnderlineStrike {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user