Regression test for non-Latin character rendering

Dr Andrew John Hughes ahughes at redhat.com
Mon Nov 1 06:17:14 PDT 2010


On 18:26 Fri 29 Oct     , Pavel Tisnovsky wrote:
> Hi all,
> 
> I created regression test (reproducer) for RH bug
> https://bugzilla.redhat.com/show_bug.cgi?id=643674
> 
> This regression test checks if all required fonts are properly installed
> and that *.src.properties font configuration file is correct for given
> system (Debian, Fedora, Ubuntu, SuSE...). The test is based on rendering
> certain characters from selected code pages into BufferedImage.
> 
> When the required font does not exists or font configuration is broken
> in any way, only empty rectangle is rendered instead of the selected
> character shape. This rectangle is filtered and then the destination
> image is tested whether it is empty or almost empty (ie if it contains
> only white area). White area means that selected character is not
> correctly rendered and vice versa.
> 
> If test images with rendered characters needs to be created, please use
> the following flag:
> 
> -create-images
> 
> (name of each image start with character code)
> 
> Is it ok to push this test (as patch) to IcedTea6 HEAD?
> 

Sounds ok.  Can you post the actual patch for IcedTea6?

Probably worth backporting to the branches too.

> (I'd like to push it to OpenJDK6 & OpenJDK7, but it takes some time to
> get approved...)
> 

Yeah, do that when you have time.

> Cheers,
> Pavel

