Проблемы с вращением BufferedImage

У меня есть некоторые проблемы с вращающимися изображениями в Java, используя класс AffineTransform.

У меня есть следующий способ создания повернутой (90 градусов) копии изображения:

private BufferedImage createRotatedCopy(BufferedImage img, Rotation rotation) {
    int w = img.getWidth();
    int h = img.getHeight();

    BufferedImage rot = new BufferedImage(h, w, BufferedImage.TYPE_INT_RGB);

    double theta;
    switch (rotation) {
        case CLOCKWISE:
            theta = Math.PI / 2;
            break;
        case COUNTERCLOCKWISE:
            theta = -Math.PI / 2;
            break;
        default:
            throw new AssertionError();
    }

    AffineTransform xform = AffineTransform.getRotateInstance(theta, w / 2, h / 2);
    Graphics2D g = (Graphics2D) rot.createGraphics();
    g.drawImage(img, xform, null);
    g.dispose();

    return rot;
}

Вращение - это простое перечисление со значениями NONE, CLOCKWISE и COUNTERCLOCKWISE.

Симптомы моих проблем отображаются здесь:

http://perp.se/so/rotate_problems.html

Итак, вращение работает нормально, но результирующие изображения не привязаны к правильным координатам (или как их следует поместить). И поскольку я не знаю, что я делаю в первую очередь (моя линейная алгебра слаба), я не знаю, как решить это самостоятельно.

Я попытался с некоторым случайным возиться с экземпляром AffineTransform, но мне это не помогло (конечно). Я пробовал поиск в Интернете (и поиск SO), но все примеры, которые я видел, в основном используют тот же подход, что и я..., который не работает для меня.

Спасибо за советы.

Ответ 1

Если вы должны выразить преобразование как однократное вращение, точка привязки зависит от направления вращения: либо (w/2, w/2), либо (h/2, h/2).

Но, вероятно, проще выразить как translate; rotate; translate, например.

AffineTransform xform = new AffineTransform();
xform.translate(0.5*h, 0.5*w);
xform.rotate(theta);
xform.translate(-0.5*w, -0.5*h);

Также рассмотрите использование getQuadrantRotateInstance вместо getRotateInstance.

Ответ 2

Поскольку вам нужно всего лишь 90-градусное вращение, вы можете избежать использования материала AffineTransform:

public BufferedImage rotate90DX(BufferedImage bi) {
    int width = bi.getWidth();
    int height = bi.getHeight();
    BufferedImage biFlip = new BufferedImage(height, width, bi.getType());
    for(int i=0; i<width; i++)
        for(int j=0; j<height; j++)
            biFlip.setRGB(height-1-j, width-1-i, bi.getRGB(i, j));
    return biFlip;
}

Это также позволяет избежать отсечения краев прямоугольных изображений.

От: http://snippets.dzone.com/posts/show/2936

Ответ 3

Вы можете попробовать альтернативное приложение и создать значок с изображения, а затем использовать Повернутый значок.

Или вы можете попробовать этот старый код, который я нашел на форумах Sun:

import java.awt.*;
import java.awt.geom.*;
import java.awt.image.*;
import java.io.*;
import java.net.*;
import javax.imageio.*;
import javax.swing.*;

public class RotateImage {
    public static void main(String[] args) throws IOException {
        URL url = new URL("https://blogs.oracle.com/jag/resource/JagHeadshot-small.jpg");
        BufferedImage original = ImageIO.read(url);
        GraphicsConfiguration gc = getDefaultConfiguration();
        BufferedImage rotated1 = tilt(original, -Math.PI/2, gc);
        BufferedImage rotated2 = tilt(original, +Math.PI/4, gc);
        BufferedImage rotated3 = tilt(original, Math.PI, gc);
        display(original, rotated1, rotated2, rotated3);
    }

    public static BufferedImage tilt(BufferedImage image, double angle, GraphicsConfiguration gc) {
        double sin = Math.abs(Math.sin(angle)), cos = Math.abs(Math.cos(angle));
        int w = image.getWidth(), h = image.getHeight();
        int neww = (int)Math.floor(w*cos+h*sin), newh = (int)Math.floor(h*cos+w*sin);
        int transparency = image.getColorModel().getTransparency();
        BufferedImage result = gc.createCompatibleImage(neww, newh, transparency);
        Graphics2D g = result.createGraphics();
        g.translate((neww-w)/2, (newh-h)/2);
        g.rotate(angle, w/2, h/2);
        g.drawRenderedImage(image, null);
        return result;
    }

    public static GraphicsConfiguration getDefaultConfiguration() {
        GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
        GraphicsDevice gd = ge.getDefaultScreenDevice();
        return gd.getDefaultConfiguration();
    }

    public static void display(BufferedImage im1, BufferedImage im2, BufferedImage im3, BufferedImage im4) {
        JPanel cp = new JPanel(new GridLayout(2,2));
        addImage(cp, im1, "original");
        addImage(cp, im2, "rotate -PI/2");
        addImage(cp, im3, "rotate +PI/4");
        addImage(cp, im4, "rotate PI");

        JFrame f = new JFrame("RotateImage");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setContentPane(cp);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    static void addImage(Container cp, BufferedImage im, String title) {
        JLabel lbl = new JLabel(new ImageIcon(im));
        lbl.setBorder(BorderFactory.createTitledBorder(title));
        cp.add(lbl);
    }
}

Ответ 4

Я не знаю, может ли это быть вашей проблемой.

AffineTransform xform = AffineTransform.getRotateInstance(theta, w / 2, h / 2);

Почему бы не попробовать?

AffineTransform xform = AffineTransform.getRotateInstance(theta);

ИЛИ

g.transform(AffineTransform.getRotateInstance(theta));
g.drawImage(img, 0, 0, w/2, h/2, null, null);