java - 如何在java中缩放图像上绘图?

标签 java image swing scale mouselistener

我有一个 BufferedImage,已添加到 JPanel 中。 我能够使用 AffineTransform 放大/缩小图像。 我可以在图像上书写,但不能在我想要的位置上书写。 我绘制的线条显示在图像上的错误位置。

我猜这与规模有关,但我无法弄清楚我的代码的哪一部分是错误的。

这就是我的意思的屏幕截图。

这是我的代码:


    public class MouseScaleTest {

        int xbegin = 0;
        int ybegin = 0;
        int xend = 0;
        int yend = 0;

        boolean isNewLine = true;

        int initx;
        int inity;

        int count = 0;
        Line2D linebuffer;
        Rectangle2D box;
        final ArrayList<Line2D> lineContainer = new ArrayList();
        final ArrayList<Rectangle2D> boxContainer = new ArrayList();

        public static void main(String[] args) {
            new MouseScaleTest();
        }

        public MouseScaleTest() {
            EventQueue.invokeLater(new Runnable() {
                @Override
                public void run() {

                    JFrame frame = new JFrame("Testing");
                    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                    frame.setLayout(new BorderLayout());

    //                JScrollPane
                    frame.add(new TestPane());
                    frame.pack();
                    frame.setLocationRelativeTo(null);
                    frame.setVisible(true);
                }
            });
        }

        public class TestPane extends JPanel {

            private BufferedImage img;

            private float scale = 1f;
            private float scaleDelta = 0.05f;
            JLabel lbl;
            JLabel scalePoint;
            JLabel begin;
            JLabel end;
            JLabel aft;

            public TestPane() {

                try {
                    img = ImageIO.read(new File("C:\\Users\\John Ebarita\\Documents\\report.jpg"));
                } catch (IOException ex) {
                    ex.printStackTrace();
                }
                lbl = new JLabel();
                lbl.setBounds(0, 0, 50, 10);
                scalePoint = new JLabel();
                scalePoint.setBounds(50, 0, 50, 10);
                begin = new JLabel();
                begin.setBounds(150, 0, 50, 10);
                end = new JLabel();
                end.setBounds(200, 0, 50, 10);
                aft = new JLabel();
                aft.setBounds(250, 0, 50, 10);

                add(lbl);
                add(scalePoint);
                add(begin);
                add(end);
                add(aft);

                addMouseWheelListener(new MouseWheelListener() {

                    @Override
                    public void mouseWheelMoved(MouseWheelEvent e) {
                        int rotation = e.getWheelRotation();
                        if (rotation < 0) {
                            scale -= scaleDelta;
                        } else {
                            scale += scaleDelta;
                        }
                        if (scale < 0) {
                            scale = 0;
                        }
    //                    else if (scale > 1.5) {
    //                        scale = 1;
    //                    }
                        scalePoint.setText("Scale: " + String.valueOf(scale));

                        repaint();

    //                    System.out.println(getSize());
                    }
                });

                addMouseMotionListener(new MouseMotionListener() {
                    @Override
                    public void mouseDragged(MouseEvent e) {
                        if (isNewLine == false) {
                            xend = e.getX();
                            yend = e.getY();

                            end.setText("End Values : " + xend + " " + yend);
                            repaint();
                        }
                    }

                    @Override
                    public void mouseMoved(MouseEvent e) {
                        lbl.setText("Mouse Point: " + e.getX() + " " + e.getY());
                        if (isNewLine == false) {
                            xend = e.getX();
                            yend = e.getY();
                            repaint();
                        }
                    }
                });

                addMouseListener(new MouseAdapter() {

                    @Override
                    public void mouseReleased(MouseEvent e) {
                        System.out.println(e.getX() + " " + e.getY() + "mouse");
                        if (isNewLine == true) {
                            xbegin = xend = e.getX();
                            ybegin = yend = e.getY();

                            isNewLine = false;
                            box = new Rectangle2D.Float(xend - 5, yend - 5, 12, 12);

                            boxContainer.add(box);
                        } else {

                            linebuffer = new Line2D.Float((float) xbegin, (float) ybegin, (float) xend, (float) yend);

                            System.out.println("xbegin: " + xbegin + " " + ybegin + " " + xend + " " + yend);
                            lineContainer.add(linebuffer);

                            repaint();

                            xbegin = e.getX();
                            ybegin = e.getY();

                            if (box.contains(xend, yend)) {
                                xend = (int) box.getCenterX();
                                yend = (int) box.getCenterY();
                                isNewLine = true;
                                return;
                            }

                            box = new Rectangle2D.Float(xend - 6, yend - 6, 12, 12);

                            boxContainer.add(box);
                            return;
                        }
                    }

                    @Override
                    public void mousePressed(MouseEvent e) {
                        begin.setText("Begin Values : " + xbegin + " " + ybegin);
                    }
                });
                setBorder(new BevelBorder(1));
            }

            @Override
            public Dimension getPreferredSize() {
                return img == null ? new Dimension(200, 200) : new Dimension(img.getWidth(), img.getHeight());
            }

            @Override
            protected void paintComponent(Graphics g) {
                super.paintComponent(g);
                if (img != null) {
    //
                    Graphics2D g2 = (Graphics2D) g.create();
    //
                    int x = (int) ((getWidth() - (img.getWidth() * scale)) / 2);
                    int y = (int) (getHeight() - (img.getHeight() * scale)) / 2;

                    aft.setText("AFT : " + x + " " + y);

                    AffineTransform at = new AffineTransform();
                    at.translate(x, y);
    //                System.out.println(scale);
                    at.scale(scale, scale);

                    g2.setTransform(at);
                    g2.drawImage(img, 0, 0, this);

                    g2.draw(new Line2D.Float(xbegin, ybegin, xend, yend));
                    for (int i = 0; i < lineContainer.size(); i++) {

                        g2.draw(lineContainer.get(i));
                    }
                    g2.setStroke(new BasicStroke(3));
                    for (int i = 0; i < boxContainer.size(); i++) {
                        g2.setColor(Color.BLUE);
                        g2.draw(boxContainer.get(i));
                    }

                    g2.dispose();
    //                System.out.println("int x and y " + x + " " + y);
    //                System.out.println("getWidth and getHeight " + getWidth() + " " + getHeight());
                }
            }
        }
    }

