Java AWT custom CompositeContext & anti-aliasing: RasterFormatException при рисовании за пределами клиентской области

Я пытаюсь реализовать SWT GC - как рисунок xor-mode для AWT Graphics2D. Использование встроенного XORComposite не является вариантом, так как он не реализует графический просмотр в режиме xor, как в SWT.

График режима xor-mode сочетает цвета источника и назначения с помощью бинарных эксклюзивных или. AWT XORComposite (используется через g2d.setXORMode(Color)) использует постоянный xor-цвет, который сочетается с исходными цветами через двоичный эксклюзив или, то есть цвета назначения делают не влияют на полученные цвета.

Итак, единственный вариант, который я имею в виду, это написать мои собственные Composite и CompositeContext реализаций, которые соответствующим образом сочетают источник и место назначения.

После некоторого чтения я придумал эту простую реализацию: (Да, я знаю, что накладные расходы getPixel (...), setPixel (...). Я хочу, чтобы он работал правильно, прежде чем оптимизировать.)

private static class XorComposite implements Composite {

    public static XorComposite INSTANCE = new XorComposite();

    private XorContext context = new XorContext();

    @Override
    public CompositeContext createContext(ColorModel srcColorModel,
            ColorModel dstColorModel, RenderingHints hints) {
        return context;
    }

}

private static class XorContext implements CompositeContext {

    public XorContext() {
    }

    @Override
    public void compose(Raster src, Raster dstIn, WritableRaster dstOut) {
        int w = Math.min(src.getWidth(), dstIn.getWidth());
        int h = Math.min(src.getHeight(), dstIn.getHeight());

        int[] srcRgba = new int[4];
        int[] dstRgba = new int[4];

        for (int x = 0; x < w; x++) {
            for (int y = 0; y < h; y++) {
                src.getPixel(x, y, srcRgba);
                dstIn.getPixel(x, y, dstRgba);
                for (int i = 0; i < 3; i++) {
                    dstRgba[i] ^= srcRgba[i];
                }
                dstOut.setPixel(x, y, dstRgba);
            }
        }
    }

    @Override
    public void dispose() {
    }

}

Эта реализация отлично работает при отключении сглаживания. Если сглаживание включено, оно работает до тех пор, пока мои рисунки полностью видны, т.е. Внутри моей JPanel, на которую я рисую. Но если чертеж пересекает границы JPanel, генерируется исключение RasterFormatException:

Exception in thread "AWT-EventQueue-0" java.awt.image.RasterFormatException: (y + height) is outside raster
    at sun.awt.image.IntegerInterleavedRaster.createWritableChild(IntegerInterleavedRaster.java:470)
    at sun.awt.image.IntegerInterleavedRaster.createChild(IntegerInterleavedRaster.java:514)
    at sun.java2d.pipe.GeneralCompositePipe.renderPathTile(GeneralCompositePipe.java:106)
    at sun.java2d.pipe.AAShapePipe.renderTiles(AAShapePipe.java:201)
    at sun.java2d.pipe.AAShapePipe.fillParallelogram(AAShapePipe.java:102)
    at sun.java2d.pipe.PixelToParallelogramConverter.fillRectangle(PixelToParallelogramConverter.java:322)
    at sun.java2d.pipe.PixelToParallelogramConverter.fill(PixelToParallelogramConverter.java:159)
    at sun.java2d.pipe.ValidatePipe.fill(ValidatePipe.java:160)
    at sun.java2d.SunGraphics2D.fill(SunGraphics2D.java:2422)
    at org.eclipse.gef4.graphics.examples.AwtXorTestPanel.paint(AwtXorTest.java:117)
    ... (irrelevant)

Примечательно, что мой Composite/CompositeContext не генерирует исключение, но внутренние элементы AWT генерируют исключение при попытке создать объекты Raster, которые он хочет передать в свой CompositeContext.

К сожалению, PixelToParallelogramConverter используется только для пользовательских композитов, когда включено сглаживание. Например, встроенный XORComposite использует собственный метод для рисования. Я предполагаю ошибку AWT, но я не уверен.

Помощь будет принята с благодарностью:)

