• category: swing folder: PreviewAccessory title: JFileChooserに画像プレビューを追加 tags: [JFileChooser, ImageIcon, PropertyChangeListener] author: aterai pubdate: 2006-11-20T10:35:42+09:00 description: JFileChooserに選択した画像ファイルのプレビューを表示する機能を追加します。 image: https://lh4.googleusercontent.com/_9Z4BYR88imo/TQTRN6_UpeI/AAAAAAAAAg0/eJZas5kcb34/s800/PreviewAccessory.png

概要

JFileChooserに選択した画像ファイルのプレビューを表示する機能を追加します。プレビューを表示するコンポーネントは、チュートリアルのImagePreview.javaをそのまま利用しています。

サンプルコード

fileChooser = new JFileChooser();
fileChooser.setAccessory(new ImagePreview(fileChooser));
// ...

//https://docs.oracle.com/javase/tutorial/uiswing/examples/components/FileChooserDemo2Project/src/components/ImagePreview.java
class ImagePreview extends JComponent implements PropertyChangeListener {
  private static final int PREVIEW_WIDTH  = 90;
  private static final int PREVIEW_MARGIN = 5;
  private ImageIcon thumbnail;
  private File file;
  public ImagePreview(JFileChooser fc) {
    super();
    setPreferredSize(new Dimension(PREVIEW_WIDTH + PREVIEW_MARGIN * 2, 50));
    fc.addPropertyChangeListener(this);
    //setBorder(BorderFactory.createMatteBorder(1, 0, 1, 1, SystemColor.inactiveCaption));
  }
  private void loadImage() {
    if (file == null) {
      thumbnail = null;
      return;
    }
    ImageIcon tmpIcon = new ImageIcon(file.getPath());
    if (tmpIcon.getIconWidth() > PREVIEW_WIDTH) {
      //Image img = tmpIcon.getImage().getScaledInstance(PREVIEW_WIDTH, -1, Image.SCALE_DEFAULT);
      //The Perils of Image.getScaledInstance() | Java.net
      //ttp://today.java.net/pub/a/today/2007/04/03/perils-of-image-getscaledinstance.html
      //The Perils of Image.getScaledInstance() Blog | Oracle Community
      //https://community.oracle.com/docs/DOC-983611
      float scale = PREVIEW_WIDTH / (float) tmpIcon.getIconWidth();
      int newW = (int) (tmpIcon.getIconWidth()  * scale);
      int newH = (int) (tmpIcon.getIconHeight() * scale);
      BufferedImage img = new BufferedImage(newW, newH, BufferedImage.TYPE_INT_ARGB);
      Graphics2D g2 = img.createGraphics();
      g2.setRenderingHint(RenderingHints.KEY_INTERPOLATION,
                          RenderingHints.VALUE_INTERPOLATION_BILINEAR);
      g2.drawImage(tmpIcon.getImage(), 0, 0, newW, newH, null);
      g2.dispose();
      thumbnail = new ImageIcon(img);
    } else {
      thumbnail = tmpIcon;
    }
  }
  @Override public void propertyChange(PropertyChangeEvent e) {
    boolean update = false;
    String prop = e.getPropertyName();
    if (JFileChooser.DIRECTORY_CHANGED_PROPERTY.equals(prop)) {
      file = null;
      update = true;
    } else if (JFileChooser.SELECTED_FILE_CHANGED_PROPERTY.equals(prop)) {
      file = (File) e.getNewValue();
      update = true;
    }
    if (update) {
      thumbnail = null;
      if (isShowing()) {
        loadImage();
        repaint();
      }
    }
  }
  @Override protected void paintComponent(Graphics g) {
    if (thumbnail == null) {
      loadImage();
    }
    if (thumbnail != null) {
      int x = getWidth()  / 2 - thumbnail.getIconWidth()  / 2;
      int y = getHeight() / 2 - thumbnail.getIconHeight() / 2;
      if (y < 0) {
        y = 0;
      }
      if (x < PREVIEW_MARGIN) {
        x = PREVIEW_MARGIN;
      }
      thumbnail.paintIcon(this, g, x, y);
    }
  }
}
View in GitHub: Java, Kotlin

解説

  • JFileChooserのファイルリストで画像ファイルを選択すると、そのプレビューをJFileChooser内に表示する
  • プレビュー表示に使用するコンポーネント(JLabel)はJFileChooser#setAccessory(JComponent)メソッドで追加
  • プレビューコンポーネントはPropertyChangeListenerを実装してファイルの選択、解除などのイベントを受け取る

参考リンク

コメント