8265761: Font with missed font family name is not properly printed on Windows
Reviewed-by: serb, prr
This commit is contained in:
parent
3554dc299d
commit
e9370a13b6
@ -1157,6 +1157,10 @@ public final class WPrinterJob extends RasterPrinterJob
|
||||
protected boolean setFont(String family, float size, int style,
|
||||
int rotation, float awScale) {
|
||||
|
||||
if (family.isEmpty()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
boolean didSetFont = true;
|
||||
|
||||
if (!family.equals(mLastFontFamily) ||
|
||||
|
@ -0,0 +1,289 @@
|
||||
/*
|
||||
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2021, BELLSOFT. 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.
|
||||
*/
|
||||
|
||||
/**
|
||||
* @test
|
||||
* @bug 8265761
|
||||
* @requires (os.family == "windows")
|
||||
* @summary Font with missed font family name is not properly printed on Windows
|
||||
* @run main/othervm/manual PrintFontWithMissedFontFamilyTest
|
||||
*/
|
||||
|
||||
import javax.print.PrintServiceLookup;
|
||||
import javax.swing.*;
|
||||
import java.awt.*;
|
||||
import java.awt.event.WindowAdapter;
|
||||
import java.awt.event.WindowEvent;
|
||||
import java.awt.font.FontRenderContext;
|
||||
import java.awt.font.GlyphVector;
|
||||
import java.awt.geom.AffineTransform;
|
||||
import java.awt.print.PageFormat;
|
||||
import java.awt.print.Printable;
|
||||
import java.awt.print.PrinterException;
|
||||
import java.awt.print.PrinterJob;
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
public class PrintFontWithMissedFontFamilyTest {
|
||||
|
||||
private static final String DESCRIPTION =
|
||||
" 1. Setup 'Microsoft Print to PDF' printer on Windows.\n" +
|
||||
" 2. Press Print button to print a text with a custom font to PDF.\n" +
|
||||
" 3. Choose 'Microsoft Print to PDF' on the print dialog and press OK.\n" +
|
||||
" 4. Open the PDF file.\n" +
|
||||
" Three lines with text ABCDEF and the same custom font should be printed.\n" +
|
||||
" 5. Compare the text from the pdf with three strings drawn on the right pane.\n" +
|
||||
" The pdf and the pane text should have the same font.\n" +
|
||||
" 6. If so, press PASS button, otherwise press FAIL button.\n";
|
||||
|
||||
private static final CountDownLatch testEndedSignal = new CountDownLatch(1);
|
||||
private static final int testTimeout = 300000;
|
||||
private static volatile String testFailureMsg;
|
||||
private static volatile boolean testPassed;
|
||||
private static volatile boolean testFinished;
|
||||
private static final float FONT_SIZE = 32.0f;
|
||||
private static final String TEXT = "ABCDEF";
|
||||
private static final String FONT_FILE = "SampleFontMissedFontFamily.ttf";
|
||||
private static Font customFont;
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
|
||||
SwingUtilities.invokeLater(() -> createAndShowTestDialog());
|
||||
|
||||
try {
|
||||
if (!testEndedSignal.await(testTimeout, TimeUnit.MILLISECONDS)) {
|
||||
throw new RuntimeException(String.format(
|
||||
"Test timeout '%d ms' elapsed.", testTimeout));
|
||||
}
|
||||
if (!testPassed) {
|
||||
String failureMsg = testFailureMsg;
|
||||
if ((failureMsg != null) && (!failureMsg.trim().isEmpty())) {
|
||||
throw new RuntimeException(failureMsg);
|
||||
} else {
|
||||
throw new RuntimeException("Test failed.");
|
||||
}
|
||||
}
|
||||
} catch (InterruptedException ie) {
|
||||
throw new RuntimeException(ie);
|
||||
} finally {
|
||||
testFinished = true;
|
||||
}
|
||||
}
|
||||
|
||||
private static void pass() {
|
||||
testPassed = true;
|
||||
testEndedSignal.countDown();
|
||||
}
|
||||
|
||||
private static void fail(String failureMsg) {
|
||||
testFailureMsg = failureMsg;
|
||||
testPassed = false;
|
||||
testEndedSignal.countDown();
|
||||
}
|
||||
|
||||
private static String convertMillisToTimeStr(int millis) {
|
||||
if (millis < 0) {
|
||||
return "00:00:00";
|
||||
}
|
||||
int hours = millis / 3600000;
|
||||
int minutes = (millis - hours * 3600000) / 60000;
|
||||
int seconds = (millis - hours * 3600000 - minutes * 60000) / 1000;
|
||||
return String.format("%02d:%02d:%02d", hours, minutes, seconds);
|
||||
}
|
||||
|
||||
private static void createAndShowTestDialog() {
|
||||
|
||||
final JDialog dialog = new JDialog();
|
||||
dialog.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
|
||||
dialog.addWindowListener(new WindowAdapter() {
|
||||
@Override
|
||||
public void windowClosing(WindowEvent e) {
|
||||
dialog.dispose();
|
||||
fail("Main dialog was closed.");
|
||||
}
|
||||
});
|
||||
|
||||
final JLabel testTimeoutLabel = new JLabel(String.format(
|
||||
"Test timeout: %s", convertMillisToTimeStr(testTimeout)));
|
||||
final long startTime = System.currentTimeMillis();
|
||||
final Timer timer = new Timer(0, null);
|
||||
timer.setDelay(1000);
|
||||
timer.addActionListener((e) -> {
|
||||
int leftTime = testTimeout - (int) (System.currentTimeMillis() - startTime);
|
||||
if ((leftTime < 0) || testFinished) {
|
||||
timer.stop();
|
||||
dialog.dispose();
|
||||
}
|
||||
testTimeoutLabel.setText(String.format(
|
||||
"Test timeout: %s", convertMillisToTimeStr(leftTime)));
|
||||
});
|
||||
timer.start();
|
||||
|
||||
JTextArea textArea = new JTextArea(DESCRIPTION);
|
||||
textArea.setEditable(false);
|
||||
|
||||
final JButton testButton = new JButton("Print");
|
||||
final JButton passButton = new JButton("PASS");
|
||||
final JButton failButton = new JButton("FAIL");
|
||||
|
||||
testButton.addActionListener((e) -> {
|
||||
testButton.setEnabled(false);
|
||||
new Thread(() -> {
|
||||
try {
|
||||
doTest();
|
||||
|
||||
SwingUtilities.invokeLater(() -> {
|
||||
passButton.setEnabled(true);
|
||||
failButton.setEnabled(true);
|
||||
});
|
||||
} catch (Throwable t) {
|
||||
t.printStackTrace();
|
||||
dialog.dispose();
|
||||
fail("Exception occurred in a thread executing the test.");
|
||||
}
|
||||
}).start();
|
||||
});
|
||||
passButton.setEnabled(false);
|
||||
passButton.addActionListener((e) -> {
|
||||
dialog.dispose();
|
||||
pass();
|
||||
});
|
||||
failButton.setEnabled(false);
|
||||
failButton.addActionListener((e) -> {
|
||||
dialog.dispose();
|
||||
fail("TitledBorder label is cut off");
|
||||
});
|
||||
|
||||
try {
|
||||
File dir = new File(System.getProperty("test.src", "."));
|
||||
File customFontFile = new File(dir, FONT_FILE);
|
||||
customFont = Font.createFont(Font.TRUETYPE_FONT, customFontFile);
|
||||
customFont = customFont.deriveFont(FONT_SIZE);
|
||||
} catch(Exception e) {
|
||||
fail("Unable to load the custom font: " + e);
|
||||
}
|
||||
|
||||
JPanel mainPanel = new JPanel(new BorderLayout());
|
||||
JPanel textPanel = new JPanel() {
|
||||
@Override
|
||||
public void paint(Graphics g) {
|
||||
super.paint(g);
|
||||
drawText((Graphics2D) g, customFont, TEXT);
|
||||
}
|
||||
};
|
||||
|
||||
textPanel.setPreferredSize(new Dimension(300, 300));
|
||||
|
||||
JPanel labelPanel = new JPanel(new FlowLayout());
|
||||
labelPanel.add(testTimeoutLabel);
|
||||
mainPanel.add(labelPanel, BorderLayout.NORTH);
|
||||
mainPanel.add(textPanel, BorderLayout.EAST);
|
||||
mainPanel.add(textArea, BorderLayout.CENTER);
|
||||
JPanel buttonPanel = new JPanel(new FlowLayout());
|
||||
buttonPanel.add(testButton);
|
||||
buttonPanel.add(passButton);
|
||||
buttonPanel.add(failButton);
|
||||
mainPanel.add(buttonPanel, BorderLayout.SOUTH);
|
||||
dialog.add(mainPanel);
|
||||
|
||||
dialog.pack();
|
||||
dialog.setVisible(true);
|
||||
}
|
||||
|
||||
private static void doTest() throws Exception {
|
||||
SwingUtilities.invokeAndWait(() -> {
|
||||
try {
|
||||
new TestPrintable();
|
||||
} catch (PrinterException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private static void drawText(Graphics2D g, Font font, String text) {
|
||||
|
||||
Font prevFont = g.getFont();
|
||||
g.setFont(font);
|
||||
FontRenderContext frc = new FontRenderContext(new AffineTransform(), false, true);
|
||||
|
||||
Rectangle clip = g.getClipBounds();
|
||||
int cx = (int) clip.getCenterX();
|
||||
int cy = (int) (0.5 * clip.getCenterY());
|
||||
|
||||
FontMetrics metrics = g.getFontMetrics();
|
||||
int w = metrics.stringWidth(text);
|
||||
int h = metrics.getHeight();
|
||||
|
||||
int x = cx - w / 2;
|
||||
int y = cy - h / 4;
|
||||
|
||||
g.drawString(text, x, y);
|
||||
|
||||
GlyphVector gv = font.createGlyphVector(frc, text);
|
||||
g.drawGlyphVector(gv, x, y + h);
|
||||
|
||||
gv = font.deriveFont(1.0f).createGlyphVector(frc, text);
|
||||
int fontSize = font.getSize();
|
||||
|
||||
AffineTransform scale = AffineTransform.getScaleInstance(fontSize, fontSize);
|
||||
for (int i = 0; i < gv.getNumGlyphs(); i++) {
|
||||
gv.setGlyphTransform(i, scale);
|
||||
}
|
||||
|
||||
g.drawGlyphVector(gv, x, y + 2 * h);
|
||||
|
||||
g.setFont(prevFont);
|
||||
}
|
||||
|
||||
private static class TestPrintable implements Printable {
|
||||
|
||||
TestPrintable() throws PrinterException {
|
||||
PrinterJob job = PrinterJob.getPrinterJob();
|
||||
job.setPrintService(PrintServiceLookup.lookupDefaultPrintService());
|
||||
job.setPrintable(this);
|
||||
|
||||
if (job.printDialog()) {
|
||||
job.print();
|
||||
} else {
|
||||
throw new RuntimeException("Printing was canceled!");
|
||||
}
|
||||
}
|
||||
|
||||
void paint(Graphics2D g) {
|
||||
drawText(g, customFont, TEXT);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int print(Graphics graphics, PageFormat pageFormat, int index) {
|
||||
if (index == 0) {
|
||||
paint((Graphics2D) graphics);
|
||||
return PAGE_EXISTS;
|
||||
} else {
|
||||
return NO_SUCH_PAGE;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Binary file not shown.
Loading…
Reference in New Issue
Block a user