OSDN Git Service

ズームパネルのマウス表示判定のスケール対応
[charactermanaj/CharacterManaJ.git] / src / main / java / charactermanaj / ui / PreviewPanel.java
index 8c48b17..43cc911 100644 (file)
-package charactermanaj.ui;\r
-\r
-import static java.lang.Math.*;\r
-\r
-import java.awt.BorderLayout;\r
-import java.awt.Color;\r
-import java.awt.Component;\r
-import java.awt.Cursor;\r
-import java.awt.Dimension;\r
-import java.awt.FontMetrics;\r
-import java.awt.Graphics;\r
-import java.awt.Graphics2D;\r
-import java.awt.GridBagConstraints;\r
-import java.awt.GridBagLayout;\r
-import java.awt.Insets;\r
-import java.awt.Point;\r
-import java.awt.Rectangle;\r
-import java.awt.RenderingHints;\r
-import java.awt.Toolkit;\r
-import java.awt.event.ActionEvent;\r
-import java.awt.event.ActionListener;\r
-import java.awt.event.MouseAdapter;\r
-import java.awt.event.MouseEvent;\r
-import java.awt.event.MouseMotionAdapter;\r
-import java.awt.event.MouseMotionListener;\r
-import java.awt.event.MouseWheelEvent;\r
-import java.awt.event.MouseWheelListener;\r
-import java.awt.geom.Rectangle2D;\r
-import java.awt.image.BufferedImage;\r
-import java.beans.PropertyChangeEvent;\r
-import java.beans.PropertyChangeListener;\r
-import java.util.ArrayList;\r
-import java.util.EventObject;\r
-import java.util.LinkedList;\r
-import java.util.List;\r
-import java.util.Map;\r
-import java.util.Properties;\r
-import java.util.TreeSet;\r
-import java.util.concurrent.Semaphore;\r
-import java.util.logging.Level;\r
-import java.util.logging.Logger;\r
-\r
-import javax.swing.AbstractButton;\r
-import javax.swing.BorderFactory;\r
-import javax.swing.Box;\r
-import javax.swing.Icon;\r
-import javax.swing.JButton;\r
-import javax.swing.JCheckBox;\r
-import javax.swing.JComboBox;\r
-import javax.swing.JLabel;\r
-import javax.swing.JLayeredPane;\r
-import javax.swing.JPanel;\r
-import javax.swing.JScrollBar;\r
-import javax.swing.JScrollPane;\r
-import javax.swing.JSlider;\r
-import javax.swing.JTextField;\r
-import javax.swing.JToolBar;\r
-import javax.swing.JViewport;\r
-import javax.swing.OverlayLayout;\r
-import javax.swing.SwingUtilities;\r
-import javax.swing.Timer;\r
-import javax.swing.UIManager;\r
-import javax.swing.border.Border;\r
-import javax.swing.event.ChangeEvent;\r
-import javax.swing.event.ChangeListener;\r
-import javax.swing.plaf.basic.BasicComboBoxEditor;\r
-\r
-import charactermanaj.Main;\r
-import charactermanaj.graphics.filters.BackgroundColorFilter;\r
-import charactermanaj.graphics.filters.BackgroundColorFilter.BackgroundColorMode;\r
-import charactermanaj.model.AppConfig;\r
-import charactermanaj.ui.util.ScrollPaneDragScrollSupport;\r
-import charactermanaj.util.LocalizedResourcePropertyLoader;\r
-import charactermanaj.util.UIHelper;\r
-\r
-/**\r
- * プレビューパネル\r
- *\r
- * @author seraphy\r
- */\r
-public class PreviewPanel extends JPanel {\r
-\r
-       private static final long serialVersionUID = 1L;\r
-\r
-       protected static final String STRINGS_RESOURCE = "languages/previewpanel";\r
-\r
-\r
-       /**\r
-        * プレビューパネルの上部ツールバーの通知を受けるリスナ\r
-        *\r
-        * @author seraphy\r
-        */\r
-       public interface PreviewPanelListener {\r
-\r
-               /**\r
-                * 保存\r
-                *\r
-                * @param e\r
-                */\r
-               void savePicture(PreviewPanelEvent e);\r
-\r
-               /**\r
-                * コピー\r
-                *\r
-                * @param e\r
-                */\r
-               void copyPicture(PreviewPanelEvent e);\r
-\r
-               /**\r
-                * 背景色変更\r
-                *\r
-                * @param e\r
-                */\r
-               void changeBackgroundColor(PreviewPanelEvent e);\r
-\r
-               /**\r
-                * 情報\r
-                *\r
-                * @param e\r
-                */\r
-               void showInformation(PreviewPanelEvent e);\r
-\r
-               /**\r
-                * お気に入りに追加\r
-                *\r
-                * @param e\r
-                */\r
-               void addFavorite(PreviewPanelEvent e);\r
-\r
-               /**\r
-                * 左右反転\r
-                *\r
-                * @param e\r
-                */\r
-               void flipHorizontal(PreviewPanelEvent e);\r
-       }\r
-\r
-       /**\r
-        * ロード中を示すインジケータ\r
-        */\r
-       private final String indicatorText;\r
-\r
-       /**\r
-        * ロード中であるか判定するタイマー\r
-        */\r
-       private final Timer timer;\r
-\r
-       /**\r
-        * インジケータを表示するまでのディレイ\r
-        */\r
-       private long indicatorDelay;\r
-\r
-       @Override\r
-       public void addNotify() {\r
-               super.addNotify();\r
-               if (!timer.isRunning()) {\r
-                       timer.start();\r
-               }\r
-       }\r
-\r
-       @Override\r
-       public void removeNotify() {\r
-               if (timer.isRunning()) {\r
-                       timer.stop();\r
-               }\r
-               super.removeNotify();\r
-       }\r
-\r
-       public static class PreviewPanelEvent extends EventObject {\r
-\r
-               private static final long serialVersionUID = 1L;\r
-\r
-               private int modifiers;\r
-\r
-               public PreviewPanelEvent(Object src, ActionEvent e) {\r
-                       this(src, (e == null) ? 0 : e.getModifiers());\r
-               }\r
-\r
-               public PreviewPanelEvent(Object src, int modifiers) {\r
-                       super(src);\r
-                       this.modifiers = modifiers;\r
-               }\r
-\r
-               public int getModifiers() {\r
-                       return modifiers;\r
-               }\r
-\r
-               public boolean isShiftKeyPressed() {\r
-                       return (modifiers & ActionEvent.SHIFT_MASK) != 0;\r
-               }\r
-       }\r
-\r
-       private final Object lock = new Object();\r
-\r
-       private long loadingTicket;\r
-\r
-       private long loadedTicket;\r
-\r
-       private long firstWaitingTimestamp;\r
-\r
-       private boolean indicatorShown;\r
-\r
-       private String title;\r
-\r
-       private JLabel lblTitle;\r
-\r
-       private JLayeredPane layeredPane;\r
-\r
-       private CheckInfoLayerPanel checkInfoLayerPanel;\r
-\r
-       private PreviewImagePanel previewImgPanel;\r
-\r
-       private JScrollPane previewImgScrollPane;\r
-\r
-       private ScrollPaneDragScrollSupport scrollSupport;\r
-\r
-       private PreviewControlPanel previewControlPanel;\r
-\r
-       private double latestToggleZoom = 2.;\r
-\r
-       private LinkedList<PreviewPanelListener> listeners = new LinkedList<PreviewPanelListener>();\r
-\r
-\r
-       public PreviewPanel() {\r
-               setLayout(new BorderLayout());\r
-\r
-               final AppConfig appConfig = AppConfig.getInstance();\r
-               final Properties strings = LocalizedResourcePropertyLoader.getCachedInstance()\r
-                       .getLocalizedProperties(STRINGS_RESOURCE);\r
-\r
-               // 画像をロード中であることを示すインジケータの確認サイクル.\r
-               timer = new Timer(100, new ActionListener() {\r
-                               public void actionPerformed(ActionEvent e) {\r
-                                       onTimer();\r
-                               }\r
-                       });\r
-\r
-               indicatorText = strings.getProperty("indicatorText");\r
-               indicatorDelay = appConfig.getPreviewIndicatorDelay();\r
-\r
-               UIHelper uiUtl = UIHelper.getInstance();\r
-               JButton saveBtn = uiUtl.createIconButton("icons/save.png");\r
-               JButton copyBtn = uiUtl.createIconButton("icons/copy.png");\r
-               JButton colorBtn = uiUtl.createIconButton("icons/color.png");\r
-               JButton informationBtn = uiUtl.createIconButton("icons/information.png");\r
-               JButton favoriteBtn = uiUtl.createIconButton("icons/favorite.png");\r
-               JButton flipHolizontalBtn = uiUtl.createIconButton("icons/flip.png");\r
-\r
-               saveBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               savePicture(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-               copyBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               copyPicture(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-               colorBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               changeBackgroundColor(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-               informationBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               showInformation(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-               favoriteBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               addFavorite(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-               flipHolizontalBtn.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               flipHolizontal(new PreviewPanelEvent(PreviewPanel.this, e));\r
-                       }\r
-               });\r
-\r
-               saveBtn.setToolTipText(strings.getProperty("tooltip.save"));\r
-               copyBtn.setToolTipText(strings.getProperty("tooltip.copy"));\r
-               colorBtn.setToolTipText(strings.getProperty("tooltip.changeBgColor"));\r
-               informationBtn.setToolTipText(strings.getProperty("tooltip.showInformation"));\r
-               favoriteBtn.setToolTipText(strings.getProperty("tooltip.registerFavorites"));\r
-               flipHolizontalBtn.setToolTipText(strings.getProperty("tooltip.flipHorizontal"));\r
-\r
-               final JToolBar toolBar = new JToolBar();\r
-               toolBar.setFloatable(false);\r
-               toolBar.add(flipHolizontalBtn);\r
-               toolBar.add(copyBtn);\r
-               toolBar.add(saveBtn);\r
-               toolBar.add(Box.createHorizontalStrut(8));\r
-               toolBar.add(colorBtn);\r
-               toolBar.add(Box.createHorizontalStrut(4));\r
-               toolBar.add(favoriteBtn);\r
-               toolBar.add(informationBtn);\r
-\r
-               lblTitle = new JLabel() {\r
-                       private static final long serialVersionUID = 1L;\r
-\r
-                       public Dimension getPreferredSize() {\r
-                               Dimension dim = super.getPreferredSize();\r
-                               int maxWidth = getParent().getWidth() - toolBar.getWidth();\r
-                               if (dim.width > maxWidth) {\r
-                                       dim.width = maxWidth;\r
-                               }\r
-                               return dim;\r
-                       };\r
-\r
-                       public Dimension getMaximumSize() {\r
-                               return getPreferredSize();\r
-                       };\r
-\r
-                       public Dimension getMinimumSize() {\r
-                               Dimension dim = getPreferredSize();\r
-                               dim.width = 50;\r
-                               return dim;\r
-                       };\r
-               };\r
-\r
-               lblTitle.setBorder(BorderFactory.createEmptyBorder(3, 10, 3, 3));\r
-\r
-               JPanel previewPaneHeader = new JPanel();\r
-               previewPaneHeader.setLayout(new BorderLayout());\r
-               previewPaneHeader.add(lblTitle, BorderLayout.WEST);\r
-               previewPaneHeader.add(toolBar, BorderLayout.EAST);\r
-\r
-               previewImgPanel = new PreviewImagePanel();\r
-\r
-\r
-               previewImgScrollPane = new JScrollPane(previewImgPanel);\r
-               previewImgScrollPane.setAutoscrolls(false);\r
-               previewImgScrollPane.setWheelScrollingEnabled(false);\r
-               previewImgScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_ALWAYS);\r
-               previewImgScrollPane.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS);\r
-\r
-               scrollSupport = new ScrollPaneDragScrollSupport(previewImgScrollPane) {\r
-                       @Override\r
-                       protected void setCursor(Cursor cursor) {\r
-                               PreviewPanel.this.setCursor(cursor);\r
-                       }\r
-               };\r
-\r
-               add(previewPaneHeader, BorderLayout.NORTH);\r
-\r
-               layeredPane = new JLayeredPane();\r
-               layeredPane.setLayout(new OverlayLayout(layeredPane));\r
-\r
-               layeredPane.add(previewImgScrollPane, JLayeredPane.DEFAULT_LAYER);\r
-\r
-               checkInfoLayerPanel = new CheckInfoLayerPanel();\r
-               layeredPane.add(checkInfoLayerPanel, JLayeredPane.POPUP_LAYER);\r
-               checkInfoLayerPanel.setVisible(false);\r
-\r
-               add(layeredPane, BorderLayout.CENTER);\r
-\r
-               previewControlPanel = new PreviewControlPanel();\r
-               Dimension dim = previewControlPanel.getPreferredSize();\r
-               Dimension prevDim = previewImgScrollPane.getPreferredSize();\r
-               dim.width = prevDim.width;\r
-               previewControlPanel.setPreferredSize(dim);\r
-\r
-               add(previewControlPanel, BorderLayout.SOUTH);\r
-               previewControlPanel.setPinned(appConfig.isEnableZoomPanel());\r
-\r
-               // 倍率が変更された場合\r
-               previewControlPanel.addPropertyChangeListener("zoomFactorInt", new PropertyChangeListener() {\r
-                       public void propertyChange(PropertyChangeEvent evt) {\r
-                               Integer newValue = (Integer) evt.getNewValue();\r
-                               zoomWithCenterPosition(newValue.doubleValue() / 100., null);\r
-                       }\r
-               });\r
-               // 背景モードが切り替えられた場合\r
-               previewControlPanel.addPropertyChangeListener("backgroundColorMode", new PropertyChangeListener() {\r
-                       public void propertyChange(PropertyChangeEvent evt) {\r
-                               BackgroundColorMode bgColorMode = (BackgroundColorMode) evt.getNewValue();\r
-                               previewImgPanel.setBackgroundColorMode(bgColorMode);\r
-                               if (bgColorMode != BackgroundColorMode.ALPHABREND\r
-                                               && appConfig.isEnableCheckInfoTooltip() ) {\r
-                                                       // チェック情報ツールチップの表示\r
-                                       checkInfoLayerPanel.setMessage(null);\r
-                                       checkInfoLayerPanel.setVisible(true);\r
-\r
-                               } else {\r
-                                                       // チェック情報ツールチップの非表示\r
-                                       checkInfoLayerPanel.setVisible(false);\r
-                               }\r
-                       }\r
-               });\r
-\r
-               previewImgScrollPane.addMouseMotionListener(new MouseMotionAdapter() {\r
-                       @Override\r
-                       public void mouseMoved(MouseEvent e) {\r
-                               Rectangle rct = previewImgScrollPane.getBounds();\r
-                               int y = e.getY();\r
-                               if (y > rct.height - appConfig.getZoomPanelActivationArea()) {\r
-                                       previewControlPanel.setVisible(true);\r
-                               } else {\r
-                                       if ( !previewControlPanel.isPinned()) {\r
-                                               previewControlPanel.setVisible(false);\r
-                                       }\r
-                               }\r
-                       }\r
-               });\r
-\r
-               // 標準のホイールリスナは削除する.\r
-               for (final MouseWheelListener listener : previewImgScrollPane.getMouseWheelListeners()) {\r
-                       previewImgScrollPane.removeMouseWheelListener(listener);\r
-               }\r
-\r
-               previewImgScrollPane.addMouseWheelListener(new MouseWheelListener() {\r
-                       public void mouseWheelMoved(MouseWheelEvent e) {\r
-                               if ((Main.isMacOSX() && e.isAltDown()) ||\r
-                                               ( !Main.isMacOSX() && e.isControlDown())) {\r
-                                       // Mac OS XならOptionキー、それ以外はコントロールキーとともにホイールスクロールの場合\r
-                                       zoomByWheel(e);\r
-                               } else {\r
-                                       // ズーム以外のホイール操作はスクロールとする.\r
-                                       scrollByWheel(e);\r
-                               }\r
-                               // 現在画像位置の情報の更新\r
-                               updateCheckInfoMessage(e.getPoint());\r
-                       }\r
-               });\r
-\r
-               previewImgScrollPane.addMouseListener(new MouseAdapter() {\r
-                       @Override\r
-                       public void mousePressed(MouseEvent e) {\r
-                               if (e.getClickCount() == 2) {\r
-                                       // ダブルクリック\r
-                                       // (正確に2回目。3回目以降はダブルクリック + シングルクリック)\r
-                                       toggleZoom(e.getPoint());\r
-                               } else {\r
-                                       scrollSupport.drag(true, e.getPoint());\r
-                               }\r
-                       }\r
-                       @Override\r
-                       public void mouseReleased(MouseEvent e) {\r
-                               scrollSupport.drag(false, e.getPoint());\r
-                       }\r
-               });\r
-\r
-               previewImgScrollPane.addMouseMotionListener(new MouseMotionListener() {\r
-\r
-                       public void mouseMoved(MouseEvent e) {\r
-                               updateCheckInfoMessage(e.getPoint());\r
-                       }\r
-\r
-                       public void mouseDragged(MouseEvent e) {\r
-                               scrollSupport.dragging(e.getPoint());\r
-\r
-                               // 現在画像位置の情報の更新\r
-                               updateCheckInfoMessage(e.getPoint());\r
-                       }\r
-               });\r
-       }\r
-\r
-       /**\r
-        * 倍率を切り替える.\r
-        */\r
-       protected void toggleZoom(Point mousePos) {\r
-               if (previewImgPanel.isDefaultZoom()) {\r
-                       // 等倍であれば以前の倍率を適用する.\r
-                       zoomWithCenterPosition(latestToggleZoom, mousePos);\r
-\r
-               } else {\r
-                       // 等倍でなければ現在の倍率を記憶して等倍にする.\r
-                       double currentZoomFactor = previewImgPanel.getZoomFactor();\r
-                       latestToggleZoom = currentZoomFactor;\r
-                       zoomWithCenterPosition(1., mousePos);\r
-               }\r
-       }\r
-\r
-       /**\r
-        * マウス位置に対して画像情報のツールチップを表示する\r
-        *\r
-        * @param mousePosition\r
-        *            マウス位置\r
-        */\r
-       protected void updateCheckInfoMessage(Point mousePosition) {\r
-               if ( !checkInfoLayerPanel.isVisible()) {\r
-                       return;\r
-               }\r
-               // マウス位置から画像位置を割り出す\r
-               Point imgPos = null;\r
-               if (mousePosition != null) {\r
-                       Point panelPt = SwingUtilities.convertPoint(previewImgScrollPane,\r
-                                       mousePosition, previewImgPanel);\r
-                       imgPos = previewImgPanel.getImagePosition(panelPt);\r
-               }\r
-               if (imgPos != null) {\r
-                       // 画像位置があれば、その位置の情報を取得する.\r
-                       int argb = previewImgPanel.getImageARGB(imgPos);\r
-                       int a = (argb >> 24) & 0xff;\r
-                       int r = (argb >> 16) & 0xff;\r
-                       int g = (argb >> 8) & 0xff;\r
-                       int b = argb & 0xff;\r
-                       int y = (int) (0.298912f * r + 0.586611f * g + 0.114478f * b);\r
-                       String text = String.format(\r
-                                       "(%3d,%3d)¥nA:%3d, Y:%3d¥nR:%3d, G:%3d, B:%3d", imgPos.x,\r
-                                       imgPos.y, a, y, r, g, b);\r
-                       checkInfoLayerPanel.setMessage(text);\r
-                       checkInfoLayerPanel.setPotision(mousePosition);\r
-\r
-               } else {\r
-                       // 画像位置がなければツールチップは空にする.\r
-                       checkInfoLayerPanel.setMessage(null);\r
-               }\r
-       }\r
-\r
-       /**\r
-        * マウス座標単位で指定したオフセット分スクロールする.\r
-        *\r
-        * @param diff_x\r
-        *            水平方向スクロール数\r
-        * @param diff_y\r
-        *            垂直方向スクロール数\r
-        */\r
-       protected void scroll(int diff_x, int diff_y) {\r
-               scrollSupport.scroll(diff_x, diff_y);\r
-       }\r
-\r
-       /**\r
-        * マウスホイールによる水平・垂直スクロール.<br>\r
-        * シフトキーで水平、それ以外は垂直とする.<br>\r
-        *\r
-        * @param e\r
-        *            ホイールイベント\r
-        */\r
-       protected void scrollByWheel(final MouseWheelEvent e) {\r
-               scrollSupport.scrollByWheel(e);\r
-\r
-               // イベントは処理済みとする.\r
-               e.consume();\r
-       }\r
-\r
-       /**\r
-        * ホイールによる拡大縮小.<br>\r
-        * ホイールの量は関係なく、方向だけで判定する.<br>\r
-        * プラットフォームごとに修飾キーの判定が異なるので、 呼び出しもとであらかじめ切り分けて呼び出すこと.<br>\r
-        *\r
-        * @param e\r
-        *            ホイールイベント\r
-        */\r
-       protected void zoomByWheel(final MouseWheelEvent e) {\r
-               int wheelRotation = e.getWheelRotation();\r
-               double currentZoom = previewImgPanel.getZoomFactor();\r
-               double zoomFactor;\r
-               if (wheelRotation < 0) {\r
-                       // ホイール上で拡大\r
-                       zoomFactor = currentZoom * 1.1;\r
-\r
-               } else if (wheelRotation > 0){\r
-                       // ホイール下で縮小\r
-                       zoomFactor = currentZoom * 0.9;\r
-\r
-               } else {\r
-                       return;\r
-               }\r
-\r
-               // 倍率変更する\r
-               zoomWithCenterPosition(zoomFactor, e.getPoint());\r
-\r
-               // イベント処理済み\r
-               e.consume();\r
-       }\r
-\r
-       /**\r
-        * ズームスライダまたはコンボのいずれかの値を更新すると、他方からも更新通知があがるため 二重処理を防ぐためのセマフォ.<br>\r
-        */\r
-       private Semaphore zoomLock = new Semaphore(1);\r
-\r
-       /**\r
-        * プレビューに表示する画像の倍率を更新する.<br>\r
-        * 指定した座標が拡大縮小の中心点になるようにスクロールを試みる.<br>\r
-        * 座標がnullの場合は現在表示されている中央を中心とするようにスクロールを試みる.<br>\r
-        * (スクロールバーが表示されていない、もしくは十分にスクロールできない場合は必ずしも中心とはならない.)<br>\r
-        * コントロールパネルの表示値も更新する.<br>\r
-        * コントロールパネルからの更新通知をうけて再入しないように、 同時に一つしか実行されないようにしている.<br>\r
-        *\r
-        * @param zoomFactor\r
-        *            倍率、範囲外のものは範囲内に補正される.\r
-        * @param mousePos\r
-        *            スクロールペイン上のマウス座標、もしくはnull(nullの場合は表示中央)\r
-        */\r
-       protected void zoomWithCenterPosition(double zoomFactor, Point mousePos) {\r
-               if ( !zoomLock.tryAcquire()) {\r
-                       return;\r
-               }\r
-               try {\r
-                       // 範囲制限.\r
-                       if (zoomFactor < 0.2) {\r
-                               zoomFactor = 0.2;\r
-                       } else if (zoomFactor > 8.) {\r
-                               zoomFactor = 8.;\r
-                       }\r
-\r
-                       JViewport vp = previewImgScrollPane.getViewport();\r
-\r
-                       Point viewCenter;\r
-                       if (mousePos != null) {\r
-                               // スクロールペインのマウス座標を表示パネルの位置に換算する.\r
-                               viewCenter = SwingUtilities.convertPoint(this, mousePos, previewImgPanel);\r
-\r
-                       } else {\r
-                               // 表示パネル上の現在表示しているビューポートの中央の座標を求める\r
-                               Rectangle viewRect = vp.getViewRect();\r
-                               viewCenter = new Point(\r
-                                               (viewRect.x + viewRect.width / 2),\r
-                                               (viewRect.y + viewRect.height / 2)\r
-                                               );\r
-                       }\r
-\r
-                       // 現在のビューサイズ(余白があれば余白も含む)\r
-                       Dimension viewSize = previewImgPanel.getScaledSize(true);\r
-\r
-                       // 倍率変更\r
-                       previewControlPanel.setZoomFactor(zoomFactor);\r
-                       previewImgPanel.setZoomFactor(zoomFactor);\r
-\r
-                       // 新しいのビューサイズ(余白があれば余白も含む)\r
-                       Dimension viewSizeAfter = previewImgPanel.getScaledSize(true);\r
-                       Dimension visibleSize = vp.getExtentSize();\r
-\r
-                       if (viewSize != null && viewSizeAfter != null &&\r
-                               viewSizeAfter.width > 0 && viewSizeAfter.height > 0 &&\r
-                               viewSizeAfter.width > visibleSize.width &&\r
-                               viewSizeAfter.height > visibleSize.height) {\r
-                               // 新しいビューの大きさよりも表示可能領域が小さい場合のみ\r
-                               vp.setViewSize(viewSizeAfter);\r
-\r
-                               // スクロールペインに表示されている画面サイズを求める.\r
-                               // スクロールバーがある方向は、コンテンツの最大と等しいが\r
-                               // スクロールバーがない場合は画面サイズのほうが大きいため、\r
-                               // 倍率変更による縦横の移動比は、それぞれ異なる.\r
-                               int visible_width = max(visibleSize.width, viewSize.width);\r
-                               int visible_height = max(visibleSize.height, viewSize.height);\r
-                               int visible_width_after = max(visibleSize.width, viewSizeAfter.width);\r
-                               int visible_height_after = max(visibleSize.height, viewSizeAfter.height);\r
-\r
-                               // 前回の倍率から今回の倍率の倍率.\r
-                               // オリジナルに対する倍率ではない.\r
-                               // また、画像は縦横同率であるが表示ウィンドウはスクロールバー有無により同率とは限らない.\r
-                               double zoomDiffX = (double) visible_width_after / (double) visible_width;\r
-                               double zoomDiffY = (double) visible_height_after / (double) visible_height;\r
-\r
-                               // 拡大後の座標の補正\r
-                               Point viewCenterAfter = new Point();\r
-                               viewCenterAfter.x = (int) round(viewCenter.x * zoomDiffX);\r
-                               viewCenterAfter.y = (int) round(viewCenter.y * zoomDiffY);\r
-\r
-                               // 倍率適用前後の座標の差分\r
-                               int diff_x = viewCenterAfter.x - viewCenter.x;\r
-                               int diff_y = viewCenterAfter.y - viewCenter.y;\r
-\r
-                               // スクロール\r
-                               scroll(diff_x, diff_y);\r
-                       }\r
-\r
-                       // スクロールの単位を画像1ドットあたりの表示サイズに変更する.\r
-                       // (ただし1を下回らない)\r
-                       JScrollBar vsb = previewImgScrollPane.getVerticalScrollBar();\r
-                       JScrollBar hsb = previewImgScrollPane.getHorizontalScrollBar();\r
-                       vsb.setUnitIncrement(max(1, (int) ceil(zoomFactor)));\r
-                       hsb.setUnitIncrement(max(1, (int) ceil(zoomFactor)));\r
-\r
-               } finally {\r
-                       zoomLock.release();\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 現在のビューの左上位置を返す\r
-        * @return\r
-        */\r
-       public Point getViewPosition() {\r
-               JViewport vp = previewImgScrollPane.getViewport();\r
-               return vp.getViewPosition();\r
-       }\r
-\r
-       /**\r
-        * 指定した座標が中央となるようにスクロールする。\r
-        * まだ画像が表示されていない場合は次に画像を設定したときに行う。\r
-        * @param centerPt 中央\r
-        */\r
-       public void setViewPosition(Point viewPt) {\r
-               JViewport vp = previewImgScrollPane.getViewport();\r
-               if (previewImgPanel.getPreviewImage() != null) {\r
-                       if (viewPt != null) {\r
-                               vp.setViewPosition(viewPt);\r
-                       }\r
-                       requestViewPt = null;\r
-               } else {\r
-                       requestViewPt = viewPt;\r
-               }\r
-       }\r
-\r
-       private Point requestViewPt;\r
-\r
-       /**\r
-        * プレビューに表示するタイトル.<br>\r
-        *\r
-        * @param title\r
-        *            タイトル\r
-        */\r
-       public void setTitle(String title) {\r
-               if (title == null) {\r
-                       title = "";\r
-               }\r
-               if (!title.equals(this.title)) {\r
-                       this.title = title;\r
-                       lblTitle.setText(title + (indicatorShown ? indicatorText : ""));\r
-                       lblTitle.setToolTipText(title);\r
-               }\r
-       }\r
-\r
-       public String getTitle() {\r
-               return this.title;\r
-       }\r
-\r
-       /**\r
-        * ロードに時間がかかっているか判定し、 インジケータを表示するためのタイマーイベントハンドラ.<br>\r
-        */\r
-       protected void onTimer() {\r
-               boolean waiting;\r
-               long firstRequest;\r
-               synchronized (lock) {\r
-                       waiting = isWaiting();\r
-                       firstRequest = firstWaitingTimestamp;\r
-               }\r
-               boolean indicatorShown = (waiting && ((System.currentTimeMillis() - firstRequest) > indicatorDelay));\r
-               if (this.indicatorShown != indicatorShown) {\r
-                       this.indicatorShown = indicatorShown;\r
-                       lblTitle.setText(title + (indicatorShown ? indicatorText : ""));\r
-               }\r
-       }\r
-\r
-       /**\r
-        * チケットの状態が、ロード完了待ち状態であるか?<br>\r
-        * ロード中のチケットが、ロード完了のチケットより新しければロード中と見なす.<br>\r
-        *\r
-        * @return 完了待ちであればtrue、そうでなければfalse\r
-        */\r
-       protected boolean isWaiting() {\r
-               synchronized (lock) {\r
-                       return loadingTicket > loadedTicket;\r
-               }\r
-       }\r
-\r
-       /**\r
-        * ロード要求が出されるたびに、そのロード要求チケットを登録する.<br>\r
-        * チケットは要求されるたびに増加するシーケンスとする.<br>\r
-        *\r
-        * @param ticket\r
-        *            ロード要求チケット\r
-        */\r
-       public void setLoadingRequest(long ticket) {\r
-               synchronized (lock) {\r
-                       if ( !isWaiting() && this.loadedTicket < ticket) {\r
-                               // 現在認識しているチケットの状態がロード完了であり、\r
-                               // それよりも新しいチケットが要求されたならば、\r
-                               // 今回のチケットから待ち時間の計測を開始する.\r
-                               this.firstWaitingTimestamp = System.currentTimeMillis();\r
-                       }\r
-                       this.loadingTicket = ticket;\r
-               }\r
-       }\r
-\r
-       /**\r
-        * ロード完了するたびに呼び出される.<br>\r
-        *\r
-        * @param ticket\r
-        *            ロード要求チケット.\r
-        */\r
-       public void setLoadingComplete(long ticket) {\r
-               synchronized (lock) {\r
-                       this.loadedTicket = ticket;\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 表示画像を設定する.<br>\r
-        *\r
-        * @param previewImg\r
-        *            表示画像、もしくはnull\r
-        */\r
-       public void setPreviewImage(BufferedImage previewImg) {\r
-               previewImgPanel.setPreviewImage(previewImg);\r
-               if (requestViewPt != null) {\r
-                       // 画像設定前にスクロール位置の要求があれば、再適用を試みる\r
-                       setViewPosition(requestViewPt);\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 表示されている画像を取得する.<br>\r
-        * 表示画像が設定されていなければnull.<br>\r
-        *\r
-        * @return 表示画像、もしくはnull\r
-        */\r
-       public BufferedImage getPreviewImage() {\r
-               return previewImgPanel.getPreviewImage();\r
-       }\r
-\r
-       /**\r
-        * 表示している画面イメージそのままを取得する.\r
-        *\r
-        * @return 表示画像\r
-        */\r
-       public BufferedImage getScreenImage() {\r
-               JViewport vp = previewImgScrollPane.getViewport();\r
-               Dimension dim = vp.getExtentSize();\r
-               BufferedImage img = new BufferedImage(dim.width, dim.height, BufferedImage.TYPE_INT_ARGB);\r
-               Graphics2D g = img.createGraphics();\r
-               try {\r
-                       vp.paint(g);\r
-\r
-               } finally {\r
-                       g.dispose();\r
-               }\r
-               return img;\r
-       }\r
-\r
-       /**\r
-        * 壁紙を設定する.<br>\r
-        *\r
-        * @param wallpaperImg\r
-        *            壁紙、null不可\r
-        */\r
-       public void setWallpaper(Wallpaper wallpaper) {\r
-               previewImgPanel.setWallpaper(wallpaper);\r
-       }\r
-\r
-       /**\r
-        * 壁紙を取得する.<br>\r
-        * 壁紙が未設定の場合は空の壁紙インスタンスが返される.<br>\r
-        *\r
-        * @return 壁紙\r
-        */\r
-       public Wallpaper getWallpaper() {\r
-               return previewImgPanel.getWallpaper();\r
-       }\r
-\r
-       /**\r
-        * 表示倍率を取得する.\r
-        *\r
-        * @return 表示倍率\r
-        */\r
-       public double getZoomFactor() {\r
-               return previewControlPanel.getZoomFactor();\r
-       }\r
-\r
-       /**\r
-        * 表示倍率を設定する\r
-        *\r
-        * @param zoomFactor\r
-        *            表示倍率\r
-        */\r
-       public void setZoomFactor(double zoomFactor) {\r
-               previewControlPanel.setZoomFactor(zoomFactor);\r
-       }\r
-\r
-       /**\r
-        * ズームパネルのピン留め制御\r
-        *\r
-        * @param visible\r
-        *            表示する場合はtrue\r
-        */\r
-       public void setVisibleZoomBox(boolean visible) {\r
-               previewControlPanel.setPinned(visible);\r
-       }\r
-\r
-       /**\r
-        * ズームパネルがピン留めされているか?\r
-        *\r
-        * @return ピン留めされていればtrue\r
-        */\r
-       public boolean isVisibleZoomBox() {\r
-               return previewControlPanel.isPinned();\r
-       }\r
-\r
-       public void addPreviewPanelListener(PreviewPanelListener listener) {\r
-               if (listener == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-               listeners.add(listener);\r
-       }\r
-\r
-       public void removePreviewPanelListener(PreviewPanelListener listener) {\r
-               listeners.remove(listener);\r
-       }\r
-\r
-       protected void savePicture(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.savePicture(e);\r
-               }\r
-       }\r
-\r
-       protected void flipHolizontal(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.flipHorizontal(e);\r
-               }\r
-       }\r
-\r
-       protected void copyPicture(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.copyPicture(e);\r
-               }\r
-       }\r
-\r
-       protected void changeBackgroundColor(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.changeBackgroundColor(e);\r
-               }\r
-       }\r
-\r
-       protected void showInformation(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.showInformation(e);\r
-               }\r
-       }\r
-\r
-       protected void addFavorite(PreviewPanelEvent e) {\r
-               for (PreviewPanelListener listener : listeners) {\r
-                       listener.addFavorite(e);\r
-               }\r
-       }\r
-}\r
-\r
-/**\r
- * チェック情報の表示用レイヤーパネル.<br>\r
- *\r
- * @author seraphy\r
- */\r
-class CheckInfoLayerPanel extends JPanel {\r
-       private static final long serialVersionUID = 1L;\r
-\r
-       /**\r
-        * ロガー\r
-        */\r
-       private static final Logger logger = Logger.getLogger(CheckInfoLayerPanel.class.getName());\r
-\r
-       /**\r
-        * ボックスの余白\r
-        */\r
-       private Insets padding = new Insets(3, 3, 3, 3);\r
-\r
-       /**\r
-        * 表示位置プロパティ\r
-        */\r
-       private Point pos = new Point();\r
-\r
-       /**\r
-        * 表示メッセージプロパティ.<br>\r
-        * ¥nで改行となる.<br>\r
-        * 空文字ならば非表示.<br>\r
-        */\r
-       private String message = "";\r
-\r
-       /**\r
-        * 解析済みメッセージ.<br>\r
-        * 業に分割される.<br>\r
-        * 空文字は空のリストとなる.<br>\r
-        */\r
-       private String[] messageLines;\r
-\r
-       /**\r
-        * 解析済みフォントの高さ.<br>\r
-        */\r
-       private int fontHeight;\r
-\r
-       /**\r
-        * 描画済みエリア.<br>\r
-        * 次回描画前に消去する必要のある領域.<br>\r
-        * まだ一度も描画してなければnull.<br>\r
-        */\r
-       private Rectangle eraseRect;\r
-\r
-       /**\r
-        * 現在、描画すべきエリア.<br>\r
-        * なければnull.<br>\r
-        */\r
-       private Rectangle requestRect;\r
-\r
-       /**\r
-        * 画面に関連づけられていない状態でのテキスト表示サイズは 計算できないため、画面追加時に再計算させるための 予約フラグ.<br>\r
-        */\r
-       private boolean requestRecalcOnAdd;\r
-\r
-       /**\r
-        * フォントのためのデスクトップヒント.(あれば)\r
-        */\r
-       @SuppressWarnings("rawtypes")\r
-       private Map desktopHintsForFont;\r
-\r
-       /**\r
-        * 透明コンポーネントとして構築する.<br>\r
-        */\r
-       @SuppressWarnings("rawtypes")\r
-       public CheckInfoLayerPanel() {\r
-               setOpaque(false);\r
-\r
-               Toolkit tk = Toolkit.getDefaultToolkit();\r
-               desktopHintsForFont = (Map) tk.getDesktopProperty("awt.font.desktophints");\r
-               logger.log(Level.CONFIG, "awt.font.desktophints=" + desktopHintsForFont);\r
-       }\r
-\r
-       /**\r
-        * 指定エリアに情報を描画する.<br>\r
-        */\r
-       @Override\r
-       protected void paintComponent(Graphics g0) {\r
-               Graphics2D g = (Graphics2D) g0;\r
-               super.paintComponent(g);\r
-\r
-               // クリップ領域\r
-               Rectangle clip = g.getClipBounds();\r
-               // System.out.println("clip:" + clip + " /eraseRect:" + eraseRect + " /drawRect:" + requestRect);\r
-\r
-               // 削除すべき領域が描画範囲に含まれているか?\r
-               // (含まれていれば、その領域は消去済みである.)\r
-               if (clip == null || (eraseRect != null && clip.contains(eraseRect))) {\r
-                       eraseRect = null;\r
-               }\r
-\r
-               // 表示領域の判定\r
-               if (requestRect == null || requestRect.isEmpty()\r
-                               || !(clip != null && clip.intersects(requestRect))) {\r
-                       // 表示すべき領域が存在しないか、描画要求範囲にない.\r
-                       return;\r
-               }\r
-               if (messageLines == null || messageLines.length == 0) {\r
-                       // 表示するものがなければ何もしない.\r
-                       return;\r
-               }\r
-\r
-               // フォントのレンダリングヒント\r
-               if (desktopHintsForFont != null) {\r
-                       g.addRenderingHints(desktopHintsForFont);\r
-               }\r
-\r
-               // 箱の描画\r
-               g.setColor(new Color(255, 255, 255, 192));\r
-               g.fillRect(requestRect.x, requestRect.y, requestRect.width, requestRect.height);\r
-               g.setColor(Color.GRAY);\r
-               g.drawRect(requestRect.x, requestRect.y, requestRect.width - 1, requestRect.height - 1);\r
-\r
-               // 情報の描画\r
-               g.setColor(Color.BLACK);\r
-               int oy = fontHeight;\r
-               for (String messageLine : messageLines) {\r
-                       g.drawString(messageLine, requestRect.x + padding.left, requestRect.y + padding.top - 1 + oy);\r
-                       oy += fontHeight;\r
-               }\r
-\r
-               // 描画された領域を次回消去領域として記憶する.\r
-               if (eraseRect == null || eraseRect.isEmpty()) {\r
-                       // 消去済みであれば、今回分のみを次回消去領域とする.\r
-                       eraseRect = (Rectangle) requestRect.clone();\r
-\r
-               } else {\r
-                       // 消去済みエリアが未消去で残っている場合は\r
-                       // 今回領域を結合する.\r
-                       eraseRect.add(requestRect);\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 画面にアタッチされた場合、描画領域の再計算が 必要であれば計算する.<br>\r
-        */\r
-       @Override\r
-       public void addNotify() {\r
-               super.addNotify();\r
-               if (requestRecalcOnAdd) {\r
-                       requestRecalcOnAdd = false;\r
-                       calcRepaint();\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 要求されたプロパティから、フォント高さによる表示領域を計算し、 その領域の再描画を要求する.(描画する内容がなれば、描画要求しない.)<br>\r
-        * 前回表示領域があれば、消去するために、そのエリアも再描画を要求する.<br>\r
-        * それ以外のエリアは描画要求しない.(描画の最適化による負荷軽減策)<br>\r
-        * フォントサイズを求めるためにグラフィクスへのアクセスが必要となるが、 まだ取得できない場合は{@link #addNotify()}の呼び出し時に\r
-        * 再計算するようにフラグを立てておく.<br>\r
-        */\r
-       protected void calcRepaint() {\r
-               Graphics2D g = (Graphics2D) getGraphics();\r
-               if (g == null) {\r
-                       requestRecalcOnAdd = true;\r
-                       return;\r
-               }\r
-               try {\r
-                       // 前回描画領域のクリアのために呼び出す.\r
-                       if (eraseRect != null && !eraseRect.isEmpty()) {\r
-                               repaint(eraseRect);\r
-                       }\r
-\r
-                       // 空であれば新たな描画なし.\r
-                       if (message.length() == 0) {\r
-                               requestRect = null;\r
-                               return;\r
-                       }\r
-\r
-                       FontMetrics fm = g.getFontMetrics();\r
-                       String[] messageLines = message.split("¥n");\r
-\r
-                       Rectangle2D rct = null;\r
-                       for (String messageLine : messageLines) {\r
-                               Rectangle2D tmp = fm.getStringBounds(messageLine, g);\r
-                               if (rct != null) {\r
-                                       rct.add(tmp);\r
-\r
-                               } else {\r
-                                       rct = tmp;\r
-                               }\r
-                       }\r
-\r
-                       int fw = (int) rct.getWidth();\r
-                       int fh = (int) rct.getHeight();\r
-\r
-                       int w = fw + padding.left + padding.right;\r
-                       int h = fh * messageLines.length + padding.top + padding.bottom;\r
-\r
-                       // 指定した位置の右上あたりにする\r
-                       int x = pos.x + 16;\r
-                       int y = pos.y - h;\r
-\r
-                       // サイズ\r
-                       int client_w = getWidth();\r
-                       int client_h = getHeight();\r
-\r
-                       if (x + w > client_w) {\r
-                               // 画面右の場合はカーソルの左に移動\r
-                               x = pos.x - w - 10;\r
-                       }\r
-                       if (y < 0) {\r
-                               // 画面上の場合はカーソルの下に移動\r
-                               y = pos.y + 10;\r
-                       }\r
-                       if (y + h > client_h) {\r
-                               y -= (y + h - client_h);\r
-                       }\r
-\r
-                       // 結果の格納\r
-                       this.requestRect = new Rectangle(x, y, w, h);\r
-                       this.messageLines = messageLines;\r
-                       this.fontHeight = fh;\r
-\r
-                       // 再描画の要求\r
-                       Rectangle paintRect = (Rectangle) requestRect.clone();\r
-                       repaint(paintRect);\r
-\r
-               } finally {\r
-                       g.dispose();\r
-               }\r
-       }\r
-\r
-       public void setPotision(Point requestPt) {\r
-               if (requestPt == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-               if ( !requestPt.equals(pos)) {\r
-                       Point oldpos = pos;\r
-                       pos = (Point) requestPt.clone();\r
-                       calcRepaint();\r
-                       firePropertyChange("position", oldpos, pos);\r
-               }\r
-       }\r
-\r
-       public Point getPosition() {\r
-               return (Point) pos.clone();\r
-       }\r
-\r
-       public void setMessage(String message) {\r
-               if (message == null) {\r
-                       message = "";\r
-               }\r
-               message = message.replace("¥r¥n", "¥n");\r
-               if ( !message.equals(this.message)) {\r
-                       String oldmes = this.message;\r
-                       this.message = message;\r
-                       calcRepaint();\r
-                       firePropertyChange("message", oldmes, message);\r
-               }\r
-       }\r
-\r
-       public String getMessage() {\r
-               return message;\r
-       }\r
-}\r
-\r
-/**\r
- * 画像表示パネル\r
- *\r
- * @author seraphy\r
- */\r
-class PreviewImagePanel extends JPanel {\r
-       private static final long serialVersionUID = 1L;\r
-\r
-       /**\r
-        * 背景モード.<br>\r
-        */\r
-       private BackgroundColorMode bgColorMode;\r
-\r
-       /**\r
-        * 壁紙.<br>\r
-        */\r
-       private Wallpaper wallpaper;\r
-\r
-       /**\r
-        * 壁紙変更イベントのリスナ\r
-        */\r
-       private PropertyChangeListener wallpaperListener;\r
-\r
-       /**\r
-        * 透過オリジナル画像.<br>\r
-        */\r
-       private BufferedImage previewImg;\r
-\r
-       /**\r
-        * 表示用画像(背景モードによる調整あり).<br>\r
-        * 事前に拡大縮小を適用済みの場合は、{@link #scaledZoomFactor}に 適用している倍率が設定される.<br>\r
-        * 表示用に改めてイメージを生成する必要がない場合は、 透過オリジナルと同じインスタンスとなりえる.<br>\r
-        */\r
-       private BufferedImage previewImgForDraw;\r
-\r
-       /**\r
-        * 表示用画像がスケール済みである場合、そのスケールが設定される.<br>\r
-        * スケール済みでない場合はnullとなる.<br>\r
-        */\r
-       private Double scaledZoomFactor;\r
-\r
-\r
-       /**\r
-        * 倍率\r
-        */\r
-       private double zoomFactor = 1.;\r
-\r
-       /**\r
-        * 許容誤差\r
-        */\r
-       private static final double TOLERANT = 0.001;\r
-\r
-\r
-       /**\r
-        * コンストラクタ\r
-        */\r
-       public PreviewImagePanel() {\r
-               super();\r
-\r
-               // 通常モード\r
-               bgColorMode = BackgroundColorMode.ALPHABREND;\r
-\r
-               // 壁紙変更通知リスナ\r
-               wallpaperListener = new PropertyChangeListener() {\r
-                       public void propertyChange(PropertyChangeEvent evt) {\r
-                               onChangeWallpaper();\r
-                       }\r
-               };\r
-\r
-               // 壁紙\r
-               wallpaper = new Wallpaper();\r
-               wallpaper.addPropertyChangeListener(wallpaperListener);\r
-       }\r
-\r
-       /**\r
-        * 画像を表示する.\r
-        */\r
-       @Override\r
-       protected void paintComponent(Graphics g0) {\r
-               Graphics2D g = (Graphics2D) g0;\r
-               super.paintComponent(g);\r
-\r
-               if (previewImgForDraw == null) {\r
-                       return;\r
-               }\r
-\r
-               // 倍率を適用した画像を画面の中央に配置できるように計算する.\r
-               // (画像が倍率適用済みであれば1倍とする)\r
-               Rectangle imgRct = adjustImageRectangle();\r
-\r
-               // 表示用画像がスケール済みでない場合はレンダリングオプションを適用する.\r
-               if (scaledZoomFactor == null) {\r
-                       Object renderingOption = getRenderingOption();\r
-                       g.setRenderingHint(RenderingHints.KEY_INTERPOLATION, renderingOption);\r
-               }\r
-\r
-               // 背景処理\r
-               if (bgColorMode == BackgroundColorMode.ALPHABREND) {\r
-                       // 表示の最大範囲 (可視領域外も含む)\r
-                       int w = getWidth();\r
-                       int h = getHeight();\r
-                       wallpaper.drawWallpaper(g, w, h);\r
-               }\r
-\r
-               // レンダリング\r
-               g.drawImage(previewImgForDraw,\r
-                               imgRct.x, imgRct.y,\r
-                               imgRct.x + imgRct.width, imgRct.y + imgRct.height,\r
-                               0, 0,\r
-                               previewImgForDraw.getWidth(), previewImgForDraw.getHeight(),\r
-                               null);\r
-\r
-               // 通常モード以外のグリッド描画に該当するモードはグリッドを前景に描く\r
-               AppConfig appConfig = AppConfig.getInstance();\r
-               int drawGridMask = appConfig.getDrawGridMask();\r
-               if ((drawGridMask & bgColorMode.mask()) != 0) {\r
-                       Color oldc = g.getColor();\r
-                       try {\r
-                               g.setColor(new Color(appConfig.getPreviewGridColor(), true));\r
-                               drawGrid(g, imgRct.x, imgRct.y, appConfig.getPreviewGridSize());\r
-\r
-                       } finally {\r
-                               g.setColor(oldc);\r
-                       }\r
-               }\r
-       }\r
-\r
-\r
-       /**\r
-        * グリッドを描画する.<br>\r
-        * 開始位置の-1単位位置から画像サイズの+1単位までがグリッド範囲となる。\r
-        *\r
-        * @param g\r
-        * @param offset_x\r
-        *            開始位置\r
-        * @param offset_y\r
-        *            開始位置\r
-        * @param unit\r
-        *            グリッド単位(pixel)\r
-        */\r
-       protected void drawGrid(Graphics2D g, int offset_x, int offset_y, int unit) {\r
-               Rectangle clip = g.getClipBounds();\r
-\r
-               int src_w = previewImg.getWidth();\r
-               int src_h = previewImg.getHeight();\r
-               int my = src_h / unit;\r
-               int mx = src_w / unit;\r
-\r
-               int st_x = offset_x + (int)(-1 * unit * zoomFactor);\r
-               int en_x = offset_x + (int)((mx + 1) * unit * zoomFactor);\r
-               int w = en_x - st_x + 1;\r
-\r
-               for (int y = -1; y <= my + 1; y++) {\r
-                       int y1 = y * unit;\r
-                       Rectangle rct = new Rectangle(\r
-                                       st_x, offset_y + (int)(y1 * zoomFactor),\r
-                                       w, 1);\r
-                       if (clip == null || clip.intersects(rct)) {\r
-                               g.drawLine(rct.x, rct.y, rct.x + rct.width, rct.y);\r
-                       }\r
-               }\r
-\r
-               int st_y = offset_y + (int)(-1 * unit * zoomFactor);\r
-               int en_y = offset_y + (int)((my + 1) * unit * zoomFactor);\r
-               int h = en_y - st_y + 1;\r
-\r
-               for (int x = -1; x <= mx + 1; x++) {\r
-                       int x1 = x * unit;\r
-                       Rectangle rct = new Rectangle(\r
-                                       offset_x + (int)(x1 * zoomFactor), st_y,\r
-                                       1, h);\r
-                       g.drawLine(rct.x, rct.y, rct.x, rct.y + rct.height);\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 現在の倍率に応じたレンダリングオプションを取得する.<br>\r
-        *\r
-        * @return レンダリングオプション\r
-        */\r
-       protected Object getRenderingOption() {\r
-               AppConfig appConfig = AppConfig.getInstance();\r
-               double rendringOptimizeThreshold;\r
-               if (bgColorMode == BackgroundColorMode.ALPHABREND) {\r
-                       rendringOptimizeThreshold = appConfig.getRenderingOptimizeThresholdForNormal();\r
-               } else {\r
-                       rendringOptimizeThreshold = appConfig.getRenderingOptimizeThresholdForCheck();\r
-               }\r
-               Object renderingHint;\r
-               if (zoomFactor < rendringOptimizeThreshold) {\r
-                       // 補正を適用する最大倍率以内である場合\r
-                       if (zoomFactor <= 1. || !appConfig.isEnableInterpolationBicubic()) {\r
-                               // 縮小する場合、もしくはバイキュービックをサポートしない場合\r
-                               renderingHint = RenderingHints.VALUE_INTERPOLATION_BILINEAR;\r
-                       } else {\r
-                               // 拡大する場合でバイキュービックをサポートしている場合\r
-                               renderingHint = RenderingHints.VALUE_INTERPOLATION_BICUBIC;\r
-                       }\r
-\r
-               } else {\r
-                       // 補正を適用する最大倍率を超えている場合は補正なし.\r
-                       renderingHint = RenderingHints.VALUE_INTERPOLATION_NEAREST_NEIGHBOR;\r
-               }\r
-               return renderingHint;\r
-       }\r
-\r
-       /**\r
-        * 倍率と、画面のサイズと、表示するオリジナルの画像サイズをもとに、 倍率を適用した画像サイズを、画面に収まる位置に補正して返す.<br>\r
-        * 返される矩形の幅と高さ(width, height)は拡大後の画像サイズに等しい.<br>\r
-        * 拡大後の画像が画面よりも小さければセンタリングするように矩形の開始位置(x, y)がオフセットされる.<br>\r
-        * そうでなければ矩形の開始位置(x, y)は0となる.<br>\r
-        * 画像が設定されていなければ幅と高さがゼロの矩形が返される.<br>\r
-        *\r
-        * @return 画像を表示するオフセットと大きさ、もしくは空の矩形\r
-        */\r
-       public Rectangle adjustImageRectangle() {\r
-               if (previewImg == null) {\r
-                       return new Rectangle(0, 0, 0, 0); // 幅・高さともにゼロ\r
-               }\r
-               int client_w = getWidth();\r
-               int client_h = getHeight();\r
-\r
-               int src_w = previewImg.getWidth();\r
-               int src_h = previewImg.getHeight();\r
-\r
-               int w = (int) round(src_w * zoomFactor);\r
-               int h = (int) round(src_h * zoomFactor);\r
-\r
-               int offset_x = 0;\r
-               if (w < client_w) {\r
-                       offset_x = (client_w - w) / 2;\r
-               }\r
-               int offset_y = 0;\r
-               if (h < client_h) {\r
-                       offset_y = (client_h - h) / 2;\r
-               }\r
-\r
-               return new Rectangle(offset_x, offset_y, w, h);\r
-       }\r
-\r
-       /**\r
-        * パネルのマウス座標から、実寸の画像のピクセル位置を返す.<br>\r
-        * 画像が表示されていないか、範囲外であればnullを返す.<br>\r
-        *\r
-        * @param pt\r
-        *            パネルの座標\r
-        * @return 画像の位置、もしくはnull\r
-        */\r
-       public Point getImagePosition(Point pt) {\r
-               if (pt == null || previewImg == null) {\r
-                       // プレビュー画像が設定されていなければnull\r
-                       return null;\r
-               }\r
-\r
-               Rectangle imgRct = adjustImageRectangle();\r
-\r
-               if ( !imgRct.contains(pt.x, pt.y)) {\r
-                       // 範囲外であればnull\r
-                       return null;\r
-               }\r
-\r
-               // オフセットを除去する.\r
-               Point ret = (Point) pt.clone();\r
-               ret.x -= imgRct.x;\r
-               ret.y -= imgRct.y;\r
-\r
-               // 倍率を解除する.\r
-               ret.x = (int) floor(ret.x / zoomFactor);\r
-               ret.y = (int) floor(ret.y / zoomFactor);\r
-\r
-               return ret;\r
-       }\r
-\r
-       /**\r
-        * 画像の位置から画面の位置を割り出す.<br>\r
-        *\r
-        * @param pt\r
-        *            画像の位置\r
-        * @return 画面の位置\r
-        */\r
-       public Point getMousePosition(Point pt) {\r
-               if (pt == null || previewImg == null) {\r
-                       // プレビュー画像が設定されていなければnull\r
-                       return null;\r
-               }\r
-\r
-               Rectangle imgRct = adjustImageRectangle();\r
-\r
-               // 表示倍率を加える\r
-               Point ret = (Point) pt.clone();\r
-               ret.x = (int) ceil(ret.x * zoomFactor);\r
-               ret.y = (int) ceil(ret.y * zoomFactor);\r
-\r
-               // オフセットを加える\r
-               ret.x += imgRct.x;\r
-               ret.y += imgRct.y;\r
-\r
-               return ret;\r
-       }\r
-\r
-       /**\r
-        * 指定した位置のRGB値を取得する.<br>\r
-        * 範囲外の場合は0が返される.<br>\r
-        *\r
-        * @param pt\r
-        *            イメージの位置\r
-        * @return イメージのARGB値 (ビット順序は、A:24, R:16, G:8, B:0)\r
-        */\r
-       public int getImageARGB(Point pt) {\r
-               if (pt == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-               try {\r
-                       return previewImg.getRGB(pt.x, pt.y);\r
-\r
-               } catch (RuntimeException ex) {\r
-                       return 0; // 範囲外\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 倍率を適用した画像パネルのサイズを計算し適用する.<br>\r
-        * モードにより余白が加えられる.<br>\r
-        */\r
-       protected void recalcScaledSize() {\r
-               Dimension scaledSize = getScaledSize(true);\r
-               if (scaledSize != null) {\r
-                       setPreferredSize(scaledSize);\r
-                       revalidate();\r
-               }\r
-       }\r
-\r
-       /**\r
-        * 元画像の倍率適用後のサイズを返す.<br>\r
-        * 元画像が設定されていなければnull.<br>\r
-        * needOffsetがfalseであれば表示モードに関わらず、画像の拡大・縮小後の純粋なサイズ、\r
-        * trueであれば余白が必要な表示モードの場合の余白が付与された場合のサイズが返される.<br>\r
-        *\r
-        * @param needOffset\r
-        *            余白を加味したサイズが必要な場合はtrue\r
-        * @return 倍率適用後のサイズ、もしくはnull\r
-        */\r
-       protected Dimension getScaledSize(boolean needOffset) {\r
-               if (previewImg == null) {\r
-                       return null;\r
-               }\r
-               int src_w = previewImg.getWidth();\r
-               int src_h = previewImg.getHeight();\r
-\r
-               int w = (int) round(src_w * zoomFactor);\r
-               int h = (int) round(src_h * zoomFactor);\r
-\r
-               Dimension scaledSize = new Dimension(w, h);\r
-\r
-               if (bgColorMode != BackgroundColorMode.ALPHABREND) {\r
-                       // 通常モード以外は画像よりも少し大きめにすることで\r
-                       // キャンバスに余白をつける\r
-                       AppConfig appConfig = AppConfig.getInstance();\r
-                       int unfilledSpace = appConfig.getPreviewUnfilledSpaceForCheckMode();\r
-                       scaledSize.width += max(0, unfilledSpace * 2);\r
-                       scaledSize.height += max(0, unfilledSpace * 2);\r
-               }\r
-\r
-               return scaledSize;\r
-       }\r
-\r
-       /**\r
-        * プレビュー画像を設定する.\r
-        *\r
-        * @param previewImg\r
-        */\r
-       public void setPreviewImage(BufferedImage previewImg) {\r
-               BufferedImage oldimg = this.previewImg;\r
-               this.previewImg = previewImg;\r
-\r
-               recalcScaledSize();\r
-               makeDrawImage(true);\r
-               repaint();\r
-\r
-               firePropertyChange("previewImage", oldimg, previewImg);\r
-       }\r
-\r
-       public BufferedImage getPreviewImage() {\r
-               return previewImg;\r
-       }\r
-\r
-       /**\r
-        * 壁紙を設定する.\r
-        *\r
-        * @param wallpaper\r
-        */\r
-       public void setWallpaper(Wallpaper wallpaper) {\r
-               if (wallpaper == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-               if ( !this.wallpaper.equals(wallpaper)) {\r
-                       Wallpaper wallpaperOld = this.wallpaper;\r
-                       if (wallpaperOld != null) {\r
-                               wallpaperOld.removePropertyChangeListener(wallpaperListener);\r
-                       }\r
-                       this.wallpaper = wallpaper;\r
-                       if (this.wallpaper != null) {\r
-                               this.wallpaper.addPropertyChangeListener(wallpaperListener);\r
-                       }\r
-                       firePropertyChange("wallpaper", wallpaperOld, this.wallpaper);\r
-                       onChangeWallpaper();\r
-               }\r
-       }\r
-\r
-       public Wallpaper getWallpaper() {\r
-               return wallpaper;\r
-       }\r
-\r
-       protected void onChangeWallpaper() {\r
-               repaint();\r
-       }\r
-\r
-       /**\r
-        * 背景モード調整済みの表示用画像を作成する.\r
-        *\r
-        * @param changeImage\r
-        *            画像の変更あり\r
-        */\r
-       protected void makeDrawImage(boolean changeImage) {\r
-               if (previewImg == null) {\r
-                       // 画像が設定されていなければ空\r
-                       this.previewImgForDraw = null;\r
-                       scaledZoomFactor = null;\r
-                       return;\r
-               }\r
-\r
-               BufferedImage img;\r
-               if (changeImage || scaledZoomFactor != null) {\r
-                       // 画像が変更されているか、スケール済みであれば\r
-                       // 背景モードの再適用が必要.\r
-                       if (bgColorMode == BackgroundColorMode.ALPHABREND) {\r
-                               // アルファブレンド通常モードは背景用にあえて作成する必要はない.\r
-                               img = previewImg;\r
-\r
-                       } else {\r
-                               // アルファブレンド通常モード以外は背景に作成する\r
-                               Color bgColor = wallpaper.getBackgroundColor();\r
-                               BackgroundColorFilter bgColorFilter = new BackgroundColorFilter(bgColorMode, bgColor);\r
-                               img = bgColorFilter.filter(previewImg, null);\r
-                       }\r
-\r
-               } else {\r
-                       // 画像が変更されておらず、スケール済みでもなければ\r
-                       // すでに作成済みの画像が使用できる.\r
-                       img = previewImgForDraw;\r
-               }\r
-\r
-               // レンダリングオプション\r
-               Object renderingOption = getRenderingOption();\r
-\r
-               // バイキュービックでなければ、事前の拡大縮小は行わずに、表示時に行う.\r
-               if ( !renderingOption.equals(RenderingHints.VALUE_INTERPOLATION_BICUBIC)) {\r
-                       previewImgForDraw = img;\r
-                       scaledZoomFactor = null;\r
-                       return;\r
-               }\r
-\r
-               // バイキュービックの場合、倍率を適用したサイズに予め加工しておく.\r
-               Dimension scaledSize = getScaledSize(false);\r
-               BufferedImage offscreen = new BufferedImage(\r
-                               scaledSize.width, scaledSize.height, BufferedImage.TYPE_INT_ARGB);\r
-               Graphics2D g = offscreen.createGraphics();\r
-               try {\r
-                       g.setRenderingHint(RenderingHints.KEY_INTERPOLATION,\r
-                                       RenderingHints.VALUE_INTERPOLATION_BICUBIC);\r
-\r
-                       g.drawImage(img,\r
-                                       0, 0, scaledSize.width, scaledSize.height,\r
-                                       0, 0, img.getWidth(), img.getHeight(),\r
-                                       null);\r
-\r
-               } finally {\r
-                       g.dispose();\r
-               }\r
-               previewImgForDraw = offscreen;\r
-               scaledZoomFactor = Double.valueOf(zoomFactor);\r
-       }\r
-\r
-       public void setBackgroundColorMode(BackgroundColorMode bgColorMode) {\r
-               if (bgColorMode == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-               if (this.bgColorMode != bgColorMode) {\r
-                       BackgroundColorMode oldcm = bgColorMode;\r
-                       this.bgColorMode = bgColorMode;\r
-\r
-                       makeDrawImage(true);\r
-                       recalcScaledSize();\r
-                       repaint();\r
-\r
-                       firePropertyChange("backgroundColorMode", oldcm, bgColorMode);\r
-               }\r
-       }\r
-\r
-       public BackgroundColorMode setBackgroundColorMode() {\r
-               return bgColorMode;\r
-       }\r
-\r
-       public void setZoomFactor(double zoomFactor) {\r
-               if (abs(zoomFactor - this.zoomFactor) > TOLERANT) {\r
-                       // 0.001未満の差異は誤差とみなして反映しない.\r
-                       double oldzoom = this.zoomFactor;\r
-                       this.zoomFactor = zoomFactor;\r
-\r
-                       recalcScaledSize();\r
-                       makeDrawImage(false);\r
-                       repaint();\r
-\r
-                       firePropertyChange("zoomFactor", oldzoom, zoomFactor);\r
-               }\r
-       }\r
-\r
-       public double getZoomFactor() {\r
-               return zoomFactor;\r
-       }\r
-\r
-       /**\r
-        * 倍率が100%であるか?\r
-        *\r
-        * @return 100%であればtrue\r
-        */\r
-       public boolean isDefaultZoom() {\r
-               return zoomFactor - 1 < TOLERANT;\r
-       }\r
-}\r
-\r
-/**\r
- * 倍率・背景モードを操作するための下部パネル用\r
- *\r
- * @author seraphy\r
- */\r
-class PreviewControlPanel extends JPanel {\r
-       private static final long serialVersionUID = 1L;\r
-\r
-       private static final Logger logger = Logger.getLogger(PreviewControlPanel.class.getName());\r
-\r
-       protected static final String STRINGS_RESOURCE = "languages/previewpanel";\r
-\r
-       /**\r
-        * ピン留めチェックボックス\r
-        */\r
-       private JCheckBox chkPinning;\r
-\r
-       /**\r
-        * アルファ確認チェックボックス\r
-        */\r
-       private JCheckBox chkNoAlpha;\r
-\r
-       /**\r
-        * グレースケール確認チェックボックス\r
-        */\r
-       private JCheckBox chkGrayscale;\r
-\r
-       /**\r
-        * 倍率用スライダ\r
-        */\r
-       private JSlider zoomSlider;\r
-\r
-       /**\r
-        * 倍率入力用コンボボックス\r
-        */\r
-       private JComboBox zoomCombo;\r
-\r
-\r
-       /**\r
-        * スライダの最小値\r
-        */\r
-       private static final int MIN_INDEX = -170;\r
-\r
-       /**\r
-        * スライダの最大値\r
-        */\r
-       private static final int MAX_INDEX = 219;\r
-\r
-       /**\r
-        * 最小倍率\r
-        */\r
-       private double minimumZoomFactor;\r
-\r
-       /**\r
-        * 最大倍率\r
-        */\r
-       private double maximumZoomFactor;\r
-\r
-\r
-       /**\r
-        * 現在の倍率(100倍済み)\r
-        */\r
-       private int currentZoomFactorInt;\r
-\r
-       /**\r
-        * 現在の背景色モード\r
-        */\r
-       private BackgroundColorMode backgroundColorMode;\r
-\r
-\r
-       /**\r
-        * 任意の底Aをもつ対数 logA(N)を計算して返す.\r
-        *\r
-        * @param a\r
-        *            底\r
-        * @param x\r
-        *            引数\r
-        * @return logA(N)\r
-        */\r
-       private static double logN(double a, double x) {\r
-               return log(x) / log(a);\r
-       }\r
-\r
-       /**\r
-        * 倍率(等倍を1とする)に対するスライダのインデックス値を返す.<br>\r
-        * スライダは10ステップごとに前のステップの10%づつ増減する.(複利式)<br>\r
-        *\r
-        * @param zoomFactor\r
-        *            倍率(1を等倍とする)\r
-        * @return インデックス\r
-        */\r
-       private static int zoomFactorToIndex(double zoomFactor) {\r
-               return (int) round(logN((1. + 0.1), zoomFactor) * 10);\r
-       }\r
-\r
-       /**\r
-        * スライダのインデックス値から倍率(等倍を1とする)を返す.<br>\r
-        * 10ステップごとに10%づつ増減する.(複利式)<br>\r
-        *\r
-        * @param index\r
-        *            インデックス\r
-        * @return 倍率(1を等倍とする)\r
-        */\r
-       private static double zoomFactorFromIndex(int index) {\r
-               return pow(1. + 0.1, index / 10.);\r
-       }\r
-\r
-\r
-       /**\r
-        * コンストラクタ\r
-        */\r
-       public PreviewControlPanel() {\r
-               final Properties strings = LocalizedResourcePropertyLoader.getCachedInstance()\r
-                               .getLocalizedProperties(STRINGS_RESOURCE);\r
-\r
-               UIHelper uiHelper = UIHelper.getInstance();\r
-\r
-               // ピンアイコン\r
-               Icon pinIcon = uiHelper.createTwoStateIcon(\r
-                               "icons/pin-icon1.png", "icons/pin-icon2.png");\r
-\r
-               // ピンチェックボックス\r
-               chkPinning = new JCheckBox(pinIcon);\r
-               chkPinning.setToolTipText(strings.getProperty("tooltip.zoompanel.pinning"));\r
-\r
-               // 円ボタン型チェックボックス用アイコンの実装\r
-               final Icon stateIcon = new Icon() {\r
-                       public int getIconHeight() {\r
-                               return 12;\r
-                       }\r
-                       public int getIconWidth() {\r
-                               return 6;\r
-                       };\r
-                       public void paintIcon(Component c, Graphics g, int x, int y) {\r
-                               boolean sw = false;\r
-                               if (c instanceof AbstractButton) {\r
-                                       AbstractButton btn = (AbstractButton) c;\r
-                                       sw = btn.isSelected();\r
-                               }\r
-\r
-                               int w = getIconWidth();\r
-                               int h = getIconHeight();\r
-\r
-                               int s = min(w, h);\r
-\r
-                               int ox = 0;\r
-                               int oy = 0;\r
-                               if (w > s) {\r
-                                       ox = (w - s) / 2;\r
-                               }\r
-                               if (h > s) {\r
-                                       oy = (h - s) / 2;\r
-                               }\r
-\r
-                               if (sw) {\r
-                                       AppConfig appConfig = AppConfig.getInstance();\r
-                                       Color fillColor = appConfig.getSelectedItemBgColor();\r
-                                       g.setColor(fillColor);\r
-                                       g.fillOval(x + ox, y + oy, s, w);\r
-                               }\r
-                               g.setColor(Color.GRAY);\r
-                               g.drawOval(x + ox, y + oy, s, s);\r
-                       }\r
-               };\r
-\r
-               // アルファ確認とグレースケール確認用のチェックボックス\r
-               chkNoAlpha = new JCheckBox(stateIcon);\r
-               chkGrayscale = new JCheckBox(stateIcon);\r
-\r
-               chkNoAlpha.setToolTipText(strings.getProperty("tooltip.zoompanel.checkalpha"));\r
-               chkGrayscale.setToolTipText(strings.getProperty("tooltip.zoompanel.checkgrayscale"));\r
-\r
-               backgroundColorMode = BackgroundColorMode.ALPHABREND;\r
-\r
-               final ChangeListener chkAlphaGrayChangeListener = new ChangeListener() {\r
-                       public void stateChanged(ChangeEvent e) {\r
-                               onChangeCheckAlphaGray();\r
-                       }\r
-               };\r
-               chkNoAlpha.addChangeListener(chkAlphaGrayChangeListener);\r
-               chkGrayscale.addChangeListener(chkAlphaGrayChangeListener);\r
-\r
-               // 倍率スライダ\r
-               zoomSlider = new JSlider(JSlider.HORIZONTAL, MIN_INDEX, MAX_INDEX, 0);\r
-               zoomSlider.setToolTipText(strings.getProperty("tooltip.zoompanel.zoomfactor_slider"));\r
-\r
-               // 倍率コンボ\r
-               zoomCombo = new JComboBox();\r
-               zoomCombo.setToolTipText(strings.getProperty("tooltip.zoompanel.zoomfactor_combo"));\r
-\r
-               // 倍率の既定リストの設定と、最大・最小値の算定\r
-               minimumZoomFactor = zoomFactorFromIndex(zoomSlider.getMinimum());\r
-               maximumZoomFactor = zoomFactorFromIndex(zoomSlider.getMaximum());\r
-\r
-               int minZoomRange = (int) round(minimumZoomFactor * 100.);\r
-               int maxZoomRange = (int) round(maximumZoomFactor * 100.);\r
-\r
-               List<Integer> predefinedZoomRanges = getPredefinedZoomRanges();\r
-               for (int zoomRange : predefinedZoomRanges) {\r
-                       if (zoomRange < minZoomRange) {\r
-                               minZoomRange = zoomRange;\r
-                       }\r
-                       if (zoomRange > maxZoomRange) {\r
-                               maxZoomRange = zoomRange;\r
-                       }\r
-                       zoomCombo.addItem(Integer.toString(zoomRange));\r
-               }\r
-               final int[] zoomRanges = {minZoomRange, maxZoomRange};\r
-\r
-               currentZoomFactorInt = 100;\r
-               zoomCombo.setSelectedItem(Integer.toString(currentZoomFactorInt));\r
-               zoomCombo.setEditable(true);\r
-               if ( !Main.isMacOSX()) {\r
-                       // Windows環境だとデフォルトで9桁分のテキストフィールドが作成され\r
-                       // それがレイアウトの推奨サイズとして実際に使われてしまうため、\r
-                       // 明示的に3桁にとどめておくようにオーバーライドする.\r
-                       // Mac OS Xならば問題ない.\r
-                       zoomCombo.setEditor(new BasicComboBoxEditor() {\r
-                               {\r
-                                       editor = new JTextField(3) {\r
-                                               private static final long serialVersionUID = 1L;\r
-                                               @Override\r
-                                               public void setBorder(Border border) {\r
-                                                       // 何もしない.\r
-                                               }\r
-                                               public void setText(String s) {\r
-                                                       if (getText().equals(s)) {\r
-                                                               return;\r
-                                                       }\r
-                                                       super.setText(s);\r
-                                               }\r
-                                       };\r
-                               }\r
-                       });\r
-               }\r
-\r
-               // スライダを変更することによりコンボボックスを変更する、\r
-               // もしくはコンボボックスを変更することでスライダを変更するが、\r
-               // 互いに通知を呼び合うことになるため他方を無視するためのセマフォ\r
-               final Semaphore changeLock = new Semaphore(1);\r
-\r
-               zoomCombo.addActionListener(new ActionListener() {\r
-                       public void actionPerformed(ActionEvent e) {\r
-                               boolean adjusted = false;\r
-                               String value = (String) zoomCombo.getSelectedItem();\r
-                               int zoomFactorInt;\r
-                               try {\r
-                                       zoomFactorInt = Integer.parseInt(value);\r
-                                       if (zoomFactorInt < zoomRanges[0]) {\r
-                                               zoomFactorInt = zoomRanges[0];\r
-                                               adjusted = true;\r
-\r
-                                       } else if (zoomFactorInt > zoomRanges[1]) {\r
-                                               zoomFactorInt = zoomRanges[1];\r
-                                               adjusted = true;\r
-                                       }\r
-\r
-                               } catch (RuntimeException ex) {\r
-                                       zoomFactorInt = 100;\r
-                                       adjusted = true;\r
-                               }\r
-                               if (adjusted) {\r
-                                       zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));\r
-                                       Toolkit tk = Toolkit.getDefaultToolkit();\r
-                                       tk.beep();\r
-                               }\r
-                               if (changeLock.tryAcquire()) {\r
-                                       try {\r
-                                               zoomSlider.setValue(zoomFactorToIndex(zoomFactorInt / 100.));\r
-\r
-                                       } finally {\r
-                                               changeLock.release();\r
-                                       }\r
-                               }\r
-                               fireZoomFactorChange(zoomFactorInt);\r
-                       }\r
-               });\r
-\r
-               zoomSlider.addChangeListener(new ChangeListener() {\r
-                       public void stateChanged(ChangeEvent e) {\r
-                               int index = zoomSlider.getValue();\r
-                               double zoomFactor = zoomFactorFromIndex(index);\r
-                               int zoomFactorInt = (int) round(zoomFactor * 100);\r
-\r
-                               if (changeLock.tryAcquire()) {\r
-                                       try {\r
-                                               zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));\r
-\r
-                                       } finally {\r
-                                               changeLock.release();\r
-                                       }\r
-                                       fireZoomFactorChange(zoomFactorInt);\r
-                               }\r
-                       }\r
-               });\r
-\r
-               // パーツの配備\r
-\r
-               GridBagLayout gbl = new GridBagLayout();\r
-               setLayout(gbl);\r
-\r
-               GridBagConstraints gbc = new GridBagConstraints();\r
-               gbc.gridx = 0;\r
-               gbc.gridy = 0;\r
-               gbc.ipadx = 0;\r
-               gbc.ipady = 0;\r
-               gbc.gridheight = 1;\r
-               gbc.gridwidth = 1;\r
-               gbc.fill = GridBagConstraints.NONE;\r
-               gbc.anchor = GridBagConstraints.CENTER;\r
-               gbc.insets = new Insets(0, 0, 0, 5);\r
-               gbc.weightx = 0.;\r
-               gbc.weighty = 0.;\r
-\r
-               add(chkPinning, gbc);\r
-\r
-               gbc.gridx = 1;\r
-               gbc.weightx = 0.;\r
-               gbc.insets = new Insets(0, 0, 0, 0);\r
-               add(chkGrayscale, gbc);\r
-\r
-               gbc.gridx = 2;\r
-               gbc.weightx = 0.;\r
-               gbc.insets = new Insets(0, 0, 0, 5);\r
-               add(chkNoAlpha, gbc);\r
-\r
-               gbc.gridx = 3;\r
-               gbc.weightx = 1.;\r
-               gbc.fill = GridBagConstraints.HORIZONTAL;\r
-               add(zoomSlider, gbc);\r
-\r
-               gbc.gridx = 4;\r
-               gbc.weightx = 0.;\r
-               gbc.insets = new Insets(3, 0, 3, 0);\r
-               gbc.fill = GridBagConstraints.VERTICAL;\r
-               add(zoomCombo, gbc);\r
-\r
-               Integer scrollbarWidth = (Integer) UIManager.get("ScrollBar.width");\r
-               logger.log(Level.CONFIG, "ScrollBar.width=" + scrollbarWidth);\r
-               if (scrollbarWidth == null) {\r
-                       scrollbarWidth = Integer.parseInt(\r
-                                       strings.getProperty("uiconstraint.scrollbar.width"));\r
-               }\r
-\r
-               gbc.gridx = 5;\r
-               gbc.weightx = 0.;\r
-               gbc.anchor = GridBagConstraints.WEST;\r
-               gbc.insets = new Insets(0, 0, 0, scrollbarWidth);\r
-               add(new JLabel("%"), gbc);\r
-       }\r
-\r
-       /**\r
-        * アプリケーション設定より事前定義済みの倍率候補を取得する\r
-        *\r
-        * @return 倍率候補のリスト(順序あり)\r
-        */\r
-       protected List<Integer> getPredefinedZoomRanges() {\r
-               AppConfig appConfig = AppConfig.getInstance();\r
-               String strs = appConfig.getPredefinedZoomRanges();\r
-               TreeSet<Integer> ranges = new TreeSet<Integer>();\r
-               for (String str : strs.split(",")) {\r
-                       str = str.trim();\r
-                       if (str.length() > 0) {\r
-                               try {\r
-                                       int zoomFactor = Integer.parseInt(str);\r
-                                       ranges.add(Integer.valueOf(zoomFactor));\r
-\r
-                               } catch (RuntimeException ex) {\r
-                                       // 無視する.\r
-                               }\r
-                       }\r
-               }\r
-               ranges.add(Integer.valueOf(100)); // 等倍は常に設定する.\r
-               return new ArrayList<Integer>(ranges);\r
-       }\r
-\r
-       /**\r
-        * 倍率が変更されたことを通知する.\r
-        */\r
-       protected void fireZoomFactorChange(int newZoomFactor) {\r
-               if (currentZoomFactorInt != newZoomFactor) {\r
-                       int oldValue = currentZoomFactorInt;\r
-                       currentZoomFactorInt = newZoomFactor;\r
-                       firePropertyChange("zoomFactorInt", oldValue, newZoomFactor);\r
-               }\r
-       }\r
-\r
-       private Semaphore changeChkLock = new Semaphore(1);\r
-\r
-       protected void onChangeCheckAlphaGray() {\r
-               changeChkLock.tryAcquire();\r
-               try {\r
-                       BackgroundColorMode backgroundColorMode = BackgroundColorMode.valueOf(\r
-                                       chkNoAlpha.isSelected(),\r
-                                       chkGrayscale.isSelected()\r
-                                       );\r
-                       setBackgroundColorMode(backgroundColorMode);\r
-\r
-               } finally {\r
-                       changeChkLock.release();\r
-               }\r
-       }\r
-\r
-       public BackgroundColorMode getBackgroundColorMode() {\r
-               return this.backgroundColorMode;\r
-       }\r
-\r
-       public void setBackgroundColorMode(BackgroundColorMode backgroundColorMode) {\r
-               if (backgroundColorMode == null) {\r
-                       throw new IllegalArgumentException();\r
-               }\r
-\r
-               BackgroundColorMode oldcm = this.backgroundColorMode;\r
-               if (oldcm != backgroundColorMode) {\r
-                       this.backgroundColorMode = backgroundColorMode;\r
-                       changeChkLock.tryAcquire();\r
-                       try {\r
-                               chkNoAlpha.setSelected(backgroundColorMode.isNoAlphaChannel());\r
-                               chkGrayscale.setSelected(backgroundColorMode.isGrayscale());\r
-\r
-                       } finally {\r
-                               changeChkLock.release();\r
-                       }\r
-                       firePropertyChange("backgroundColorMode", oldcm, backgroundColorMode);\r
-               }\r
-       }\r
-\r
-       public boolean isPinned() {\r
-               return chkPinning.isSelected();\r
-       }\r
-\r
-       public void setPinned(boolean pinned) {\r
-               chkPinning.setSelected(pinned);\r
-               if (isDisplayable()) {\r
-                       setVisible(pinned);\r
-               }\r
-       }\r
-\r
-       public double getZoomFactor() {\r
-               return (double) currentZoomFactorInt / 100.;\r
-       }\r
-\r
-       public void setZoomFactor(double zoomFactor) {\r
-               if (zoomFactor < minimumZoomFactor) {\r
-                       zoomFactor = minimumZoomFactor;\r
-               }\r
-               if (zoomFactor > maximumZoomFactor) {\r
-                       zoomFactor = maximumZoomFactor;\r
-               }\r
-               int zoomFactorInt = (int) round(zoomFactor * 100.);\r
-               if (zoomFactorInt != currentZoomFactorInt) {\r
-                       zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));\r
-               }\r
-       }\r
-}\r
+package charactermanaj.ui;
+
+import static java.lang.Math.*;
+
+import java.awt.BorderLayout;
+import java.awt.Color;
+import java.awt.Component;
+import java.awt.Cursor;
+import java.awt.Dimension;
+import java.awt.FontMetrics;
+import java.awt.Graphics;
+import java.awt.Graphics2D;
+import java.awt.GridBagConstraints;
+import java.awt.GridBagLayout;
+import java.awt.Insets;
+import java.awt.Point;
+import java.awt.Rectangle;
+import java.awt.RenderingHints;
+import java.awt.Toolkit;
+import java.awt.event.ActionEvent;
+import java.awt.event.ActionListener;
+import java.awt.event.MouseAdapter;
+import java.awt.event.MouseEvent;
+import java.awt.event.MouseMotionAdapter;
+import java.awt.event.MouseMotionListener;
+import java.awt.event.MouseWheelEvent;
+import java.awt.event.MouseWheelListener;
+import java.awt.geom.Rectangle2D;
+import java.awt.image.BufferedImage;
+import java.beans.PropertyChangeEvent;
+import java.beans.PropertyChangeListener;
+import java.util.ArrayList;
+import java.util.EventObject;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Map;
+import java.util.Properties;
+import java.util.TreeSet;
+import java.util.concurrent.Semaphore;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import javax.swing.AbstractButton;
+import javax.swing.BorderFactory;
+import javax.swing.Box;
+import javax.swing.Icon;
+import javax.swing.JButton;
+import javax.swing.JCheckBox;
+import javax.swing.JComboBox;
+import javax.swing.JLabel;
+import javax.swing.JLayeredPane;
+import javax.swing.JPanel;
+import javax.swing.JScrollBar;
+import javax.swing.JScrollPane;
+import javax.swing.JSlider;
+import javax.swing.JTextField;
+import javax.swing.JToolBar;
+import javax.swing.JViewport;
+import javax.swing.OverlayLayout;
+import javax.swing.SwingUtilities;
+import javax.swing.Timer;
+import javax.swing.UIManager;
+import javax.swing.border.Border;
+import javax.swing.event.ChangeEvent;
+import javax.swing.event.ChangeListener;
+import javax.swing.plaf.basic.BasicComboBoxEditor;
+
+import charactermanaj.Main;
+import charactermanaj.graphics.filters.BackgroundColorFilter;
+import charactermanaj.graphics.filters.BackgroundColorFilter.BackgroundColorMode;
+import charactermanaj.model.AppConfig;
+import charactermanaj.ui.util.ScaleSupport;
+import charactermanaj.ui.util.ScrollPaneDragScrollSupport;
+import charactermanaj.util.LocalizedResourcePropertyLoader;
+import charactermanaj.util.UIHelper;
+
+/**
+ * プレビューパネル
+ *
+ * @author seraphy
+ */
+public class PreviewPanel extends JPanel {
+
+       private static final long serialVersionUID = 1L;
+
+       protected static final String STRINGS_RESOURCE = "languages/previewpanel";
+
+
+       /**
+        * プレビューパネルの上部ツールバーの通知を受けるリスナ
+        *
+        * @author seraphy
+        */
+       public interface PreviewPanelListener {
+
+               /**
+                * 保存
+                *
+                * @param e
+                */
+               void savePicture(PreviewPanelEvent e);
+
+               /**
+                * コピー
+                *
+                * @param e
+                */
+               void copyPicture(PreviewPanelEvent e);
+
+               /**
+                * 背景色変更
+                *
+                * @param e
+                */
+               void changeBackgroundColor(PreviewPanelEvent e);
+
+               /**
+                * 情報
+                *
+                * @param e
+                */
+               void showInformation(PreviewPanelEvent e);
+
+               /**
+                * お気に入りに追加
+                *
+                * @param e
+                */
+               void addFavorite(PreviewPanelEvent e);
+
+               /**
+                * 左右反転
+                *
+                * @param e
+                */
+               void flipHorizontal(PreviewPanelEvent e);
+       }
+
+       /**
+        * ロード中を示すインジケータ
+        */
+       private final String indicatorText;
+
+       /**
+        * ロード中であるか判定するタイマー
+        */
+       private final Timer timer;
+
+       /**
+        * インジケータを表示するまでのディレイ
+        */
+       private long indicatorDelay;
+
+       @Override
+       public void addNotify() {
+               super.addNotify();
+               if (!timer.isRunning()) {
+                       timer.start();
+               }
+       }
+
+       @Override
+       public void removeNotify() {
+               if (timer.isRunning()) {
+                       timer.stop();
+               }
+               super.removeNotify();
+       }
+
+       public static class PreviewPanelEvent extends EventObject {
+
+               private static final long serialVersionUID = 1L;
+
+               private int modifiers;
+
+               public PreviewPanelEvent(Object src, ActionEvent e) {
+                       this(src, (e == null) ? 0 : e.getModifiers());
+               }
+
+               public PreviewPanelEvent(Object src, int modifiers) {
+                       super(src);
+                       this.modifiers = modifiers;
+               }
+
+               public int getModifiers() {
+                       return modifiers;
+               }
+
+               public boolean isShiftKeyPressed() {
+                       return (modifiers & ActionEvent.SHIFT_MASK) != 0;
+               }
+       }
+
+       private final Object lock = new Object();
+
+       private long loadingTicket;
+
+       private long loadedTicket;
+
+       private long firstWaitingTimestamp;
+
+       private boolean indicatorShown;
+
+       private String title;
+
+       private JLabel lblTitle;
+
+       private JLayeredPane layeredPane;
+
+       private CheckInfoLayerPanel checkInfoLayerPanel;
+
+       private PreviewImagePanel previewImgPanel;
+
+       private JScrollPane previewImgScrollPane;
+
+       private ScrollPaneDragScrollSupport scrollSupport;
+
+       private PreviewControlPanel previewControlPanel;
+
+       private double latestToggleZoom = 2.;
+
+       private LinkedList<PreviewPanelListener> listeners = new LinkedList<PreviewPanelListener>();
+
+
+       public PreviewPanel(ScaleSupport scaleSupport) {
+               setLayout(new BorderLayout());
+
+               final AppConfig appConfig = AppConfig.getInstance();
+               final Properties strings = LocalizedResourcePropertyLoader.getCachedInstance()
+                       .getLocalizedProperties(STRINGS_RESOURCE);
+
+               // 画像をロード中であることを示すインジケータの確認サイクル.
+               timer = new Timer(100, new ActionListener() {
+                               public void actionPerformed(ActionEvent e) {
+                                       onTimer();
+                               }
+                       });
+
+               indicatorText = strings.getProperty("indicatorText");
+               indicatorDelay = appConfig.getPreviewIndicatorDelay();
+
+               UIHelper uiUtl = UIHelper.getInstance();
+               JButton saveBtn = uiUtl.createIconButton("icons/save.png");
+               JButton copyBtn = uiUtl.createIconButton("icons/copy.png");
+               JButton colorBtn = uiUtl.createIconButton("icons/color.png");
+               JButton informationBtn = uiUtl.createIconButton("icons/information.png");
+               JButton favoriteBtn = uiUtl.createIconButton("icons/favorite.png");
+               JButton flipHolizontalBtn = uiUtl.createIconButton("icons/flip.png");
+
+               saveBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               savePicture(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+               copyBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               copyPicture(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+               colorBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               changeBackgroundColor(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+               informationBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               showInformation(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+               favoriteBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               addFavorite(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+               flipHolizontalBtn.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               flipHolizontal(new PreviewPanelEvent(PreviewPanel.this, e));
+                       }
+               });
+
+               saveBtn.setToolTipText(strings.getProperty("tooltip.save"));
+               copyBtn.setToolTipText(strings.getProperty("tooltip.copy"));
+               colorBtn.setToolTipText(strings.getProperty("tooltip.changeBgColor"));
+               informationBtn.setToolTipText(strings.getProperty("tooltip.showInformation"));
+               favoriteBtn.setToolTipText(strings.getProperty("tooltip.registerFavorites"));
+               flipHolizontalBtn.setToolTipText(strings.getProperty("tooltip.flipHorizontal"));
+
+               final JToolBar toolBar = new JToolBar();
+               toolBar.setFloatable(false);
+               toolBar.add(flipHolizontalBtn);
+               toolBar.add(copyBtn);
+               toolBar.add(saveBtn);
+               toolBar.add(Box.createHorizontalStrut(8));
+               toolBar.add(colorBtn);
+               toolBar.add(Box.createHorizontalStrut(4));
+               toolBar.add(favoriteBtn);
+               toolBar.add(informationBtn);
+
+               lblTitle = new JLabel() {
+                       private static final long serialVersionUID = 1L;
+
+                       public Dimension getPreferredSize() {
+                               Dimension dim = super.getPreferredSize();
+                               int maxWidth = getParent().getWidth() - toolBar.getWidth();
+                               if (dim.width > maxWidth) {
+                                       dim.width = maxWidth;
+                               }
+                               return dim;
+                       };
+
+                       public Dimension getMaximumSize() {
+                               return getPreferredSize();
+                       };
+
+                       public Dimension getMinimumSize() {
+                               Dimension dim = getPreferredSize();
+                               dim.width = 50;
+                               return dim;
+                       };
+               };
+
+               lblTitle.setBorder(BorderFactory.createEmptyBorder(3, 10, 3, 3));
+
+               JPanel previewPaneHeader = new JPanel();
+               previewPaneHeader.setLayout(new BorderLayout());
+               previewPaneHeader.add(lblTitle, BorderLayout.WEST);
+               previewPaneHeader.add(toolBar, BorderLayout.EAST);
+
+               previewImgPanel = new PreviewImagePanel();
+
+
+               previewImgScrollPane = new JScrollPane(previewImgPanel);
+               previewImgScrollPane.setAutoscrolls(false);
+               previewImgScrollPane.setWheelScrollingEnabled(false);
+               previewImgScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_ALWAYS);
+               previewImgScrollPane.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS);
+
+               scrollSupport = new ScrollPaneDragScrollSupport(previewImgScrollPane) {
+                       @Override
+                       protected void setCursor(Cursor cursor) {
+                               PreviewPanel.this.setCursor(cursor);
+                       }
+               };
+
+               add(previewPaneHeader, BorderLayout.NORTH);
+
+               layeredPane = new JLayeredPane();
+               layeredPane.setLayout(new OverlayLayout(layeredPane));
+
+               layeredPane.add(previewImgScrollPane, JLayeredPane.DEFAULT_LAYER);
+
+               checkInfoLayerPanel = new CheckInfoLayerPanel();
+               layeredPane.add(checkInfoLayerPanel, JLayeredPane.POPUP_LAYER);
+               checkInfoLayerPanel.setVisible(false);
+
+               add(layeredPane, BorderLayout.CENTER);
+
+               previewControlPanel = new PreviewControlPanel();
+               Dimension dim = previewControlPanel.getPreferredSize();
+               Dimension prevDim = previewImgScrollPane.getPreferredSize();
+               dim.width = prevDim.width;
+               previewControlPanel.setPreferredSize(dim);
+
+               add(previewControlPanel, BorderLayout.SOUTH);
+               previewControlPanel.setPinned(appConfig.isEnableZoomPanel());
+
+               // 倍率が変更された場合
+               previewControlPanel.addPropertyChangeListener("zoomFactorInt", new PropertyChangeListener() {
+                       public void propertyChange(PropertyChangeEvent evt) {
+                               Integer newValue = (Integer) evt.getNewValue();
+                               zoomWithCenterPosition(newValue.doubleValue() / 100., null);
+                       }
+               });
+               // 背景モードが切り替えられた場合
+               previewControlPanel.addPropertyChangeListener("backgroundColorMode", new PropertyChangeListener() {
+                       public void propertyChange(PropertyChangeEvent evt) {
+                               BackgroundColorMode bgColorMode = (BackgroundColorMode) evt.getNewValue();
+                               previewImgPanel.setBackgroundColorMode(bgColorMode);
+                               if (bgColorMode != BackgroundColorMode.ALPHABREND
+                                               && appConfig.isEnableCheckInfoTooltip() ) {
+                                                       // チェック情報ツールチップの表示
+                                       checkInfoLayerPanel.setMessage(null);
+                                       checkInfoLayerPanel.setVisible(true);
+
+                               } else {
+                                                       // チェック情報ツールチップの非表示
+                                       checkInfoLayerPanel.setVisible(false);
+                               }
+                       }
+               });
+
+               previewImgScrollPane.addMouseMotionListener(new MouseMotionAdapter() {
+                       @Override
+                       public void mouseMoved(MouseEvent e) {
+                               Rectangle rct = previewImgScrollPane.getBounds();
+                               int y = e.getY();
+                               UIHelper uiUtl = UIHelper.getInstance();
+                               if (y > rct.height - (int) (appConfig.getZoomPanelActivationArea() * uiUtl.getScaleY())) {
+                                       previewControlPanel.setVisible(true);
+                               } else {
+                                       if ( !previewControlPanel.isPinned()) {
+                                               previewControlPanel.setVisible(false);
+                                       }
+                               }
+                       }
+               });
+
+               // 標準のホイールリスナは削除する.
+               for (final MouseWheelListener listener : previewImgScrollPane.getMouseWheelListeners()) {
+                       previewImgScrollPane.removeMouseWheelListener(listener);
+               }
+
+               previewImgScrollPane.addMouseWheelListener(new MouseWheelListener() {
+                       public void mouseWheelMoved(MouseWheelEvent e) {
+                               if ((Main.isMacOSX() && e.isAltDown()) ||
+                                               ( !Main.isMacOSX() && e.isControlDown())) {
+                                       // Mac OS XならOptionキー、それ以外はコントロールキーとともにホイールスクロールの場合
+                                       zoomByWheel(e);
+                               } else {
+                                       // ズーム以外のホイール操作はスクロールとする.
+                                       scrollByWheel(e);
+                               }
+                               // 現在画像位置の情報の更新
+                               updateCheckInfoMessage(e.getPoint());
+                       }
+               });
+
+               previewImgScrollPane.addMouseListener(new MouseAdapter() {
+                       @Override
+                       public void mousePressed(MouseEvent e) {
+                               if (e.getClickCount() == 2) {
+                                       // ダブルクリック
+                                       // (正確に2回目。3回目以降はダブルクリック + シングルクリック)
+                                       toggleZoom(e.getPoint());
+                               } else {
+                                       scrollSupport.drag(true, e.getPoint());
+                               }
+                       }
+                       @Override
+                       public void mouseReleased(MouseEvent e) {
+                               scrollSupport.drag(false, e.getPoint());
+                       }
+               });
+
+               previewImgScrollPane.addMouseMotionListener(new MouseMotionListener() {
+
+                       public void mouseMoved(MouseEvent e) {
+                               updateCheckInfoMessage(e.getPoint());
+                       }
+
+                       public void mouseDragged(MouseEvent e) {
+                               scrollSupport.dragging(e.getPoint());
+
+                               // 現在画像位置の情報の更新
+                               updateCheckInfoMessage(e.getPoint());
+                       }
+               });
+       }
+
+       /**
+        * 倍率を切り替える.
+        */
+       protected void toggleZoom(Point mousePos) {
+               if (previewImgPanel.isDefaultZoom()) {
+                       // 等倍であれば以前の倍率を適用する.
+                       zoomWithCenterPosition(latestToggleZoom, mousePos);
+
+               } else {
+                       // 等倍でなければ現在の倍率を記憶して等倍にする.
+                       double currentZoomFactor = previewImgPanel.getZoomFactor();
+                       latestToggleZoom = currentZoomFactor;
+                       zoomWithCenterPosition(1., mousePos);
+               }
+       }
+
+       /**
+        * マウス位置に対して画像情報のツールチップを表示する
+        *
+        * @param mousePosition
+        *            マウス位置
+        */
+       protected void updateCheckInfoMessage(Point mousePosition) {
+               if ( !checkInfoLayerPanel.isVisible()) {
+                       return;
+               }
+               // マウス位置から画像位置を割り出す
+               Point imgPos = null;
+               if (mousePosition != null) {
+                       Point panelPt = SwingUtilities.convertPoint(previewImgScrollPane,
+                                       mousePosition, previewImgPanel);
+                       imgPos = previewImgPanel.getImagePosition(panelPt);
+               }
+               if (imgPos != null) {
+                       // 画像位置があれば、その位置の情報を取得する.
+                       int argb = previewImgPanel.getImageARGB(imgPos);
+                       int a = (argb >> 24) & 0xff;
+                       int r = (argb >> 16) & 0xff;
+                       int g = (argb >> 8) & 0xff;
+                       int b = argb & 0xff;
+                       int y = (int) (0.298912f * r + 0.586611f * g + 0.114478f * b);
+                       String text = String.format(
+                                       "(%3d,%3d)¥nA:%3d, Y:%3d¥nR:%3d, G:%3d, B:%3d", imgPos.x,
+                                       imgPos.y, a, y, r, g, b);
+                       checkInfoLayerPanel.setMessage(text);
+                       checkInfoLayerPanel.setPotision(mousePosition);
+
+               } else {
+                       // 画像位置がなければツールチップは空にする.
+                       checkInfoLayerPanel.setMessage(null);
+               }
+       }
+
+       /**
+        * マウス座標単位で指定したオフセット分スクロールする.
+        *
+        * @param diff_x
+        *            水平方向スクロール数
+        * @param diff_y
+        *            垂直方向スクロール数
+        */
+       protected void scroll(int diff_x, int diff_y) {
+               scrollSupport.scroll(diff_x, diff_y);
+       }
+
+       /**
+        * マウスホイールによる水平・垂直スクロール.<br>
+        * シフトキーで水平、それ以外は垂直とする.<br>
+        *
+        * @param e
+        *            ホイールイベント
+        */
+       protected void scrollByWheel(final MouseWheelEvent e) {
+               scrollSupport.scrollByWheel(e);
+
+               // イベントは処理済みとする.
+               e.consume();
+       }
+
+       /**
+        * ホイールによる拡大縮小.<br>
+        * ホイールの量は関係なく、方向だけで判定する.<br>
+        * プラットフォームごとに修飾キーの判定が異なるので、 呼び出しもとであらかじめ切り分けて呼び出すこと.<br>
+        *
+        * @param e
+        *            ホイールイベント
+        */
+       protected void zoomByWheel(final MouseWheelEvent e) {
+               int wheelRotation = e.getWheelRotation();
+               double currentZoom = previewImgPanel.getZoomFactor();
+               double zoomFactor;
+               if (wheelRotation < 0) {
+                       // ホイール上で拡大
+                       zoomFactor = currentZoom * 1.1;
+
+               } else if (wheelRotation > 0){
+                       // ホイール下で縮小
+                       zoomFactor = currentZoom * 0.9;
+
+               } else {
+                       return;
+               }
+
+               // 倍率変更する
+               zoomWithCenterPosition(zoomFactor, e.getPoint());
+
+               // イベント処理済み
+               e.consume();
+       }
+
+       /**
+        * ズームスライダまたはコンボのいずれかの値を更新すると、他方からも更新通知があがるため 二重処理を防ぐためのセマフォ.<br>
+        */
+       private Semaphore zoomLock = new Semaphore(1);
+
+       /**
+        * プレビューに表示する画像の倍率を更新する.<br>
+        * 指定した座標が拡大縮小の中心点になるようにスクロールを試みる.<br>
+        * 座標がnullの場合は現在表示されている中央を中心とするようにスクロールを試みる.<br>
+        * (スクロールバーが表示されていない、もしくは十分にスクロールできない場合は必ずしも中心とはならない.)<br>
+        * コントロールパネルの表示値も更新する.<br>
+        * コントロールパネルからの更新通知をうけて再入しないように、 同時に一つしか実行されないようにしている.<br>
+        *
+        * @param zoomFactor
+        *            倍率、範囲外のものは範囲内に補正される.
+        * @param mousePos
+        *            スクロールペイン上のマウス座標、もしくはnull(nullの場合は表示中央)
+        */
+       protected void zoomWithCenterPosition(double zoomFactor, Point mousePos) {
+               if ( !zoomLock.tryAcquire()) {
+                       return;
+               }
+               try {
+                       // 範囲制限.
+                       if (zoomFactor < 0.2) {
+                               zoomFactor = 0.2;
+                       } else if (zoomFactor > 8.) {
+                               zoomFactor = 8.;
+                       }
+
+                       JViewport vp = previewImgScrollPane.getViewport();
+
+                       Point viewCenter;
+                       if (mousePos != null) {
+                               // スクロールペインのマウス座標を表示パネルの位置に換算する.
+                               viewCenter = SwingUtilities.convertPoint(this, mousePos, previewImgPanel);
+
+                       } else {
+                               // 表示パネル上の現在表示しているビューポートの中央の座標を求める
+                               Rectangle viewRect = vp.getViewRect();
+                               viewCenter = new Point(
+                                               (viewRect.x + viewRect.width / 2),
+                                               (viewRect.y + viewRect.height / 2)
+                                               );
+                       }
+
+                       // 現在のビューサイズ(余白があれば余白も含む)
+                       Dimension viewSize = previewImgPanel.getScaledSize(true);
+
+                       // 倍率変更
+                       previewControlPanel.setZoomFactor(zoomFactor);
+                       previewImgPanel.setZoomFactor(zoomFactor);
+
+                       // 新しいのビューサイズ(余白があれば余白も含む)
+                       Dimension viewSizeAfter = previewImgPanel.getScaledSize(true);
+                       Dimension visibleSize = vp.getExtentSize();
+
+                       if (viewSize != null && viewSizeAfter != null &&
+                               viewSizeAfter.width > 0 && viewSizeAfter.height > 0 &&
+                               viewSizeAfter.width > visibleSize.width &&
+                               viewSizeAfter.height > visibleSize.height) {
+                               // 新しいビューの大きさよりも表示可能領域が小さい場合のみ
+                               vp.setViewSize(viewSizeAfter);
+
+                               // スクロールペインに表示されている画面サイズを求める.
+                               // スクロールバーがある方向は、コンテンツの最大と等しいが
+                               // スクロールバーがない場合は画面サイズのほうが大きいため、
+                               // 倍率変更による縦横の移動比は、それぞれ異なる.
+                               int visible_width = max(visibleSize.width, viewSize.width);
+                               int visible_height = max(visibleSize.height, viewSize.height);
+                               int visible_width_after = max(visibleSize.width, viewSizeAfter.width);
+                               int visible_height_after = max(visibleSize.height, viewSizeAfter.height);
+
+                               // 前回の倍率から今回の倍率の倍率.
+                               // オリジナルに対する倍率ではない.
+                               // また、画像は縦横同率であるが表示ウィンドウはスクロールバー有無により同率とは限らない.
+                               double zoomDiffX = (double) visible_width_after / (double) visible_width;
+                               double zoomDiffY = (double) visible_height_after / (double) visible_height;
+
+                               // 拡大後の座標の補正
+                               Point viewCenterAfter = new Point();
+                               viewCenterAfter.x = (int) round(viewCenter.x * zoomDiffX);
+                               viewCenterAfter.y = (int) round(viewCenter.y * zoomDiffY);
+
+                               // 倍率適用前後の座標の差分
+                               int diff_x = viewCenterAfter.x - viewCenter.x;
+                               int diff_y = viewCenterAfter.y - viewCenter.y;
+
+                               // スクロール
+                               scroll(diff_x, diff_y);
+                       }
+
+                       // スクロールの単位を画像1ドットあたりの表示サイズに変更する.
+                       // (ただし1を下回らない)
+                       JScrollBar vsb = previewImgScrollPane.getVerticalScrollBar();
+                       JScrollBar hsb = previewImgScrollPane.getHorizontalScrollBar();
+                       vsb.setUnitIncrement(max(1, (int) ceil(zoomFactor)));
+                       hsb.setUnitIncrement(max(1, (int) ceil(zoomFactor)));
+
+               } finally {
+                       zoomLock.release();
+               }
+       }
+
+       /**
+        * 現在のビューの左上位置を返す
+        * @return
+        */
+       public Point getViewPosition() {
+               JViewport vp = previewImgScrollPane.getViewport();
+               return vp.getViewPosition();
+       }
+
+       /**
+        * 指定した座標が中央となるようにスクロールする。
+        * まだ画像が表示されていない場合は次に画像を設定したときに行う。
+        * @param centerPt 中央
+        */
+       public void setViewPosition(Point viewPt) {
+               JViewport vp = previewImgScrollPane.getViewport();
+               if (previewImgPanel.getPreviewImage() != null) {
+                       if (viewPt != null) {
+                               vp.setViewPosition(viewPt);
+                       }
+                       requestViewPt = null;
+               } else {
+                       requestViewPt = viewPt;
+               }
+       }
+
+       private Point requestViewPt;
+
+       /**
+        * プレビューに表示するタイトル.<br>
+        *
+        * @param title
+        *            タイトル
+        */
+       public void setTitle(String title) {
+               if (title == null) {
+                       title = "";
+               }
+               if (!title.equals(this.title)) {
+                       this.title = title;
+                       lblTitle.setText(title + (indicatorShown ? indicatorText : ""));
+                       lblTitle.setToolTipText(title);
+               }
+       }
+
+       public String getTitle() {
+               return this.title;
+       }
+
+       /**
+        * ロードに時間がかかっているか判定し、 インジケータを表示するためのタイマーイベントハンドラ.<br>
+        */
+       protected void onTimer() {
+               boolean waiting;
+               long firstRequest;
+               synchronized (lock) {
+                       waiting = isWaiting();
+                       firstRequest = firstWaitingTimestamp;
+               }
+               boolean indicatorShown = (waiting && ((System.currentTimeMillis() - firstRequest) > indicatorDelay));
+               if (this.indicatorShown != indicatorShown) {
+                       this.indicatorShown = indicatorShown;
+                       lblTitle.setText(title + (indicatorShown ? indicatorText : ""));
+               }
+       }
+
+       /**
+        * チケットの状態が、ロード完了待ち状態であるか?<br>
+        * ロード中のチケットが、ロード完了のチケットより新しければロード中と見なす.<br>
+        *
+        * @return 完了待ちであればtrue、そうでなければfalse
+        */
+       protected boolean isWaiting() {
+               synchronized (lock) {
+                       return loadingTicket > loadedTicket;
+               }
+       }
+
+       /**
+        * ロード要求が出されるたびに、そのロード要求チケットを登録する.<br>
+        * チケットは要求されるたびに増加するシーケンスとする.<br>
+        *
+        * @param ticket
+        *            ロード要求チケット
+        */
+       public void setLoadingRequest(long ticket) {
+               synchronized (lock) {
+                       if ( !isWaiting() && this.loadedTicket < ticket) {
+                               // 現在認識しているチケットの状態がロード完了であり、
+                               // それよりも新しいチケットが要求されたならば、
+                               // 今回のチケットから待ち時間の計測を開始する.
+                               this.firstWaitingTimestamp = System.currentTimeMillis();
+                       }
+                       this.loadingTicket = ticket;
+               }
+       }
+
+       /**
+        * ロード完了するたびに呼び出される.<br>
+        *
+        * @param ticket
+        *            ロード要求チケット.
+        */
+       public void setLoadingComplete(long ticket) {
+               synchronized (lock) {
+                       this.loadedTicket = ticket;
+               }
+       }
+
+       /**
+        * 表示画像を設定する.<br>
+        *
+        * @param previewImg
+        *            表示画像、もしくはnull
+        */
+       public void setPreviewImage(BufferedImage previewImg) {
+               previewImgPanel.setPreviewImage(previewImg);
+               if (requestViewPt != null) {
+                       // 画像設定前にスクロール位置の要求があれば、再適用を試みる
+                       setViewPosition(requestViewPt);
+               }
+       }
+
+       /**
+        * 表示されている画像を取得する.<br>
+        * 表示画像が設定されていなければnull.<br>
+        *
+        * @return 表示画像、もしくはnull
+        */
+       public BufferedImage getPreviewImage() {
+               return previewImgPanel.getPreviewImage();
+       }
+
+       /**
+        * 表示している画面イメージそのままを取得する.
+        *
+        * @return 表示画像
+        */
+       public BufferedImage getScreenImage() {
+               JViewport vp = previewImgScrollPane.getViewport();
+               Dimension dim = vp.getExtentSize();
+               BufferedImage img = new BufferedImage(dim.width, dim.height, BufferedImage.TYPE_INT_ARGB);
+               Graphics2D g = img.createGraphics();
+               try {
+                       vp.paint(g);
+
+               } finally {
+                       g.dispose();
+               }
+               return img;
+       }
+
+       /**
+        * 壁紙を設定する.<br>
+        *
+        * @param wallpaperImg
+        *            壁紙、null不可
+        */
+       public void setWallpaper(Wallpaper wallpaper) {
+               previewImgPanel.setWallpaper(wallpaper);
+       }
+
+       /**
+        * 壁紙を取得する.<br>
+        * 壁紙が未設定の場合は空の壁紙インスタンスが返される.<br>
+        *
+        * @return 壁紙
+        */
+       public Wallpaper getWallpaper() {
+               return previewImgPanel.getWallpaper();
+       }
+
+       /**
+        * 表示倍率を取得する.
+        *
+        * @return 表示倍率
+        */
+       public double getZoomFactor() {
+               return previewControlPanel.getZoomFactor();
+       }
+
+       /**
+        * 表示倍率を設定する
+        *
+        * @param zoomFactor
+        *            表示倍率
+        */
+       public void setZoomFactor(double zoomFactor) {
+               previewControlPanel.setZoomFactor(zoomFactor);
+       }
+
+       /**
+        * ズームパネルのピン留め制御
+        *
+        * @param visible
+        *            表示する場合はtrue
+        */
+       public void setVisibleZoomBox(boolean visible) {
+               previewControlPanel.setPinned(visible);
+       }
+
+       /**
+        * ズームパネルがピン留めされているか?
+        *
+        * @return ピン留めされていればtrue
+        */
+       public boolean isVisibleZoomBox() {
+               return previewControlPanel.isPinned();
+       }
+
+       public void addPreviewPanelListener(PreviewPanelListener listener) {
+               if (listener == null) {
+                       throw new IllegalArgumentException();
+               }
+               listeners.add(listener);
+       }
+
+       public void removePreviewPanelListener(PreviewPanelListener listener) {
+               listeners.remove(listener);
+       }
+
+       protected void savePicture(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.savePicture(e);
+               }
+       }
+
+       protected void flipHolizontal(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.flipHorizontal(e);
+               }
+       }
+
+       protected void copyPicture(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.copyPicture(e);
+               }
+       }
+
+       protected void changeBackgroundColor(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.changeBackgroundColor(e);
+               }
+       }
+
+       protected void showInformation(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.showInformation(e);
+               }
+       }
+
+       protected void addFavorite(PreviewPanelEvent e) {
+               for (PreviewPanelListener listener : listeners) {
+                       listener.addFavorite(e);
+               }
+       }
+}
+
+/**
+ * チェック情報の表示用レイヤーパネル.<br>
+ *
+ * @author seraphy
+ */
+class CheckInfoLayerPanel extends JPanel {
+       private static final long serialVersionUID = 1L;
+
+       /**
+        * ロガー
+        */
+       private static final Logger logger = Logger.getLogger(CheckInfoLayerPanel.class.getName());
+
+       /**
+        * ボックスの余白
+        */
+       private Insets padding = new Insets(3, 3, 3, 3);
+
+       /**
+        * 表示位置プロパティ
+        */
+       private Point pos = new Point();
+
+       /**
+        * 表示メッセージプロパティ.<br>
+        * ¥nで改行となる.<br>
+        * 空文字ならば非表示.<br>
+        */
+       private String message = "";
+
+       /**
+        * 解析済みメッセージ.<br>
+        * 業に分割される.<br>
+        * 空文字は空のリストとなる.<br>
+        */
+       private String[] messageLines;
+
+       /**
+        * 解析済みフォントの高さ.<br>
+        */
+       private int fontHeight;
+
+       /**
+        * 描画済みエリア.<br>
+        * 次回描画前に消去する必要のある領域.<br>
+        * まだ一度も描画してなければnull.<br>
+        */
+       private Rectangle eraseRect;
+
+       /**
+        * 現在、描画すべきエリア.<br>
+        * なければnull.<br>
+        */
+       private Rectangle requestRect;
+
+       /**
+        * 画面に関連づけられていない状態でのテキスト表示サイズは 計算できないため、画面追加時に再計算させるための 予約フラグ.<br>
+        */
+       private boolean requestRecalcOnAdd;
+
+       /**
+        * フォントのためのデスクトップヒント.(あれば)
+        */
+       @SuppressWarnings("rawtypes")
+       private Map desktopHintsForFont;
+
+       /**
+        * 透明コンポーネントとして構築する.<br>
+        */
+       @SuppressWarnings("rawtypes")
+       public CheckInfoLayerPanel() {
+               setOpaque(false);
+
+               Toolkit tk = Toolkit.getDefaultToolkit();
+               desktopHintsForFont = (Map) tk.getDesktopProperty("awt.font.desktophints");
+               logger.log(Level.CONFIG, "awt.font.desktophints=" + desktopHintsForFont);
+       }
+
+       /**
+        * 指定エリアに情報を描画する.<br>
+        */
+       @Override
+       protected void paintComponent(Graphics g0) {
+               Graphics2D g = (Graphics2D) g0;
+               super.paintComponent(g);
+
+               // クリップ領域
+               Rectangle clip = g.getClipBounds();
+               // System.out.println("clip:" + clip + " /eraseRect:" + eraseRect + " /drawRect:" + requestRect);
+
+               // 削除すべき領域が描画範囲に含まれているか?
+               // (含まれていれば、その領域は消去済みである.)
+               if (clip == null || (eraseRect != null && clip.contains(eraseRect))) {
+                       eraseRect = null;
+               }
+
+               // 表示領域の判定
+               if (requestRect == null || requestRect.isEmpty()
+                               || !(clip != null && clip.intersects(requestRect))) {
+                       // 表示すべき領域が存在しないか、描画要求範囲にない.
+                       return;
+               }
+               if (messageLines == null || messageLines.length == 0) {
+                       // 表示するものがなければ何もしない.
+                       return;
+               }
+
+               // フォントのレンダリングヒント
+               if (desktopHintsForFont != null) {
+                       g.addRenderingHints(desktopHintsForFont);
+               }
+
+               // 箱の描画
+               g.setColor(new Color(255, 255, 255, 192));
+               g.fillRect(requestRect.x, requestRect.y, requestRect.width, requestRect.height);
+               g.setColor(Color.GRAY);
+               g.drawRect(requestRect.x, requestRect.y, requestRect.width - 1, requestRect.height - 1);
+
+               // 情報の描画
+               g.setColor(Color.BLACK);
+               int oy = fontHeight;
+               for (String messageLine : messageLines) {
+                       g.drawString(messageLine, requestRect.x + padding.left, requestRect.y + padding.top - 1 + oy);
+                       oy += fontHeight;
+               }
+
+               // 描画された領域を次回消去領域として記憶する.
+               if (eraseRect == null || eraseRect.isEmpty()) {
+                       // 消去済みであれば、今回分のみを次回消去領域とする.
+                       eraseRect = (Rectangle) requestRect.clone();
+
+               } else {
+                       // 消去済みエリアが未消去で残っている場合は
+                       // 今回領域を結合する.
+                       eraseRect.add(requestRect);
+               }
+       }
+
+       /**
+        * 画面にアタッチされた場合、描画領域の再計算が 必要であれば計算する.<br>
+        */
+       @Override
+       public void addNotify() {
+               super.addNotify();
+               if (requestRecalcOnAdd) {
+                       requestRecalcOnAdd = false;
+                       calcRepaint();
+               }
+       }
+
+       /**
+        * 要求されたプロパティから、フォント高さによる表示領域を計算し、 その領域の再描画を要求する.(描画する内容がなれば、描画要求しない.)<br>
+        * 前回表示領域があれば、消去するために、そのエリアも再描画を要求する.<br>
+        * それ以外のエリアは描画要求しない.(描画の最適化による負荷軽減策)<br>
+        * フォントサイズを求めるためにグラフィクスへのアクセスが必要となるが、 まだ取得できない場合は{@link #addNotify()}の呼び出し時に
+        * 再計算するようにフラグを立てておく.<br>
+        */
+       protected void calcRepaint() {
+               Graphics2D g = (Graphics2D) getGraphics();
+               if (g == null) {
+                       requestRecalcOnAdd = true;
+                       return;
+               }
+               try {
+                       // 前回描画領域のクリアのために呼び出す.
+                       if (eraseRect != null && !eraseRect.isEmpty()) {
+                               repaint(eraseRect);
+                       }
+
+                       // 空であれば新たな描画なし.
+                       if (message.length() == 0) {
+                               requestRect = null;
+                               return;
+                       }
+
+                       FontMetrics fm = g.getFontMetrics();
+                       String[] messageLines = message.split("¥n");
+
+                       Rectangle2D rct = null;
+                       for (String messageLine : messageLines) {
+                               Rectangle2D tmp = fm.getStringBounds(messageLine, g);
+                               if (rct != null) {
+                                       rct.add(tmp);
+
+                               } else {
+                                       rct = tmp;
+                               }
+                       }
+
+                       int fw = (int) rct.getWidth();
+                       int fh = (int) rct.getHeight();
+
+                       int w = fw + padding.left + padding.right;
+                       int h = fh * messageLines.length + padding.top + padding.bottom;
+
+                       // 指定した位置の右上あたりにする
+                       int x = pos.x + 16;
+                       int y = pos.y - h;
+
+                       // サイズ
+                       int client_w = getWidth();
+                       int client_h = getHeight();
+
+                       if (x + w > client_w) {
+                               // 画面右の場合はカーソルの左に移動
+                               x = pos.x - w - 10;
+                       }
+                       if (y < 0) {
+                               // 画面上の場合はカーソルの下に移動
+                               y = pos.y + 10;
+                       }
+                       if (y + h > client_h) {
+                               y -= (y + h - client_h);
+                       }
+
+                       // 結果の格納
+                       this.requestRect = new Rectangle(x, y, w, h);
+                       this.messageLines = messageLines;
+                       this.fontHeight = fh;
+
+                       // 再描画の要求
+                       Rectangle paintRect = (Rectangle) requestRect.clone();
+                       repaint(paintRect);
+
+               } finally {
+                       g.dispose();
+               }
+       }
+
+       public void setPotision(Point requestPt) {
+               if (requestPt == null) {
+                       throw new IllegalArgumentException();
+               }
+               if ( !requestPt.equals(pos)) {
+                       Point oldpos = pos;
+                       pos = (Point) requestPt.clone();
+                       calcRepaint();
+                       firePropertyChange("position", oldpos, pos);
+               }
+       }
+
+       public Point getPosition() {
+               return (Point) pos.clone();
+       }
+
+       public void setMessage(String message) {
+               if (message == null) {
+                       message = "";
+               }
+               message = message.replace("¥r¥n", "¥n");
+               if ( !message.equals(this.message)) {
+                       String oldmes = this.message;
+                       this.message = message;
+                       calcRepaint();
+                       firePropertyChange("message", oldmes, message);
+               }
+       }
+
+       public String getMessage() {
+               return message;
+       }
+}
+
+/**
+ * 画像表示パネル
+ *
+ * @author seraphy
+ */
+class PreviewImagePanel extends JPanel {
+       private static final long serialVersionUID = 1L;
+
+       /**
+        * 背景モード.<br>
+        */
+       private BackgroundColorMode bgColorMode;
+
+       /**
+        * 壁紙.<br>
+        */
+       private Wallpaper wallpaper;
+
+       /**
+        * 壁紙変更イベントのリスナ
+        */
+       private PropertyChangeListener wallpaperListener;
+
+       /**
+        * 透過オリジナル画像.<br>
+        */
+       private BufferedImage previewImg;
+
+       /**
+        * 表示用画像(背景モードによる調整あり).<br>
+        * 事前に拡大縮小を適用済みの場合は、{@link #scaledZoomFactor}に 適用している倍率が設定される.<br>
+        * 表示用に改めてイメージを生成する必要がない場合は、 透過オリジナルと同じインスタンスとなりえる.<br>
+        */
+       private BufferedImage previewImgForDraw;
+
+       /**
+        * 表示用画像がスケール済みである場合、そのスケールが設定される.<br>
+        * スケール済みでない場合はnullとなる.<br>
+        */
+       private Double scaledZoomFactor;
+
+
+       /**
+        * 倍率
+        */
+       private double zoomFactor = 1.;
+
+       /**
+        * 許容誤差
+        */
+       private static final double TOLERANT = 0.001;
+
+
+       /**
+        * コンストラクタ
+        */
+       public PreviewImagePanel() {
+               super();
+
+               // 通常モード
+               bgColorMode = BackgroundColorMode.ALPHABREND;
+
+               // 壁紙変更通知リスナ
+               wallpaperListener = new PropertyChangeListener() {
+                       public void propertyChange(PropertyChangeEvent evt) {
+                               onChangeWallpaper();
+                       }
+               };
+
+               // 壁紙
+               wallpaper = new Wallpaper();
+               wallpaper.addPropertyChangeListener(wallpaperListener);
+       }
+
+       /**
+        * 画像を表示する.
+        */
+       @Override
+       protected void paintComponent(Graphics g0) {
+               Graphics2D g = (Graphics2D) g0;
+               super.paintComponent(g);
+
+               if (previewImgForDraw == null) {
+                       return;
+               }
+
+               // 倍率を適用した画像を画面の中央に配置できるように計算する.
+               // (画像が倍率適用済みであれば1倍とする)
+               Rectangle imgRct = adjustImageRectangle();
+
+               // 表示用画像がスケール済みでない場合はレンダリングオプションを適用する.
+               if (scaledZoomFactor == null) {
+                       Object renderingOption = getRenderingOption();
+                       g.setRenderingHint(RenderingHints.KEY_INTERPOLATION, renderingOption);
+               }
+
+               // 背景処理
+               if (bgColorMode == BackgroundColorMode.ALPHABREND) {
+                       // 表示の最大範囲 (可視領域外も含む)
+                       int w = getWidth();
+                       int h = getHeight();
+                       wallpaper.drawWallpaper(g, w, h);
+               }
+
+               // レンダリング
+               g.drawImage(previewImgForDraw,
+                               imgRct.x, imgRct.y,
+                               imgRct.x + imgRct.width, imgRct.y + imgRct.height,
+                               0, 0,
+                               previewImgForDraw.getWidth(), previewImgForDraw.getHeight(),
+                               null);
+
+               // 通常モード以外のグリッド描画に該当するモードはグリッドを前景に描く
+               AppConfig appConfig = AppConfig.getInstance();
+               int drawGridMask = appConfig.getDrawGridMask();
+               if ((drawGridMask & bgColorMode.mask()) != 0) {
+                       Color oldc = g.getColor();
+                       try {
+                               g.setColor(appConfig.getPreviewGridColor());
+                               drawGrid(g, imgRct.x, imgRct.y, appConfig.getPreviewGridSize());
+
+                       } finally {
+                               g.setColor(oldc);
+                       }
+               }
+       }
+
+
+       /**
+        * グリッドを描画する.<br>
+        * 開始位置の-1単位位置から画像サイズの+1単位までがグリッド範囲となる。
+        *
+        * @param g
+        * @param offset_x
+        *            開始位置
+        * @param offset_y
+        *            開始位置
+        * @param unit
+        *            グリッド単位(pixel)
+        */
+       protected void drawGrid(Graphics2D g, int offset_x, int offset_y, int unit) {
+               Rectangle clip = g.getClipBounds();
+
+               int src_w = previewImg.getWidth();
+               int src_h = previewImg.getHeight();
+               int my = src_h / unit;
+               int mx = src_w / unit;
+
+               int st_x = offset_x + (int)(-1 * unit * zoomFactor);
+               int en_x = offset_x + (int)((mx + 1) * unit * zoomFactor);
+               int w = en_x - st_x + 1;
+
+               for (int y = -1; y <= my + 1; y++) {
+                       int y1 = y * unit;
+                       Rectangle rct = new Rectangle(
+                                       st_x, offset_y + (int)(y1 * zoomFactor),
+                                       w, 1);
+                       if (clip == null || clip.intersects(rct)) {
+                               g.drawLine(rct.x, rct.y, rct.x + rct.width, rct.y);
+                       }
+               }
+
+               int st_y = offset_y + (int)(-1 * unit * zoomFactor);
+               int en_y = offset_y + (int)((my + 1) * unit * zoomFactor);
+               int h = en_y - st_y + 1;
+
+               for (int x = -1; x <= mx + 1; x++) {
+                       int x1 = x * unit;
+                       Rectangle rct = new Rectangle(
+                                       offset_x + (int)(x1 * zoomFactor), st_y,
+                                       1, h);
+                       g.drawLine(rct.x, rct.y, rct.x, rct.y + rct.height);
+               }
+       }
+
+       /**
+        * 現在の倍率に応じたレンダリングオプションを取得する.<br>
+        *
+        * @return レンダリングオプション
+        */
+       protected Object getRenderingOption() {
+               AppConfig appConfig = AppConfig.getInstance();
+               double rendringOptimizeThreshold;
+               if (bgColorMode == BackgroundColorMode.ALPHABREND) {
+                       rendringOptimizeThreshold = appConfig.getRenderingOptimizeThresholdForNormal();
+               } else {
+                       rendringOptimizeThreshold = appConfig.getRenderingOptimizeThresholdForCheck();
+               }
+               Object renderingHint;
+               if (zoomFactor < rendringOptimizeThreshold) {
+                       // 補正を適用する最大倍率以内である場合
+                       if (zoomFactor <= 1. || !appConfig.isEnableInterpolationBicubic()) {
+                               // 縮小する場合、もしくはバイキュービックをサポートしない場合
+                               renderingHint = RenderingHints.VALUE_INTERPOLATION_BILINEAR;
+                       } else {
+                               // 拡大する場合でバイキュービックをサポートしている場合
+                               renderingHint = RenderingHints.VALUE_INTERPOLATION_BICUBIC;
+                       }
+
+               } else {
+                       // 補正を適用する最大倍率を超えている場合は補正なし.
+                       renderingHint = RenderingHints.VALUE_INTERPOLATION_NEAREST_NEIGHBOR;
+               }
+               return renderingHint;
+       }
+
+       /**
+        * 倍率と、画面のサイズと、表示するオリジナルの画像サイズをもとに、 倍率を適用した画像サイズを、画面に収まる位置に補正して返す.<br>
+        * 返される矩形の幅と高さ(width, height)は拡大後の画像サイズに等しい.<br>
+        * 拡大後の画像が画面よりも小さければセンタリングするように矩形の開始位置(x, y)がオフセットされる.<br>
+        * そうでなければ矩形の開始位置(x, y)は0となる.<br>
+        * 画像が設定されていなければ幅と高さがゼロの矩形が返される.<br>
+        *
+        * @return 画像を表示するオフセットと大きさ、もしくは空の矩形
+        */
+       public Rectangle adjustImageRectangle() {
+               if (previewImg == null) {
+                       return new Rectangle(0, 0, 0, 0); // 幅・高さともにゼロ
+               }
+               int client_w = getWidth();
+               int client_h = getHeight();
+
+               int src_w = previewImg.getWidth();
+               int src_h = previewImg.getHeight();
+
+               int w = (int) round(src_w * zoomFactor);
+               int h = (int) round(src_h * zoomFactor);
+
+               int offset_x = 0;
+               if (w < client_w) {
+                       offset_x = (client_w - w) / 2;
+               }
+               int offset_y = 0;
+               if (h < client_h) {
+                       offset_y = (client_h - h) / 2;
+               }
+
+               return new Rectangle(offset_x, offset_y, w, h);
+       }
+
+       /**
+        * パネルのマウス座標から、実寸の画像のピクセル位置を返す.<br>
+        * 画像が表示されていないか、範囲外であればnullを返す.<br>
+        *
+        * @param pt
+        *            パネルの座標
+        * @return 画像の位置、もしくはnull
+        */
+       public Point getImagePosition(Point pt) {
+               if (pt == null || previewImg == null) {
+                       // プレビュー画像が設定されていなければnull
+                       return null;
+               }
+
+               Rectangle imgRct = adjustImageRectangle();
+
+               if ( !imgRct.contains(pt.x, pt.y)) {
+                       // 範囲外であればnull
+                       return null;
+               }
+
+               // オフセットを除去する.
+               Point ret = (Point) pt.clone();
+               ret.x -= imgRct.x;
+               ret.y -= imgRct.y;
+
+               // 倍率を解除する.
+               ret.x = (int) floor(ret.x / zoomFactor);
+               ret.y = (int) floor(ret.y / zoomFactor);
+
+               return ret;
+       }
+
+       /**
+        * 画像の位置から画面の位置を割り出す.<br>
+        *
+        * @param pt
+        *            画像の位置
+        * @return 画面の位置
+        */
+       public Point getMousePosition(Point pt) {
+               if (pt == null || previewImg == null) {
+                       // プレビュー画像が設定されていなければnull
+                       return null;
+               }
+
+               Rectangle imgRct = adjustImageRectangle();
+
+               // 表示倍率を加える
+               Point ret = (Point) pt.clone();
+               ret.x = (int) ceil(ret.x * zoomFactor);
+               ret.y = (int) ceil(ret.y * zoomFactor);
+
+               // オフセットを加える
+               ret.x += imgRct.x;
+               ret.y += imgRct.y;
+
+               return ret;
+       }
+
+       /**
+        * 指定した位置のRGB値を取得する.<br>
+        * 範囲外の場合は0が返される.<br>
+        *
+        * @param pt
+        *            イメージの位置
+        * @return イメージのARGB値 (ビット順序は、A:24, R:16, G:8, B:0)
+        */
+       public int getImageARGB(Point pt) {
+               if (pt == null) {
+                       throw new IllegalArgumentException();
+               }
+               try {
+                       return previewImg.getRGB(pt.x, pt.y);
+
+               } catch (RuntimeException ex) {
+                       return 0; // 範囲外
+               }
+       }
+
+       /**
+        * 倍率を適用した画像パネルのサイズを計算し適用する.<br>
+        * モードにより余白が加えられる.<br>
+        */
+       protected void recalcScaledSize() {
+               Dimension scaledSize = getScaledSize(true);
+               if (scaledSize != null) {
+                       setPreferredSize(scaledSize);
+                       revalidate();
+               }
+       }
+
+       /**
+        * 元画像の倍率適用後のサイズを返す.<br>
+        * 元画像が設定されていなければnull.<br>
+        * needOffsetがfalseであれば表示モードに関わらず、画像の拡大・縮小後の純粋なサイズ、
+        * trueであれば余白が必要な表示モードの場合の余白が付与された場合のサイズが返される.<br>
+        *
+        * @param needOffset
+        *            余白を加味したサイズが必要な場合はtrue
+        * @return 倍率適用後のサイズ、もしくはnull
+        */
+       protected Dimension getScaledSize(boolean needOffset) {
+               if (previewImg == null) {
+                       return null;
+               }
+               int src_w = previewImg.getWidth();
+               int src_h = previewImg.getHeight();
+
+               int w = (int) round(src_w * zoomFactor);
+               int h = (int) round(src_h * zoomFactor);
+
+               Dimension scaledSize = new Dimension(w, h);
+
+               if (bgColorMode != BackgroundColorMode.ALPHABREND) {
+                       // 通常モード以外は画像よりも少し大きめにすることで
+                       // キャンバスに余白をつける
+                       AppConfig appConfig = AppConfig.getInstance();
+                       int unfilledSpace = appConfig.getPreviewUnfilledSpaceForCheckMode();
+                       scaledSize.width += max(0, unfilledSpace * 2);
+                       scaledSize.height += max(0, unfilledSpace * 2);
+               }
+
+               return scaledSize;
+       }
+
+       /**
+        * プレビュー画像を設定する.
+        *
+        * @param previewImg
+        */
+       public void setPreviewImage(BufferedImage previewImg) {
+               BufferedImage oldimg = this.previewImg;
+               this.previewImg = previewImg;
+
+               recalcScaledSize();
+               makeDrawImage(true);
+               repaint();
+
+               firePropertyChange("previewImage", oldimg, previewImg);
+       }
+
+       public BufferedImage getPreviewImage() {
+               return previewImg;
+       }
+
+       /**
+        * 壁紙を設定する.
+        *
+        * @param wallpaper
+        */
+       public void setWallpaper(Wallpaper wallpaper) {
+               if (wallpaper == null) {
+                       throw new IllegalArgumentException();
+               }
+               if ( !this.wallpaper.equals(wallpaper)) {
+                       Wallpaper wallpaperOld = this.wallpaper;
+                       if (wallpaperOld != null) {
+                               wallpaperOld.removePropertyChangeListener(wallpaperListener);
+                       }
+                       this.wallpaper = wallpaper;
+                       if (this.wallpaper != null) {
+                               this.wallpaper.addPropertyChangeListener(wallpaperListener);
+                       }
+                       firePropertyChange("wallpaper", wallpaperOld, this.wallpaper);
+                       onChangeWallpaper();
+               }
+       }
+
+       public Wallpaper getWallpaper() {
+               return wallpaper;
+       }
+
+       protected void onChangeWallpaper() {
+               repaint();
+       }
+
+       /**
+        * 背景モード調整済みの表示用画像を作成する.
+        *
+        * @param changeImage
+        *            画像の変更あり
+        */
+       protected void makeDrawImage(boolean changeImage) {
+               if (previewImg == null) {
+                       // 画像が設定されていなければ空
+                       this.previewImgForDraw = null;
+                       scaledZoomFactor = null;
+                       return;
+               }
+
+               BufferedImage img;
+               if (changeImage || scaledZoomFactor != null) {
+                       // 画像が変更されているか、スケール済みであれば
+                       // 背景モードの再適用が必要.
+                       if (bgColorMode == BackgroundColorMode.ALPHABREND) {
+                               // アルファブレンド通常モードは背景用にあえて作成する必要はない.
+                               img = previewImg;
+
+                       } else {
+                               // アルファブレンド通常モード以外は背景に作成する
+                               Color bgColor = wallpaper.getBackgroundColor();
+                               BackgroundColorFilter bgColorFilter = new BackgroundColorFilter(bgColorMode, bgColor);
+                               img = bgColorFilter.filter(previewImg, null);
+                       }
+
+               } else {
+                       // 画像が変更されておらず、スケール済みでもなければ
+                       // すでに作成済みの画像が使用できる.
+                       img = previewImgForDraw;
+               }
+
+               // レンダリングオプション
+               Object renderingOption = getRenderingOption();
+
+               // バイキュービックでなければ、事前の拡大縮小は行わずに、表示時に行う.
+               if ( !renderingOption.equals(RenderingHints.VALUE_INTERPOLATION_BICUBIC)) {
+                       previewImgForDraw = img;
+                       scaledZoomFactor = null;
+                       return;
+               }
+
+               // バイキュービックの場合、倍率を適用したサイズに予め加工しておく.
+               Dimension scaledSize = getScaledSize(false);
+               BufferedImage offscreen = new BufferedImage(
+                               scaledSize.width, scaledSize.height, BufferedImage.TYPE_INT_ARGB);
+               Graphics2D g = offscreen.createGraphics();
+               try {
+                       g.setRenderingHint(RenderingHints.KEY_INTERPOLATION,
+                                       RenderingHints.VALUE_INTERPOLATION_BICUBIC);
+
+                       g.drawImage(img,
+                                       0, 0, scaledSize.width, scaledSize.height,
+                                       0, 0, img.getWidth(), img.getHeight(),
+                                       null);
+
+               } finally {
+                       g.dispose();
+               }
+               previewImgForDraw = offscreen;
+               scaledZoomFactor = Double.valueOf(zoomFactor);
+       }
+
+       public void setBackgroundColorMode(BackgroundColorMode bgColorMode) {
+               if (bgColorMode == null) {
+                       throw new IllegalArgumentException();
+               }
+               if (this.bgColorMode != bgColorMode) {
+                       BackgroundColorMode oldcm = bgColorMode;
+                       this.bgColorMode = bgColorMode;
+
+                       makeDrawImage(true);
+                       recalcScaledSize();
+                       repaint();
+
+                       firePropertyChange("backgroundColorMode", oldcm, bgColorMode);
+               }
+       }
+
+       public BackgroundColorMode setBackgroundColorMode() {
+               return bgColorMode;
+       }
+
+       public void setZoomFactor(double zoomFactor) {
+               if (abs(zoomFactor - this.zoomFactor) > TOLERANT) {
+                       // 0.001未満の差異は誤差とみなして反映しない.
+                       double oldzoom = this.zoomFactor;
+                       this.zoomFactor = zoomFactor;
+
+                       recalcScaledSize();
+                       makeDrawImage(false);
+                       repaint();
+
+                       firePropertyChange("zoomFactor", oldzoom, zoomFactor);
+               }
+       }
+
+       public double getZoomFactor() {
+               return zoomFactor;
+       }
+
+       /**
+        * 倍率が100%であるか?
+        *
+        * @return 100%であればtrue
+        */
+       public boolean isDefaultZoom() {
+               return zoomFactor - 1 < TOLERANT;
+       }
+}
+
+/**
+ * 倍率・背景モードを操作するための下部パネル用
+ *
+ * @author seraphy
+ */
+class PreviewControlPanel extends JPanel {
+       private static final long serialVersionUID = 1L;
+
+       private static final Logger logger = Logger.getLogger(PreviewControlPanel.class.getName());
+
+       protected static final String STRINGS_RESOURCE = "languages/previewpanel";
+
+       /**
+        * ピン留めチェックボックス
+        */
+       private JCheckBox chkPinning;
+
+       /**
+        * アルファ確認チェックボックス
+        */
+       private JCheckBox chkNoAlpha;
+
+       /**
+        * グレースケール確認チェックボックス
+        */
+       private JCheckBox chkGrayscale;
+
+       /**
+        * 倍率用スライダ
+        */
+       private JSlider zoomSlider;
+
+       /**
+        * 倍率入力用コンボボックス
+        */
+       private JComboBox zoomCombo;
+
+
+       /**
+        * スライダの最小値
+        */
+       private static final int MIN_INDEX = -170;
+
+       /**
+        * スライダの最大値
+        */
+       private static final int MAX_INDEX = 219;
+
+       /**
+        * 最小倍率
+        */
+       private double minimumZoomFactor;
+
+       /**
+        * 最大倍率
+        */
+       private double maximumZoomFactor;
+
+
+       /**
+        * 現在の倍率(100倍済み)
+        */
+       private int currentZoomFactorInt;
+
+       /**
+        * 現在の背景色モード
+        */
+       private BackgroundColorMode backgroundColorMode;
+
+
+       /**
+        * 任意の底Aをもつ対数 logA(N)を計算して返す.
+        *
+        * @param a
+        *            底
+        * @param x
+        *            引数
+        * @return logA(N)
+        */
+       private static double logN(double a, double x) {
+               return log(x) / log(a);
+       }
+
+       /**
+        * 倍率(等倍を1とする)に対するスライダのインデックス値を返す.<br>
+        * スライダは10ステップごとに前のステップの10%づつ増減する.(複利式)<br>
+        *
+        * @param zoomFactor
+        *            倍率(1を等倍とする)
+        * @return インデックス
+        */
+       private static int zoomFactorToIndex(double zoomFactor) {
+               return (int) round(logN((1. + 0.1), zoomFactor) * 10);
+       }
+
+       /**
+        * スライダのインデックス値から倍率(等倍を1とする)を返す.<br>
+        * 10ステップごとに10%づつ増減する.(複利式)<br>
+        *
+        * @param index
+        *            インデックス
+        * @return 倍率(1を等倍とする)
+        */
+       private static double zoomFactorFromIndex(int index) {
+               return pow(1. + 0.1, index / 10.);
+       }
+
+
+       /**
+        * コンストラクタ
+        */
+       public PreviewControlPanel() {
+               final Properties strings = LocalizedResourcePropertyLoader.getCachedInstance()
+                               .getLocalizedProperties(STRINGS_RESOURCE);
+
+               final UIHelper uiHelper = UIHelper.getInstance();
+
+               // ピンアイコン
+               Icon pinIcon = uiHelper.createTwoStateIcon(
+                               "icons/pin-icon1.png", "icons/pin-icon2.png");
+
+               // ピンチェックボックス
+               chkPinning = new JCheckBox(pinIcon);
+               chkPinning.setToolTipText(strings.getProperty("tooltip.zoompanel.pinning"));
+
+               // 円ボタン型チェックボックス用アイコンの実装
+               final Icon stateIcon = new Icon() {
+                       public int getIconHeight() {
+                               return (int)(12 * uiHelper.getScaleX());
+                       }
+                       public int getIconWidth() {
+                               return (int)(6 * uiHelper.getScaleY());
+                       };
+                       public void paintIcon(Component c, Graphics g, int x, int y) {
+                               boolean sw = false;
+                               if (c instanceof AbstractButton) {
+                                       AbstractButton btn = (AbstractButton) c;
+                                       sw = btn.isSelected();
+                               }
+
+                               int w = getIconWidth();
+                               int h = getIconHeight();
+
+                               int s = min(w, h);
+
+                               int ox = 0;
+                               int oy = 0;
+                               if (w > s) {
+                                       ox = (w - s) / 2;
+                               }
+                               if (h > s) {
+                                       oy = (h - s) / 2;
+                               }
+
+                               if (sw) {
+                                       AppConfig appConfig = AppConfig.getInstance();
+                                       Color fillColor = appConfig.getSelectedItemBgColor();
+                                       g.setColor(fillColor);
+                                       g.fillOval(x + ox, y + oy, s, w);
+                               }
+                               g.setColor(Color.GRAY);
+                               g.drawOval(x + ox, y + oy, s, s);
+                       }
+               };
+
+               // アルファ確認とグレースケール確認用のチェックボックス
+               chkNoAlpha = new JCheckBox(stateIcon);
+               chkGrayscale = new JCheckBox(stateIcon);
+
+               chkNoAlpha.setToolTipText(strings.getProperty("tooltip.zoompanel.checkalpha"));
+               chkGrayscale.setToolTipText(strings.getProperty("tooltip.zoompanel.checkgrayscale"));
+
+               backgroundColorMode = BackgroundColorMode.ALPHABREND;
+
+               final ChangeListener chkAlphaGrayChangeListener = new ChangeListener() {
+                       public void stateChanged(ChangeEvent e) {
+                               onChangeCheckAlphaGray();
+                       }
+               };
+               chkNoAlpha.addChangeListener(chkAlphaGrayChangeListener);
+               chkGrayscale.addChangeListener(chkAlphaGrayChangeListener);
+
+               // 倍率スライダ
+               zoomSlider = new JSlider(JSlider.HORIZONTAL, MIN_INDEX, MAX_INDEX, 0);
+               zoomSlider.setToolTipText(strings.getProperty("tooltip.zoompanel.zoomfactor_slider"));
+
+               // 倍率コンボ
+               zoomCombo = new JComboBox();
+               zoomCombo.setToolTipText(strings.getProperty("tooltip.zoompanel.zoomfactor_combo"));
+
+               // 倍率の既定リストの設定と、最大・最小値の算定
+               minimumZoomFactor = zoomFactorFromIndex(zoomSlider.getMinimum());
+               maximumZoomFactor = zoomFactorFromIndex(zoomSlider.getMaximum());
+
+               int minZoomRange = (int) round(minimumZoomFactor * 100.);
+               int maxZoomRange = (int) round(maximumZoomFactor * 100.);
+
+               List<Integer> predefinedZoomRanges = getPredefinedZoomRanges();
+               for (int zoomRange : predefinedZoomRanges) {
+                       if (zoomRange < minZoomRange) {
+                               minZoomRange = zoomRange;
+                       }
+                       if (zoomRange > maxZoomRange) {
+                               maxZoomRange = zoomRange;
+                       }
+                       zoomCombo.addItem(Integer.toString(zoomRange));
+               }
+               final int[] zoomRanges = {minZoomRange, maxZoomRange};
+
+               currentZoomFactorInt = 100;
+               zoomCombo.setSelectedItem(Integer.toString(currentZoomFactorInt));
+               zoomCombo.setEditable(true);
+               if ( !Main.isMacOSX()) {
+                       // Windows環境だとデフォルトで9桁分のテキストフィールドが作成され
+                       // それがレイアウトの推奨サイズとして実際に使われてしまうため、
+                       // 明示的に3桁にとどめておくようにオーバーライドする.
+                       // Mac OS Xならば問題ない.
+                       zoomCombo.setEditor(new BasicComboBoxEditor() {
+                               {
+                                       editor = new JTextField(3) {
+                                               private static final long serialVersionUID = 1L;
+                                               @Override
+                                               public void setBorder(Border border) {
+                                                       // 何もしない.
+                                               }
+                                               public void setText(String s) {
+                                                       if (getText().equals(s)) {
+                                                               return;
+                                                       }
+                                                       super.setText(s);
+                                               }
+                                       };
+                               }
+                       });
+               }
+
+               // スライダを変更することによりコンボボックスを変更する、
+               // もしくはコンボボックスを変更することでスライダを変更するが、
+               // 互いに通知を呼び合うことになるため他方を無視するためのセマフォ
+               final Semaphore changeLock = new Semaphore(1);
+
+               zoomCombo.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               boolean adjusted = false;
+                               String value = (String) zoomCombo.getSelectedItem();
+                               int zoomFactorInt;
+                               try {
+                                       zoomFactorInt = Integer.parseInt(value);
+                                       if (zoomFactorInt < zoomRanges[0]) {
+                                               zoomFactorInt = zoomRanges[0];
+                                               adjusted = true;
+
+                                       } else if (zoomFactorInt > zoomRanges[1]) {
+                                               zoomFactorInt = zoomRanges[1];
+                                               adjusted = true;
+                                       }
+
+                               } catch (RuntimeException ex) {
+                                       zoomFactorInt = 100;
+                                       adjusted = true;
+                               }
+                               if (adjusted) {
+                                       zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));
+                                       Toolkit tk = Toolkit.getDefaultToolkit();
+                                       tk.beep();
+                               }
+                               if (changeLock.tryAcquire()) {
+                                       try {
+                                               zoomSlider.setValue(zoomFactorToIndex(zoomFactorInt / 100.));
+
+                                       } finally {
+                                               changeLock.release();
+                                       }
+                               }
+                               fireZoomFactorChange(zoomFactorInt);
+                       }
+               });
+
+               zoomSlider.addChangeListener(new ChangeListener() {
+                       public void stateChanged(ChangeEvent e) {
+                               int index = zoomSlider.getValue();
+                               double zoomFactor = zoomFactorFromIndex(index);
+                               int zoomFactorInt = (int) round(zoomFactor * 100);
+
+                               if (changeLock.tryAcquire()) {
+                                       try {
+                                               zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));
+
+                                       } finally {
+                                               changeLock.release();
+                                       }
+                                       fireZoomFactorChange(zoomFactorInt);
+                               }
+                       }
+               });
+
+               // パーツの配備
+
+               GridBagLayout gbl = new GridBagLayout();
+               setLayout(gbl);
+
+               GridBagConstraints gbc = new GridBagConstraints();
+               gbc.gridx = 0;
+               gbc.gridy = 0;
+               gbc.ipadx = 0;
+               gbc.ipady = 0;
+               gbc.gridheight = 1;
+               gbc.gridwidth = 1;
+               gbc.fill = GridBagConstraints.NONE;
+               gbc.anchor = GridBagConstraints.CENTER;
+               gbc.insets = new Insets(0, 0, 0, 5);
+               gbc.weightx = 0.;
+               gbc.weighty = 0.;
+
+               add(chkPinning, gbc);
+
+               gbc.gridx = 1;
+               gbc.weightx = 0.;
+               gbc.insets = new Insets(0, 0, 0, 0);
+               add(chkGrayscale, gbc);
+
+               gbc.gridx = 2;
+               gbc.weightx = 0.;
+               gbc.insets = new Insets(0, 0, 0, 5);
+               add(chkNoAlpha, gbc);
+
+               gbc.gridx = 3;
+               gbc.weightx = 1.;
+               gbc.fill = GridBagConstraints.HORIZONTAL;
+               add(zoomSlider, gbc);
+
+               gbc.gridx = 4;
+               gbc.weightx = 0.;
+               gbc.insets = new Insets(3, 0, 3, 0);
+               gbc.fill = GridBagConstraints.VERTICAL;
+               add(zoomCombo, gbc);
+
+               Integer scrollbarWidth = (Integer) UIManager.get("ScrollBar.width");
+               logger.log(Level.CONFIG, "ScrollBar.width=" + scrollbarWidth);
+               if (scrollbarWidth == null) {
+                       scrollbarWidth = Integer.parseInt(
+                                       strings.getProperty("uiconstraint.scrollbar.width"));
+               }
+
+               gbc.gridx = 5;
+               gbc.weightx = 0.;
+               gbc.anchor = GridBagConstraints.WEST;
+               gbc.insets = new Insets(0, 0, 0, scrollbarWidth);
+               add(new JLabel("%"), gbc);
+       }
+
+       /**
+        * アプリケーション設定より事前定義済みの倍率候補を取得する
+        *
+        * @return 倍率候補のリスト(順序あり)
+        */
+       protected List<Integer> getPredefinedZoomRanges() {
+               AppConfig appConfig = AppConfig.getInstance();
+               String strs = appConfig.getPredefinedZoomRanges();
+               TreeSet<Integer> ranges = new TreeSet<Integer>();
+               for (String str : strs.split(",")) {
+                       str = str.trim();
+                       if (str.length() > 0) {
+                               try {
+                                       int zoomFactor = Integer.parseInt(str);
+                                       ranges.add(Integer.valueOf(zoomFactor));
+
+                               } catch (RuntimeException ex) {
+                                       // 無視する.
+                               }
+                       }
+               }
+               ranges.add(Integer.valueOf(100)); // 等倍は常に設定する.
+               return new ArrayList<Integer>(ranges);
+       }
+
+       /**
+        * 倍率が変更されたことを通知する.
+        */
+       protected void fireZoomFactorChange(int newZoomFactor) {
+               if (currentZoomFactorInt != newZoomFactor) {
+                       int oldValue = currentZoomFactorInt;
+                       currentZoomFactorInt = newZoomFactor;
+                       firePropertyChange("zoomFactorInt", oldValue, newZoomFactor);
+               }
+       }
+
+       private Semaphore changeChkLock = new Semaphore(1);
+
+       protected void onChangeCheckAlphaGray() {
+               changeChkLock.tryAcquire();
+               try {
+                       BackgroundColorMode backgroundColorMode = BackgroundColorMode.valueOf(
+                                       chkNoAlpha.isSelected(),
+                                       chkGrayscale.isSelected()
+                                       );
+                       setBackgroundColorMode(backgroundColorMode);
+
+               } finally {
+                       changeChkLock.release();
+               }
+       }
+
+       public BackgroundColorMode getBackgroundColorMode() {
+               return this.backgroundColorMode;
+       }
+
+       public void setBackgroundColorMode(BackgroundColorMode backgroundColorMode) {
+               if (backgroundColorMode == null) {
+                       throw new IllegalArgumentException();
+               }
+
+               BackgroundColorMode oldcm = this.backgroundColorMode;
+               if (oldcm != backgroundColorMode) {
+                       this.backgroundColorMode = backgroundColorMode;
+                       changeChkLock.tryAcquire();
+                       try {
+                               chkNoAlpha.setSelected(backgroundColorMode.isNoAlphaChannel());
+                               chkGrayscale.setSelected(backgroundColorMode.isGrayscale());
+
+                       } finally {
+                               changeChkLock.release();
+                       }
+                       firePropertyChange("backgroundColorMode", oldcm, backgroundColorMode);
+               }
+       }
+
+       public boolean isPinned() {
+               return chkPinning.isSelected();
+       }
+
+       public void setPinned(boolean pinned) {
+               chkPinning.setSelected(pinned);
+               if (isDisplayable()) {
+                       setVisible(pinned);
+               }
+       }
+
+       public double getZoomFactor() {
+               return (double) currentZoomFactorInt / 100.;
+       }
+
+       public void setZoomFactor(double zoomFactor) {
+               if (zoomFactor < minimumZoomFactor) {
+                       zoomFactor = minimumZoomFactor;
+               }
+               if (zoomFactor > maximumZoomFactor) {
+                       zoomFactor = maximumZoomFactor;
+               }
+               int zoomFactorInt = (int) round(zoomFactor * 100.);
+               if (zoomFactorInt != currentZoomFactorInt) {
+                       zoomCombo.setSelectedItem(Integer.toString(zoomFactorInt));
+               }
+       }
+}