> import java.awt.Color;
> import java.awt.Font;
> import java.awt.Graphics2D;
> import java.awt.RenderingHints;
> import java.awt.image.BufferedImage;
> import java.io.File;
> import java.io.IOException;
> import java.util.Arrays;
> 
> import javax.imageio.ImageIO;
> 
> /**
>  * @test
>  * @run main InternationalFontsRendering
>  * @author Pavel Tisnovsky
>  *
>  * @summary This test check if all required fonts are properly installed and configured.
>  *
>  * This regression test checks if all required fonts are properly installed and
>  * that .src.properties font configuration is correct. The test is based on
>  * rendering certain characters from selected code pages into BufferedImage.
>  *
>  * When the font does not exists or font configuration is broken, only empty
>  * rectangle is rendered instead of the selected character shape. This rectangle
>  * is filtered and then the destination image is tested whether it is empty
>  * (=white).
>  *
>  * If test images with rendered characters needs to be created use following flag:
>  * -create-images
>  *
>  * regression for bug: https://bugzilla.redhat.com/show_bug.cgi?id=643674
>  * (Bug 643674 Misconfigured Path for Asian Font)
>  *
>  * @author Pavel Tisnovsky
>  *
>  */
> public class InternationalFontsRendering
> {
>     private static int WIDTH = 200;
>     private static int HEIGHT = 200;
> 
>     private static final int FONT_SIZE = 160;
> 
>     private static final int MINIMUM_HORIZONTAL_LINE_LENGTH = 70;
>     private static final int MINIMUM_VERTICAL_LINE_LENGTH = 80;
>     private static final int BLACK_WHITE_THRESHOLD = 128;
>     private static final int BLACK_PIXEL_COUNT_THRESHOLD = WIDTH * HEIGHT / 1000;
> 
>     private static final String[][] testedStrings = {
>         {"Latin-1",    "abcdefABCDEF"},
>         {"Latin-2",    "??????????????????????????"},
>         {"Cyrilic",    "??????????????????"},
>         {"Greek",      "????????????????????????????????"},
>         {"Asia-Test1", "???????????????"},
>         {"Asia-Test2", "??????????????????"},
>         {"Asia-Test3", "???????????????"},
>         {"Asia-Test4", "???????????????"},
>     };
> 
>     /**
>      * Creates test image a renders one big character to it.
>      * @param testedString
>      * @return
>      */
>     private BufferedImage createTestImage(String str)
>     {
>         BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_BYTE_GRAY);
>         Graphics2D gc = image.createGraphics();
>         gc.setBackground(Color.WHITE);
>         gc.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_OFF);
>         gc.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_OFF);
>         gc.clearRect(0, 0, WIDTH, HEIGHT);
>         gc.setFont(new Font(Font.DIALOG, Font.PLAIN, FONT_SIZE));
>         gc.setColor(Color.BLACK);
>         gc.drawString(str, 0, HEIGHT - 40);
>         gc.dispose();
>         return image;
>     }
> 
>     /**
>      * Creates destination image and then copies data from source image to it
>      * @param src
>      * @return
>      */
>     private BufferedImage createDestinationImage(BufferedImage src)
>     {
>         BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_BYTE_GRAY);
>         src.copyData(image.getRaster());
>         return image;
>     }
> 
>     /**
>      * Removes long horizontal lines from image
>      */
>     private void removeHorizontalLines(BufferedImage image)
>     {
>         for (int y = 0; y < image.getHeight(); y++)
>         {
>             int startX = -1, endX = -1;
>             for (int x = 0; x < image.getWidth(); x++)
>             {
>                 int color = image.getRaster().getSample(x, y, 0);
>                 if (startX < 0 && color < 128)
>                 {
>                     startX = x;
>                     //System.out.println("> " + y + "\t" + x);
>                 }
>                 if (startX > 0 && endX < 0 && color > BLACK_WHITE_THRESHOLD)
>                 {
>                     endX = x;
>                     //System.out.println("< " + y + "\t" + x);
>                 }
>             }
>             if (startX > 0 && endX > 0 && (endX - startX) > MINIMUM_HORIZONTAL_LINE_LENGTH)
>             {
>                 for (int x = startX; x < endX; x++)
>                 {
>                     image.getRaster().setSample(x, y, 0, 255);
>                 }
>             }
>         }
>     }
> 
>     /**
>      * Removes long vertical lines from image
>      */
>     private void removeVerticalLines(BufferedImage image)
>     {
>         for (int x = 0; x < image.getWidth(); x++)
>         {
>             int startY = -1, endY = -1;
>             for (int y = 0; y < image.getHeight(); y++)
>             {
>                 int color = image.getRaster().getSample(x, y, 0);
>                 if (startY < 0 && color < 128)
>                 {
>                     startY = y;
>                 }
>                 if (startY > 0 && endY < 0 && color > BLACK_WHITE_THRESHOLD)
>                 {
>                     endY = y;
>                     //System.out.println("< " + y + "\t" + x);
>                 }
>             }
>             if (startY > 0 && endY > 0 && (endY - startY) > MINIMUM_VERTICAL_LINE_LENGTH)
>             {
>                 for (int y = startY; y < endY; y++)
>                 {
>                     image.getRaster().setSample(x, y, 0, 255);
>                 }
>             }
>         }
>     }
> 
>     /**
>      * Test if image is almost empty (one large white area)
>      * @param image
>      * @return
>      */
>     private boolean isImageAlmostEmpty(BufferedImage image)
>     {
>         int blackPixelCount = 0;
>         for (int y = 0; y < image.getHeight(); y++)
>         {
>             for (int x = 0; x < image.getWidth(); x++)
>             {
>                 if (image.getRaster().getSample(x, y, 0) < BLACK_WHITE_THRESHOLD)
>                 {
>                     blackPixelCount++;
>                 }
>             }
>         }
>         return blackPixelCount < BLACK_PIXEL_COUNT_THRESHOLD;
>     }
> 
>     public void runTest(boolean createImages) throws IOException
>     {
>         for (String[] testedString : testedStrings)
>         {
>             System.out.println("\nTesting " + testedString[0]);
>             for (int i = 0; i < testedString[1].length(); i++)
>             {
>                 String str = testedString[1].substring(i, 1+i);
>                 int code = str.charAt(0);
>                 System.out.print(code + "\t");
>                 BufferedImage src = createTestImage(str);
>                 BufferedImage dst = createDestinationImage(src);
>                 removeHorizontalLines(dst);
>                 removeVerticalLines(dst);
> 
>                 if (createImages /* || true*/)
>                 {
>                     ImageIO.write(src, "png", new File(code + "_src.png"));
>                     ImageIO.write(dst, "png", new File(code + "_dst.png"));
>                 }
> 
>                 if (isImageAlmostEmpty(dst))
>                 {
>                     throw new RuntimeException("Error in rendering of character with code: " + code);
>                 }
>             }
>         }
>     }
> 
>     public static void main(String[] args) throws IOException
>     {
>         new InternationalFontsRendering().runTest(Arrays.asList(args).contains("-create-images"));
>     }
> }


-- 
Andrew :)

Free Java Software Engineer
Red Hat, Inc. (http://www.redhat.com)

Support Free Java!
Contribute to GNU Classpath and the OpenJDK
http://www.gnu.org/software/classpath
http://openjdk.java.net
PGP Key: 94EFD9D8 (http://subkeys.pgp.net)
Fingerprint = F8EF F1EA 401E 2E60 15FA  7927 142C 2591 94EF D9D8



More information about the distro-pkg-dev mailing list