AffineTransform: масштабирование формы от ее центра

Я пытаюсь масштабировать прямоугольник от его центра, используя AffineTransform. Я уверен, что решение очевидно, но я не могу заставить его работать! Вот что я тестировал до сих пор...

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Rectangle;
import java.awt.geom.AffineTransform;

import javax.swing.JOptionPane;
import javax.swing.JPanel;

public class Test extends JPanel {
    Test()
        {
        super(null);
        setOpaque(true);
        setBackground(Color.WHITE);
        setPreferredSize(new Dimension(200,200));
        }
    @Override
    protected void paintComponent(Graphics g1) {
        super.paintComponent(g1);
        Rectangle r= new Rectangle(5,5,getWidth()-10,getHeight()-10);
        double cx= r.getCenterX();
        double cy= r.getCenterY();
        Graphics2D g=(Graphics2D)g1;
        g.setColor(Color.BLACK);
        AffineTransform old= g.getTransform();
        for(double zoom=0.9; zoom>=0.5; zoom-=0.1)
            {
            AffineTransform tr2= new AffineTransform(old);
            tr2.translate(-cx, -cy);
            tr2.scale(zoom, zoom);
            tr2.translate(cx/zoom,cy/zoom);
            g.setTransform(tr2);
            g.draw(r);
            g.setTransform(old);
            }
        }


    public static void main(String[] args) {
        JOptionPane.showMessageDialog(null, new Test());
        }
    }

Но это не работает... Любое предложение?


person Pierre    schedule 27.03.2009    source источник
comment
Я предполагаю, что вы пытаетесь отслеживать концентрические/внутренние прямоугольники?   -  person basszero    schedule 27.03.2009


Ответы (5)


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

Используйте это вместо этого:

tr2.translate(
    (this.getWidth()/2) - (r.getWidth()*(zoom))/2,
    (this.getHeight()/2) - (r.getHeight()*(zoom))/2
);
tr2.scale(zoom,zoom);
g.setTransform(tr2);

Это переводит прямоугольник в центр панели перед его масштабированием. В моих тестах работает нормально.

person Welbog    schedule 30.03.2009

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

tr2.translate(
    r.getWidth()*(1-zoom)/2,
    r.getHeight()*(1-zoom)/2
);
tr2.scale(zoom,zoom);
g.setTransform(tr2);

Таким образом, вы перемещаете прямоугольник влево и вверх на половину изменения ширины и высоты.

person Welbog    schedule 27.03.2009
comment
ваше решение не работает, но оно дает мне новые идеи, спасибо - person Pierre; 27.03.2009
comment
@mmyers: какой у тебя результат? У меня нет доступа к Java IDE на работе, поэтому я не могу протестировать ее самостоятельно. Я работаю в основном по памяти. - person Welbog; 27.03.2009
comment
@Welbog: в исходном коде все квадраты начинаются в верхнем левом углу. С вашим кодом они постепенно смещаются к середине, но этого недостаточно. - person Michael Myers; 27.03.2009
comment
А, понял. Все, что вам нужно сделать, это переместить шкалу () после перевода (), и она отлично работает. - person Michael Myers; 27.03.2009
comment
Упс, извините, я был слишком быстр. Этот центрированный квадрат был самым особым случаем. Когда это прямоугольник в любом месте экрана, ваше решение не работает. Я снова попытался сместить фигуру из ее центра, scake и повторно центрировать, но я все еще не могу найти правильное решение... - person Pierre; 31.03.2009

Я как раз работал над настольным приложением, чтобы обрезать лицо Бритни Спир (ДЕНЬ). Прямоугольник обрезки должен был масштабироваться вокруг своей центральной точки:

import javafx.scene.Cursor;
import javafx.scene.Group;
import javafx.scene.input.MouseEvent;
import javafx.scene.input.ScrollEvent;
import javafx.scene.paint.Color;
import javafx.scene.paint.Paint;
import javafx.scene.shape.Rectangle;

class ResizableRectangle extends Rectangle {

 ResizableRectangle(double x, double y, double width, double height, Group group) {

  super(x, y, width, height);

  // Set scroll listener for crop selection
  group.addEventHandler(ScrollEvent.SCROLL, event -> {
   double zoomFactor = 1.10;
   double deltaY = event.getDeltaY();
   if (deltaY > 0) {
    zoomFactor = 2.0 - zoomFactor;
   }

   super.setX(getX() + (super.getWidth() * (1 - zoomFactor) / 2)); // Set new X position
   super.setWidth(getWidth() * zoomFactor); // Set new Width

   super.setY(getY() + (super.getHeight() * (1 - zoomFactor) / 2)); // Set new Y position
   super.setHeight(getHeight() * zoomFactor); // Set new Height

   event.consume();
  });
 });
}

В общем случае алгоритм работает так:

  • Перевести значения x прямоугольника с помощью: x + (width * (1 - zoomFactor) / 2)
  • Перевести значения y с помощью: y + (height * (1 - zoomFactor) / 2)
  • Установите новую ширину: width * zoomFactor
  • Установить новую высоту: height * zoomFactor
person Martin Erlic    schedule 04.08.2017

Это включает в себя процесс, называемый сопряжением преобразования.

Если S — это масштабирование, которое вы хотите выполнить, а T — это преобразование, переводящее точку (0,0) в точку, которая должна быть центром вашего масштабирования, то преобразование, которое выполняет эту работу, — это

T (S (обратный (T)))

person paulmurray    schedule 27.03.2009

(Позже) Вот решение, работающее без каких-либо предварительных знаний о размере панели.

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.awt.geom.Ellipse2D;

import javax.swing.JOptionPane;
import javax.swing.JPanel;

public class Test extends JPanel
    {
    private static final long serialVersionUID = 1L;
    private Test()
        {
        super(null);
        setOpaque(true);
        setBackground(Color.WHITE);
        setPreferredSize(new Dimension(600,600));
        }

    @Override
    protected void paintComponent(Graphics g1) {
        super.paintComponent(g1);
        Shape r= new Ellipse2D.Double(5,380,400,200);
        double cx= r.getBounds2D().getCenterX();
        double cy= r.getBounds2D().getCenterY();
        Graphics2D g=(Graphics2D)g1;
        g.setColor(Color.BLACK);
        AffineTransform old= g.getTransform();
        g.drawLine((int)cx-10, (int)cy, (int)cx+10, (int)cy);
        g.drawLine((int)cx, (int)cy-10, (int)cx, (int)cy+10);
        for(double zoom=1; zoom>=0.1; zoom-=0.1)
                {


                AffineTransform tr2 =AffineTransform.getTranslateInstance(-cx, -cy);
                AffineTransform  tr= AffineTransform.getScaleInstance(zoom,zoom);
                tr.concatenate(tr2); tr2=tr;
                tr =AffineTransform.getTranslateInstance(cx, cy);
                tr.concatenate(tr2); tr2=tr;

                tr= new AffineTransform(old);
                tr.concatenate(tr2);  tr2=tr;

                g.setTransform(tr2);



                g.draw(r);
                g.setTransform(old);
                }
        }


    public static void main(String[] args) {
        JOptionPane.showMessageDialog(null, new Test());
        }
    }
person Pierre    schedule 23.06.2009