Update:

Как предложил Дюрандал, я проверил код с java-6-sun и java-1.6.0-openjdk. Я узнал, что OpenJDK выбрасывает исключение, а Sun-JDK этого не делает. Следовательно, я сообщил bug в отслеживании ошибок OpenJDK.

Я обновлю этот вопрос, когда проблема будет решена. Посетите соответствующую ошибку OpenJDK для получения информации о текущем ходе.

С уважением, Matthias

Вот пример программы, поэтому вы можете проверить ее локально:

/*******************************************************************************
 * Copyright (c) 2013 itemis AG and others.
 * 
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 * 
 * Contributors:
 *     Matthias Wienand (itemis AG) - initial API and implementation
 * 
 *******************************************************************************/
package org.eclipse.gef4.graphics.examples;

import java.awt.Color;

public class AwtXorTest extends JApplet {

    private static final long serialVersionUID = 1L;

    public static void main(String[] args) {
        JFrame frame = new JFrame();
        frame.setTitle("AWT XorMode Test");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        JApplet applet = new AwtXorTest();
        applet.init();
        frame.getContentPane().add(applet);
        frame.pack();
        frame.setVisible(true);
    }

    @Override
    public void init() {
        JPanel panel = new AwtXorTestPanel();
        getContentPane().add(panel);
    }

}

class AwtXorTestPanel extends JPanel {

    private static class XorComposite implements Composite {

        public static XorComposite INSTANCE = new XorComposite();

        private XorContext context = new XorContext();

        @Override
        public CompositeContext createContext(ColorModel srcColorModel,
                ColorModel dstColorModel, RenderingHints hints) {
            return context;
        }

    }

    private static class XorContext implements CompositeContext {

        public XorContext() {
        }

        @Override
        public void compose(Raster src, Raster dstIn, WritableRaster dstOut) {
            int w = Math.min(src.getWidth(), dstIn.getWidth());
            int h = Math.min(src.getHeight(), dstIn.getHeight());

            int[] srcRgba = new int[4];
            int[] dstRgba = new int[4];

            for (int x = 0; x < w; x++) {
                for (int y = 0; y < h; y++) {
                    src.getPixel(x, y, srcRgba);
                    dstIn.getPixel(x, y, dstRgba);
                    for (int i = 0; i < 3; i++) {
                        dstRgba[i] ^= srcRgba[i];
                    }
                    dstOut.setPixel(x, y, dstRgba);
                }
            }
        }

        @Override
        public void dispose() {
        }

    }

    private static final long serialVersionUID = 1L;

    public AwtXorTestPanel() {
        setPreferredSize(new Dimension(640, 480));
    }

    @Override
    public void paint(Graphics graphics) {
        Graphics2D g2d = (Graphics2D) graphics;

        // comment out to see it working:
        g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
                RenderingHints.VALUE_ANTIALIAS_ON);

        g2d.setComposite(XorComposite.INSTANCE);
        g2d.setColor(new Color(0, 255, 255)); // resulting color should be red
        g2d.fill(new Rectangle(100, 100, 500, 500));
    }

}

Ответ 1

Предостережение: Прошло некоторое время с тех пор, как я коснулся Растеров.

Похоже, вы, вероятно, обращаетесь к пикселям вне растра.

Растры имеют minX, minY, поэтому ваши петли должны быть примерно такими:

int srcMinX = src.getMinX();
int srcMinY = src.getMinY();
int dstInMinX = dstIn.getMinX();
int dstInMinY = dstIn.getMinY();
int dstOutMinX = dstOut.getMinX();
int dstOutMinY = dstOut.getMinY();

for (int x = 0; x < w; x++) {
    for (int y = 0; y < h; y++) {
        src.getPixel(x+srcMinX, y+srcMinY, srcRgba);
        dstIn.getPixel(x+dstInMinX, y+dstInMinY, dstRgba);
        for (int i = 0; i < 3; i++) {
            dstRgba[i] ^= srcRgba[i];
        }
        dstOut.setPixel(x+dstOutMinX, y+dstOutMinY, dstRgba);
    }
}

Ответ 2

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