Как согнуть изображение в Java

Есть ли способ согнуть BufferedImage на яве?

Я думал, что если я обрежу изображение на более мелкие части и поверну их, то я по сути изогну изображение, но, похоже, это не сработает.

Вот метод, который я создал:

/**
 * This is a recursive method that will accept an image the point where the bending will start and the point where the bending will end, as well as the angle of bending
 * 
 * @param original:the original image
 * @param startingPoint: the point where the bending should start
 * @param endingPoint: the point where the bending should end
 * @param radiands: the angle
 * @return the bent image
 */
public static BufferedImage getBentImage(BufferedImage original, int startingPoint, int endingPoint, double radians) {
    if (startingPoint >= endingPoint)
        return original;

    int type = BufferedImage.TYPE_INT_ARGB;
    int width = original.getWidth();
    int height = original.getHeight();

    BufferedImage crop = original.getSubimage(0, 0, startingPoint, height);
    BufferedImage crop0 = original.getSubimage(startingPoint, 0, width - startingPoint, height);
    BufferedImage bendCrop = new BufferedImage(width, height, type);
    BufferedImage image = new BufferedImage(width, height, type);

    AffineTransform rotation = new AffineTransform();
    rotation.translate(0, 0);
    rotation.rotate(radians);

    Graphics2D g = bendCrop.createGraphics();
    g.drawImage(crop0, rotation, null);
    g.dispose();

    g = image.createGraphics();
    g.drawImage(crop, 0, 0, null);
    g.drawImage(bendCrop, startingPoint, 0, null);
    g.dispose();

    return getBentImage(image, startingPoint + 1, endingPoint, radians);
}

Это оригинальное изображение:

исходное изображение

И это результат этого getBentImage(image, 200, 220, Math.toRadians(1)):

результат

Я ожидал чего-то ближе к:

ожидаемый результат

Любые идеи о том, как на самом деле реализовать getBentImage() метод?

2 ответа

Решение

Как предлагается в комментариях, простой подход состоит в том, чтобы разделить изображение на 3 части:

  1. Идентичен оригиналу.
  2. Сгибается в соответствии с преобразованием изгиба.
  3. Постоянное диагональное продолжение.

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

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

public class Main extends JFrame {

    static BufferedImage image;

    public static void main(String[] args) {

        try {
            image = ImageIO.read(ClassLoader.getSystemResource("img.png"));
        } catch (IOException e) {
            e.printStackTrace();
        }
        new Main();
    }

    public Main() {

        getContentPane().setLayout(new BorderLayout(5, 10));
        BufferedImage img2 = transform(15, 100, 300);

        JLabel label1 = new JLabel(new ImageIcon(image));
        label1.setHorizontalAlignment(JLabel.LEFT);
        label1.setOpaque(true);
        label1.setBackground(Color.YELLOW);
        add(label1, BorderLayout.NORTH);

        JLabel label2 = new JLabel(new ImageIcon(img2));
        label2.setHorizontalAlignment(JLabel.LEFT);
        label2.setOpaque(true);
        label2.setBackground(Color.CYAN);
        add(label2);

        pack();
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);
    }

    static BufferedImage transform(int t, int x1, int x2) {

        final double TH = Math.toRadians(t);
        final int D = x2 - x1;
        final int W = image.getWidth();
        final int H = image.getHeight();

        final int dD = (int) (D / (2 * TH) * Math.sin(2 * TH));
        final int dH = (int) (D / TH * Math.pow(Math.sin(TH), 2));
        final int pH = (int) ((W - x2) * Math.tan(2 * TH));

        final int width = W - (D - dD);
        final int height = (int) (H + dH + pH);

        System.out.println(W + " " + H + " -> " + width + " " + height);

        BufferedImage img2 = new BufferedImage(width, height, image.getType());

        for (int x = 0; x < x1; x++) {
            for (int y = 0; y < H; y++) {
                int rgb = image.getRGB(x, y);
                img2.setRGB(x, y, rgb);
            }
        }

        for (int x = x1; x < x2; x++) {
            for (int y = 0; y < H; y++) {
                int rgb = image.getRGB(x, y);
                int dx = (int) (D / (2 * TH) * Math.sin(2 * (x-x1) * TH / D));
                int dy = (int) (D / TH * Math.pow(Math.sin((x-x1) * TH / D), 2));
                img2.setRGB(x1 + dx, y + dy, rgb);
            }
        }

        for (int x = x2; x < W; x++) {
            for (int y = 0; y < H; y++) {
                int rgb = image.getRGB(x, y);
                int dp = (int) ((x - x2) * Math.tan(2 * TH));
                img2.setRGB(x - (D - dD), y + dH + dp, rgb);
            }
        }

        return img2;
    }
}

Что касается расчетов, я оставлю это как домашнее задание; это просто геометрия / тригонометрия, которая относится к Math.SE больше, чем к SO. Если вы не можете понять это, я дам вам направление.

Обратите внимание, что этот метод не может быть быстрым и может быть оптимизирован, я оставлю это и вам. Ох, и округление doubleс ints небрежно, поэтому результат не идеален для пикселей.

Я не знаю, что вы подразумеваете под изгибом, но по сути у вас есть прямоугольник, и вы ломаете один его кусок и вращаете его:

поэтому алгоритм выглядит следующим образом:

rotate line(x, 0, width-1, 0)
rotate line(x, height-1, width-1, height-1)
connect the pieces

По сути, вы ищете поворотную линию.

Другие вопросы по тегам