这就是我想用我的代码实现的目标。 我可以在图像顶部动态绘制,并且仍然可以放大/缩小,同时保持绘制的线条。

最佳答案

恕我直言,错误是您在单击时存储绝对值,而不是存储相对位置。 (考虑到比例和平移 - 这就是您在绘制图像时所做的事情)

1) 保持油漆的变形 2)通过反向变换得到它们后计算相对坐标

将这两个添加到 MouseScaleTest 构造函数下方的代码中:

AffineTransform atg;

Point2D calcCoordinates(MouseEvent e)
{
    Point p = new Point(e.getX(), e.getY());
    try
    {
        return atg.inverseTransform(p, null);
    }
    catch (NoninvertibleTransformException ex)
    {
        ex.printStackTrace();
        return p;
    }
}

paintComponent中分配atg:

g2.setTransform(at);
atg = (AffineTransform) at.clone(); // new!

现在到处(mouseDragged、mouseMoved、mouseReleased)你都有类似的代码

xend = e.getX();
yend = e.getY();

改为这样做:

Point2D p = calcCoordinates(e);
xend = (int) p.getX();
yend = (int) p.getY();

如果您喜欢“复制粘贴开发”(不好!),那么这里是一些部分:

addMouseWheelListener(new MouseWheelListener()
{

    @Override
    public void mouseWheelMoved(MouseWheelEvent e)
    {
        int rotation = e.getWheelRotation();
        if (rotation < 0)
        {
            scale -= scaleDelta;
        }
        else
        {
            scale += scaleDelta;
        }
        if (scale < 0)
        {
            scale = 0;
        }
        //                    else if (scale > 1.5) {
        //                        scale = 1;
        //                    }
        scalePoint.setText("Scale: " + String.valueOf(scale));

        repaint();

        //                    System.out.println(getSize());
    }
});

addMouseMotionListener(new MouseMotionListener()
{
    @Override
    public void mouseDragged(MouseEvent e)
    {
        if (isNewLine == false)
        {
            xend = e.getX();
            yend = e.getY();

            //NEW
            Point2D p = calcCoordinates(e);
            xend = (int) p.getX();
            yend = (int) p.getY();

            end.setText("End Values : " + xend + " " + yend);
            repaint();
        }
    }

    @Override
    public void mouseMoved(MouseEvent e)
    {
        lbl.setText("Mouse Point: " + e.getX() + " " + e.getY());
        if (isNewLine == false)
        {
            xend = e.getX();
            yend = e.getY();

            //NEW
            Point2D p = calcCoordinates(e);
            xend = (int) p.getX();
            yend = (int) p.getY();

            repaint();
        }
    }

});

addMouseListener(new MouseAdapter()
{

    @Override
    public void mouseReleased(MouseEvent e)
    {
        System.out.println(e.getX() + " " + e.getY() + "mouse");
        if (isNewLine == true)
        {
            xbegin = xend = e.getX();
            ybegin = yend = e.getY();

            //NEW
            Point2D p = calcCoordinates(e);
            xbegin = xend = (int) p.getX();
            ybegin = yend = (int) p.getY();

            isNewLine = false;
            box = new Rectangle2D.Float(xend - 5, yend - 5, 12, 12);

            boxContainer.add(box);
        }
        else
        {

            linebuffer = new Line2D.Float((float) xbegin, (float) ybegin, (float) xend, (float) yend);

            System.out.println("xbegin: " + xbegin + " " + ybegin + " " + xend + " " + yend);
            lineContainer.add(linebuffer);

            repaint();

            xbegin = e.getX();
            ybegin = e.getY();

            //NEW
            Point2D p = calcCoordinates(e);
            xbegin = (int) p.getX();
            ybegin = (int) p.getY();

            if (box.contains(xend, yend))
            {
                xend = (int) box.getCenterX();
                yend = (int) box.getCenterY();
                isNewLine = true;
                return;
            }

            box = new Rectangle2D.Float(xend - 6, yend - 6, 12, 12);

            boxContainer.add(box);
            return;
        }
    }

关于java - 如何在java中缩放图像上绘图?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57281709/

相关文章:

java - Java Swing中如何显示变化的文本(即显示时间随着秒数的变化)

java - 在 Java GUI 中显示 com 端口数据

Java:检索列表中值的最佳方法

silverlight - 如何将 BitmapImage/WriteableBitmap 保存为 png 格式(Silverlight/windows 手机)?

php - 识别相似图像的好方法?

c++ - 用 QImage 和 QRect 显示图像

java - 如何在java中合并两个BST?

java - jackson XML 注释 : String element with attribute

监控目录的java程序

java - 如何重新绘制我的 JTable