Skip to content

Commit 27fe3d7

Browse files
Toshio Nakamuramrserb
Toshio Nakamura
authored andcommittedMar 16, 2022
8240756: [macos] SwingSet2:TableDemo:Printed Japanese characters were garbled
Reviewed-by: prr, serb
1 parent bacfaa3 commit 27fe3d7

File tree

2 files changed

+212
-4
lines changed

2 files changed

+212
-4
lines changed
 

‎src/java.desktop/macosx/classes/sun/lwawt/macosx/CTextPipe.java

+76-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2011, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2011, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -81,17 +81,89 @@ public void drawString(final SunGraphics2D sg2d, final String s, final double x,
8181
}
8282
}
8383

84-
public void drawGlyphVector(final SunGraphics2D sg2d, final GlyphVector gV, final float x, final float y) {
85-
final Font prevFont = sg2d.getFont();
86-
sg2d.setFont(gV.getFont());
84+
private boolean hasSlotData(GlyphVector gv) {
85+
final int length = gv.getNumGlyphs();
86+
for (int i = 0; i < length; i++) {
87+
if ((gv.getGlyphCode(i) & CompositeGlyphMapper.SLOTMASK) != 0) {
88+
return true;
89+
}
90+
}
91+
return false;
92+
}
93+
94+
private Font getSlotFont(Font font, int slot) {
95+
Font2D f2d = FontUtilities.getFont2D(font);
96+
if (f2d instanceof CFont) {
97+
CompositeFont cf = ((CFont)f2d).getCompositeFont2D();
98+
PhysicalFont pf = cf.getSlotFont(slot);
99+
Font f = new Font(pf.getFontName(null),
100+
font.getStyle(), font.getSize());
101+
return f;
102+
}
103+
return null;
104+
}
105+
106+
private GlyphVector getGlyphVectorWithRange(final Font font, final GlyphVector gV, int start, int count) {
107+
int[] glyphs = new int[count];
108+
for (int i = 0; i < count; i++) {
109+
glyphs[i] = gV.getGlyphCode(start+i) & CompositeGlyphMapper.GLYPHMASK;
110+
}
111+
// Positions should be null to recalculate by native methods,
112+
// if GV was segmented.
113+
StandardGlyphVector sgv = new StandardGlyphVector(font,
114+
gV.getFontRenderContext(),
115+
glyphs,
116+
null, // positions
117+
null, // indices
118+
gV.getLayoutFlags());
119+
return sgv;
120+
}
121+
122+
private int getLengthOfSameSlot(final GlyphVector gV, final int targetSlot, final int start, final int length) {
123+
int count = 1;
124+
for (; start + count < length; count++) {
125+
int slot = (gV.getGlyphCode(start + count) &
126+
CompositeGlyphMapper.SLOTMASK) >> 24;
127+
if (targetSlot != slot) {
128+
break;
129+
}
130+
}
131+
return count;
132+
}
87133

134+
private void drawGlyphVectorImpl(final SunGraphics2D sg2d, final GlyphVector gV, final float x, final float y) {
88135
final long nativeStrikePtr = getNativeStrikePtr(sg2d);
89136
if (OSXSurfaceData.IsSimpleColor(sg2d.paint) && nativeStrikePtr != 0) {
90137
final OSXSurfaceData surfaceData = (OSXSurfaceData)sg2d.getSurfaceData();
91138
surfaceData.drawGlyphs(this, sg2d, nativeStrikePtr, gV, x, y);
92139
} else {
93140
drawGlyphVectorAsShape(sg2d, gV, x, y);
94141
}
142+
}
143+
144+
public void drawGlyphVector(final SunGraphics2D sg2d, final GlyphVector gV, final float x, final float y) {
145+
final Font prevFont = sg2d.getFont();
146+
sg2d.setFont(gV.getFont());
147+
148+
if (hasSlotData(gV)) {
149+
final int length = gV.getNumGlyphs();
150+
float[] positions = gV.getGlyphPositions(0, length, null);
151+
int start = 0;
152+
while (start < length) {
153+
int slot = (gV.getGlyphCode(start) &
154+
CompositeGlyphMapper.SLOTMASK) >> 24;
155+
sg2d.setFont(getSlotFont(gV.getFont(), slot));
156+
int count = getLengthOfSameSlot(gV, slot, start, length);
157+
GlyphVector rangeGV = getGlyphVectorWithRange(sg2d.getFont(),
158+
gV, start, count);
159+
drawGlyphVectorImpl(sg2d, rangeGV,
160+
x + positions[start * 2],
161+
y + positions[start * 2 + 1]);
162+
start += count;
163+
}
164+
} else {
165+
drawGlyphVectorImpl(sg2d, gV, x, y);
166+
}
95167
sg2d.setFont(prevFont);
96168
}
97169

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,136 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/**
25+
* @test
26+
* @bug 8240756
27+
* @summary Non-English characters are printed with wrong glyphs on MacOS
28+
* @modules java.desktop/sun.java2d java.desktop/sun.java2d.loops java.desktop/sun.font
29+
* @requires os.family == "mac"
30+
* @run main MultiSlotFontTest
31+
*/
32+
33+
import java.awt.Color;
34+
import java.awt.Dimension;
35+
import java.awt.Font;
36+
import java.awt.Graphics;
37+
import java.awt.Image;
38+
import java.awt.RenderingHints;
39+
import java.awt.font.FontRenderContext;
40+
import java.awt.font.GlyphVector;
41+
import java.awt.image.BufferedImage;
42+
import sun.font.StandardGlyphVector;
43+
import sun.java2d.OSXOffScreenSurfaceData;
44+
import sun.java2d.SunGraphics2D;
45+
import sun.java2d.SurfaceData;
46+
import sun.java2d.loops.SurfaceType;
47+
48+
public class MultiSlotFontTest {
49+
50+
private static final int WIDTH = 100;
51+
private static final int HEIGHT = 60;
52+
53+
private static final String TEST_STR = "\u3042\u3044\u3046\u3048\u304Aabc";
54+
private static final int EXPECTED_HEIGHT = 10;
55+
private static final int EXPECTED_WIDTH = 77;
56+
private static final int LIMIT_DIFF_HEIGHT = 3;
57+
private static final int LIMIT_DIFF_WIDTH = 15;
58+
59+
public static void main(String[] args) throws Exception {
60+
MultiSlotFontTest test = new MultiSlotFontTest();
61+
}
62+
63+
public MultiSlotFontTest() {
64+
BufferedImage img = createImage();
65+
66+
SurfaceData sd = OSXOffScreenSurfaceData.createDataIC(img,
67+
SurfaceType.IntRgb);
68+
SunGraphics2D g2d = new SunGraphics2D(sd,
69+
Color.BLACK, Color.WHITE, null);
70+
Font font = g2d.getFont();
71+
72+
if (font.canDisplayUpTo(TEST_STR) != -1) {
73+
System.out.println("There is no capable font. Skipping the test.");
74+
System.out.println("Font: " + font);
75+
return;
76+
}
77+
78+
FontRenderContext frc = new FontRenderContext(null, false, false);
79+
StandardGlyphVector gv = new StandardGlyphVector(font, TEST_STR, frc);
80+
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
81+
RenderingHints.VALUE_ANTIALIAS_OFF);
82+
g2d.drawGlyphVector(gv, 0.0f, (float)(HEIGHT - 5));
83+
g2d.dispose();
84+
85+
Dimension d = getBounds(img);
86+
87+
if (Math.abs(d.height - EXPECTED_HEIGHT) > LIMIT_DIFF_HEIGHT ||
88+
Math.abs(d.width - EXPECTED_WIDTH) > LIMIT_DIFF_WIDTH) {
89+
debugOut(img);
90+
throw new RuntimeException(
91+
"Incorrect GlyphVector shape " + d + "," + gv);
92+
}
93+
}
94+
95+
private static BufferedImage createImage() {
96+
BufferedImage image = new BufferedImage(WIDTH, HEIGHT,
97+
BufferedImage.TYPE_INT_RGB);
98+
Graphics g = image.createGraphics();
99+
g.setColor(Color.WHITE);
100+
g.fillRect(0, 0, WIDTH, HEIGHT);
101+
g.dispose();
102+
return image;
103+
}
104+
105+
private Dimension getBounds(BufferedImage img) {
106+
int top = HEIGHT;
107+
int left = WIDTH;
108+
int right = 0;
109+
int bottom = 0;
110+
for (int y = 0; y < HEIGHT; y++) {
111+
for (int x = 0; x < WIDTH; x++) {
112+
if ((img.getRGB(x, y) & 0xFFFFFF) == 0) {
113+
if (top > y) top = y;
114+
if (bottom < y) bottom = y;
115+
if (left > x) left = x;
116+
if (right < x) right = x;
117+
}
118+
}
119+
}
120+
return new Dimension(right - left, bottom - top);
121+
}
122+
123+
private void debugOut(BufferedImage img) {
124+
for (int y = 0; y < HEIGHT; y++) {
125+
for (int x = 0; x < WIDTH; x++) {
126+
int c = img.getRGB(x, y) & 0xFFFFFF;
127+
if (c == 0) {
128+
System.out.print("*");
129+
} else {
130+
System.out.print(" ");
131+
}
132+
}
133+
System.out.println();
134+
}
135+
}
136+
}

0 commit comments

Comments
 (0)
Please sign in to comment.