diff --git a/src/java.desktop/share/classes/javax/swing/text/html/StyleSheet.java b/src/java.desktop/share/classes/javax/swing/text/html/StyleSheet.java
index 8ae975d98ee..ee2946b5938 100644
--- a/src/java.desktop/share/classes/javax/swing/text/html/StyleSheet.java
+++ b/src/java.desktop/share/classes/javax/swing/text/html/StyleSheet.java
@@ -1728,6 +1728,19 @@ public class StyleSheet extends StyleContext {
}
}
+ /**
+ * Proxy value to compute inherited {@code font-size}.
+ *
+ * @see Assigning
+ * property values, Cascading, and Inheritance
+ */
+ private Object fontSizeInherit() {
+ if (fontSizeInherit == null) {
+ fontSizeInherit = css.new FontSize().parseCssValue("100%");
+ }
+ return fontSizeInherit;
+ }
+
/**
* A temporary class used to hold a Vector, a StringBuffer and a
@@ -2810,6 +2823,13 @@ public class StyleSheet extends StyleContext {
}
Object doGetAttribute(Object key) {
+ if (key == CSS.Attribute.FONT_SIZE && !isDefined(key)) {
+ // CSS.FontSize represents a specified value and we need
+ // to inherit a computed value so don't resolve percentage
+ // value from parent.
+ return fontSizeInherit();
+ }
+
Object retValue = super.getAttribute(key);
if (retValue != null) {
return retValue;
@@ -3206,6 +3226,8 @@ public class StyleSheet extends StyleContext {
static final int DEFAULT_FONT_SIZE = 3;
+ private transient Object fontSizeInherit;
+
private CSS css;
/**
diff --git a/test/jdk/javax/swing/text/html/StyleSheet/TestWrongCSSFontSize.java b/test/jdk/javax/swing/text/html/StyleSheet/TestWrongCSSFontSize.java
new file mode 100644
index 00000000000..b1fadc2d621
--- /dev/null
+++ b/test/jdk/javax/swing/text/html/StyleSheet/TestWrongCSSFontSize.java
@@ -0,0 +1,156 @@
+/*
+ * Copyright (c) 2021, 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.Component;
+import java.awt.Graphics;
+import java.awt.image.BufferedImage;
+import java.io.File;
+import java.io.IOException;
+import java.util.concurrent.atomic.AtomicReference;
+import java.util.function.Consumer;
+import javax.imageio.ImageIO;
+import javax.swing.JEditorPane;
+import javax.swing.SwingUtilities;
+import javax.swing.text.GlyphView;
+import javax.swing.text.View;
+
+/*
+ * @test
+ * @bug 8257664
+ * @summary Tests inherited font-size with parent percentage specification.
+ * @run main TestWrongCSSFontSize
+ */
+public class TestWrongCSSFontSize {
+
+ private static String text =
+ "
" +
+
+ "Foo
" +
+ "Bar
" +
+ "- Baz
" +
+ "" +
+ "" +
+ "" +
+ "" +
+ "Quux
" +
+
+ "";
+
+ private static int expectedFontSize = 21;
+ private static int expectedAssertions = 8;
+
+ private JEditorPane editor;
+
+ public void setUp() {
+ editor = new JEditorPane();
+ editor.setContentType("text/html");
+ editor.setText(text);
+ editor.setSize(editor.getPreferredSize()); // layout
+ }
+
+ public void run() {
+ int count = forEachTextRun(editor.getUI()
+ .getRootView(editor), this::assertFontSize);
+ if (count != expectedAssertions) {
+ throw new AssertionError("assertion count expected ["
+ + expectedAssertions + "] but found [" + count + "]");
+ }
+ }
+
+ private int forEachTextRun(View view, Consumer action) {
+ int tested = 0;
+ for (int i = 0; i < view.getViewCount(); i++) {
+ View child = view.getView(i);
+ if (child instanceof GlyphView) {
+ if (child.getElement()
+ .getAttributes().getAttribute("CR") == Boolean.TRUE) {
+ continue;
+ }
+ action.accept((GlyphView) child);
+ tested += 1;
+ } else {
+ tested += forEachTextRun(child, action);
+ }
+ }
+ return tested;
+ }
+
+ private void assertFontSize(GlyphView child) {
+ printSource(child);
+ int actualFontSize = child.getFont().getSize();
+ if (actualFontSize != expectedFontSize) {
+ throw new AssertionError("font size expected ["
+ + expectedFontSize + "] but found [" + actualFontSize +"]");
+ }
+ }
+
+ private void printSource(View textRun) {
+ try {
+ editor.getEditorKit().write(System.out,
+ editor.getDocument(), textRun.getStartOffset(),
+ textRun.getEndOffset() - textRun.getStartOffset());
+ } catch (Exception e) {
+ e.printStackTrace();
+ }
+ }
+
+ private static void captureImage(Component comp, String path) {
+ try {
+ BufferedImage capture = new BufferedImage(comp.getWidth(),
+ comp.getHeight(), BufferedImage.TYPE_INT_ARGB);
+ Graphics g = capture.getGraphics();
+ comp.paint(g);
+ g.dispose();
+
+ ImageIO.write(capture, "png", new File(path));
+ } catch (IOException e) {
+ e.printStackTrace();
+ }
+ }
+
+ public static void main(String[] args) throws Throwable {
+ TestWrongCSSFontSize test = new TestWrongCSSFontSize();
+ AtomicReference failure = new AtomicReference<>();
+ SwingUtilities.invokeAndWait(() -> {
+ try {
+ test.setUp();
+ test.run();
+ } catch (Throwable e) {
+ failure.set(e);
+ } finally {
+ if (args.length == 1) {
+ captureImage(test.editor, args[0]);
+ }
+ }
+ });
+ if (failure.get() != null) {
+ throw failure.get();
+ }
+ }
+
+}