【问题标题】:JScrollPane setViewPosition After "Zoom"“缩放”后的 JScrollPane setViewPosition
【发布时间】:2011-01-05 06:25:14
【问题描述】:

我在这里的代码是使用 MouseAdapter 来监听用户在他们想要放大的图像区域周围“绘制”一个框,并计算框与图像的比率。然后它将图像大小调整为计算的比率。这部分有效。

我遇到的问题是让 JScrollPane 视图看起来好像在调整图像大小后它仍然在同一个左上角位置。我尝试了几种方法,似乎已经接近我想要的结果,但并不完全是。

这是查找比例并设置位置的Listener:

import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.Graphics;
import java.awt.Point;

import javax.swing.JScrollPane;
import javax.swing.JViewport;
import javax.swing.JComponent;

public class DynamicZoom extends MouseAdapter {

private Point start;
private Point end;
private double zoom = 1.0;
private JScrollPane pane;
private JViewport port;


public void mousePressed(MouseEvent e) {
    if(e.getButton() == MouseEvent.BUTTON1) {
        this.pane = (JScrollPane)e.getSource();
        this.port = pane.getViewport();
        start = e.getPoint();
    }
}

public void mouseReleased(MouseEvent e) {
    if(this.pane != null) {
        Point curr = this.port.getViewPosition();
        end = e.getPoint();

        ImageComponent canvas = (ImageComponent)this.port.getView();
        zoom = canvas.getScale();
        double factor = 0.0;
        double selectedWidth = Math.abs(end.getX() - start.getX());
        double selectedHeight = Math.abs(end.getY() - start.getY());
        if(selectedWidth > selectedHeight)
            factor = this.port.getWidth() / selectedWidth;
        else
            factor = this.port.getHeight() / selectedHeight;

        zoom *= factor;
        int x = (int)((start.x+curr.x)*zoom);
        int y = (int)((start.y+curr.y)*zoom);

        Point point = new Point(x, y);

        ((ImageComponent)(this.port.getView())).setScale(zoom);

        ResizeViewport.resize(pane);
        this.port.setViewPosition(point);
    }
}

public void mouseDragged(MouseEvent e) {
    if(this.pane != null) {
        Graphics g = this.port.getGraphics();
        int width = this.start.x - e.getX();
        int height = this.start.y - e.getY();

        int w = Math.abs( width );
        int h = Math.abs( height );
        int x = width < 0 ? this.start.x : e.getX();
        int y = height < 0 ? this.start.y : e.getY();
        g.drawRect(x, y, w, h);
        this.port.repaint();
    }
}

}

这是它调整大小并显示图像的 ImageComponent:

import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;

import javax.swing.JComponent;

public class ImageComponent extends JComponent {
private static final long serialVersionUID = 1975488835382044371L;
private BufferedImage img = null;

private double scale = 0.0;

public ImageComponent() {}

public ImageComponent(BufferedImage img) {
    this.displayPage(img);
}

@Override
public void paint(Graphics g) {
    Graphics2D g2 = ((Graphics2D)g);
    if(this.img != null) {
        int width = (int)(this.img.getWidth() * this.scale);
        int height = (int)(this.img.getHeight() * this.scale);
        this.setPreferredSize(new Dimension(width, height));
        g2.drawImage(this.img, 0, 0, width, height, null, null);
        g2.dispose();
    }
}

public void displayPage(BufferedImage pic) {
    if(img != null) {
        this.img = pic;
    }
}

public BufferedImage getPage() {
    return this.img;
}

public void setScale(double ratio) {
    if(ratio > .04) {
        this.scale = ratio;
        this.repaint();
    }
}

public double getScale() {
    return this.scale;
}
}

ResizeViewport 会强制视口在放大时显示整个 ImageComponent,否则它将以原来的大小裁剪图像:

import java.awt.Dimension;

import javax.swing.JScrollPane;
public class ResizeViewport {
public static void resize(JScrollPane scroll) {
    int vw = scroll.getWidth()-scroll.getVerticalScrollBar().getWidth();
    int vh = scroll.getHeight()-scroll.getHorizontalScrollBar().getHeight();
    scroll.getViewport().setViewSize(new Dimension(vw, vh));
}
}

【问题讨论】:

  • 请说明在哪里可以找到 ImageComponent 和 ResizeViewport。
  • 你能发布一些简单的代码来展示它是如何使用的吗?也许我做错了,但我不断收到 ClassCastException,因为 DynamicZoom 尝试将我单击的 JLabel 转换为 JScrollPane。

标签: java image zooming jscrollpane


【解决方案1】:

事实证明,用于计算位置的数学或我设计代码的方式没有任何问题。问题是在计算位置时,ImageComponent 仍在另一个线程中绘制;因此为 getWidth() 和 getHeight() 方法返回“假”值。我使用以下代码解决了这个问题:

EventQueue.invokeLater(new Runnable() {
   public void run() {
      port.setViewPosition(new Point(x,y));
   }
});

这允许在尝试计算图像大小并在 JScrollPane 中设置位置之前完成绘制。问题解决了。我仍然想感谢那些花时间至少审查这个问题的人。

【讨论】:

  • 我知道这个问题现在已经有 10 年历史了,但这只是解决了我几天来一直试图修复的错误。谢谢!
【解决方案2】:

我遇到了类似的问题,但有时面板在调整大小后第一次绘制错误,经过数小时尝试寻找解决方法后,我找到了解决方案!

在滚动窗格中调整面板大小后,最好通过设置为 null 来破坏视口,然后将面板添加回滚动窗格,这将重新创建一个新视口,一切正常!

// Old viewport has to be replaced
scrollPane.setViewport(null);
scrollPane.setViewportView(zoomablePanel);

/安德斯

【讨论】:

    猜你喜欢
    • 2014-05-04
    • 2017-01-21
    • 1970-01-01
    • 1970-01-01
    • 2012-10-20
    • 2010-09-12
    • 1970-01-01
    • 2011-02-12
    • 1970-01-01
    相关资源
    最近更新 更多