Может ли Java визуализировать полупрозрачный текст с использованием субпикселя AA?

Я обнаружил, что при рендеринге непрозрачного текста в Java (последняя версия 6u23) используется субпиксель AA просто отлично, при этом полупрозрачный текст не делает.

Субпиксельный АА:

alt textalt text

Тот же текст, для которого только цвет был изменен с 0xFFFFFFFF на 0xBFFFFFFF:

alt textalt text

Как вы можете видеть, полупрозрачный текст явно является стандартным АА, а не чистым полупрозрачным рендерингом, он имеет этот ужасный внешний вид "90-х".

Это связано с техническим ограничением для субпикселя AA в целом или ошибкой на Java или просто потому, что Java даже не пытается получить полупрозрачный текст, или я что-то пропустил?


Инициализация графики

dbGraphics=(Graphics2D)dbImage.getGraphics();
if(dctRoot.properties.getBoolean("Antialias",true)) {
    try {
        Map hnts=(Map)(dctRoot.awtComponent.getToolkit().getDesktopProperty("awt.font.desktophints"));

        // SET AA ON OVERALL (NOTE: GENERAL AA MUST BE OFF FOR SUBPIXEL AA TO BE HONORED - TEXT WIDGETS MUST DO THIS THEMSELVES)
        dbGraphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING,RenderingHints.VALUE_ANTIALIAS_ON);

        if(hnts!=null) {
            // SET FONT RENDERING HINTS FROM DESKTOP
            dbGraphics.addRenderingHints(hnts);
            }
        else {
            try {
                // SET TEXT AA TO FONT-SPECIFIED GASP AA (JAVA 6+)
                dbGraphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,RenderingHints.class.getField("VALUE_TEXT_ANTIALIAS_GASP").get(null));
                }
            catch(Throwable thr3) {
                // SET TEXT AA TO DEFAULT
                dbGraphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
                }
            }
        }
    catch(Throwable thr) {
        dctRoot.log.println("Antialiasing not supported on this JVM ("+thr+").");
        dctRoot.setProperty("Antialias","False");           // turn off AA for subsequent painting
        }
    }
else {
    try {
        dbGraphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING,RenderingHints.VALUE_ANTIALIAS_OFF);
        dbGraphics.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,RenderingHints.VALUE_TEXT_ANTIALIAS_OFF);
        }
    catch(Throwable thr) {;}                                // ignore exception
    }

Текстовое рендеринг

Object oaa=disableGeneralAA(gc);
...
gc.drawString(tl,xx,(ty+(xa*met.getHeight())));
restoreGeneralAA(gc,oaa);

...


static private volatile boolean         hasRenderingHints=true;

// *****************************************************************************
// STATIC INIT & MAIN
// *****************************************************************************

// *****************************************************************************
// STATIC METHODS
// *****************************************************************************

/**
 * Disable the general anti-aliasing rendering hint, returning whether the old value was RenderingHints.VALUE_ANTIALIAS_ON.
 * <p>
 * This method is needed for text rendering due to a bug in AWT; as of Java 6_20 when general AA is on text is not rendered using subpixel
 * AA, so general AA has to be turned off before rendering text and turned back on when done.  This method abstracts that work and deals
 * with the possibility that the JVM does not support rendering hints, such as is the case with JME JVMs.
 */
static public Object disableGeneralAA(Graphics2D gc) {
    Object                              old=null;

    if(hasRenderingHints) {
        try {
            old=gc.getRenderingHint(RenderingHints.KEY_ANTIALIASING);
            gc.setRenderingHint(RenderingHints.KEY_ANTIALIASING,RenderingHints.VALUE_ANTIALIAS_OFF);
            }
        catch(NoClassDefFoundError thr) { hasRenderingHints=false; }
        catch(NoSuchFieldError     thr) { hasRenderingHints=false; }
        catch(NoSuchMethodError    thr) { hasRenderingHints=false; }
        }
    return old;
    }

/**
 * Disable the general anti-aliasing rendering hint, returning whether the old value was RenderingHints.VALUE_ANTIALIAS_ON.
 * <p>
 * This method is needed for text rendering due to a bug in AWT; as of Java 6_20 when general AA is on text is not rendered using subpixel
 * AA, so general AA has to be turned off before rendering text and turned back on when done.  This method abstracts that work and deals
 * with the possibility that the JVM does not support rendering hints, such as is the case with JME JVMs.
 */
static public void restoreGeneralAA(Graphics2D gc, Object val) {
    Object                              old=null;

    if(hasRenderingHints && val!=null) {
        try { gc.setRenderingHint(RenderingHints.KEY_ANTIALIASING,val); }
        catch(NoClassDefFoundError thr) { hasRenderingHints=false; }
        catch(NoSuchFieldError     thr) { hasRenderingHints=false; }
        catch(NoSuchMethodError    thr) { hasRenderingHints=false; }
        }
    }

Ответ 2

Я думаю, это потому, что вы используете GASP, который берет точки из стиля шрифта. вы пытались использовать VALUE_TEXT_ANTIALIAS_DEFAULT и VALUE_ALPHA_INTERPOLATION_DEFAULT? это стоит того.

http://download.oracle.com/javase/6/docs/api/java/awt/RenderingHints.html

Ответ 3

Какую версию Java вы используете? Вы не говорите. Но, по-видимому, это было исправлено как в версии Java 6 update 12 (J6u12), так и в JDK7 b43

Смотрите здесь: http://bugs.sun.com/view_bug.do?bug_id=6749060

Если ваш тест снова с Java = или выше J6u12 и все еще видит ошибку, есть RFE, где вы можете прокомментировать в базе данных ошибок Sun.

Способ исправления на платформе Java - это либо:

  • Проголосуйте в отчете об ошибке в Sun BugParade, чтобы увеличить его приоритет и дождитесь, когда к нему зайдут программисты Sun/Oracle или
  • Теперь, когда Java является открытым исходным кодом, исправьте его самостоятельно. (присоединитесь к списку рассылки на ho.io/jkp5: -)

Отчет Bugparade, который вы хотите проголосовать или комментировать здесь (в случае, если вы проверите с j6u12, он все еще присутствует) url: ho.io/jkp2

Если вы хотите реализовать известную рабочую среду, чтобы текст выглядел приятным даже в более старых JRE, здесь предлагается обход.

url: ho.io/jkpy

"Похоже, что решение, которое я использую для эмуляции прозрачности, смешивая требуемый цвет переднего плана с цветом фона компонента, по-прежнему остается тем, что нужно использовать собственный растеризатор".

Удачи!