Проблемы с вращением 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);