Сценарий: внутри области просмотра JScrollPane
у меня есть несколько JLayeredPane
. У каждого JLayeredPane
есть хотя бы один JPanel
с изображением (установленным paintComponent
).
Проблема. Трудно объяснить, не видя (код ниже): при прокрутке JScrollPane
изображения внутри JLayeredPane
, которые не полностью находятся внутри области JScrollPane
, не отображаются.
Если я продолжу прокрутку, в конечном итоге JLayeredPane
полностью окажется в области JScrollPane
, и изображения будут нарисованы.
Почему я думаю, что проблема в JLayeredPane
? Если я заменю JLayeredPane
на JPane
, проблема исчезнет. Предоставленный код может отображать оба случая. Установите первую и единственную статическую переменную открытого класса: public static boolean forceProblem = true
, чтобы управлять этим.
Вопрос. Что я делаю неправильно или что нужно сделать, чтобы решить проблему? Мне нужно продолжать использовать JLayeredPane
(или что-то еще, что может делать то же самое).
Воспроизведение проблемы:
Запустите код ниже.
Прокрутите вертикальную полосу до упора вниз.
Прокрутите горизонтальную полосу до упора вправо: проблема: верхняя строка изображений не загружена
Медленно прокручивайте вертикальную полосу вверх: изображения полностью загружаются в область прокрутки.
import java.awt.*;
import javax.swing.*;
class MYimagePanel extends JPanel {
public Image image;
public MYimagePanel( Image img ) {
this.image = img;
this.setLayout( null );
this.setBounds(0, 0, 1, 1);
}
@Override
public void paintComponent(Graphics g) {
super.paintComponent(g);
this.setSize( 100 , 100 );
this.setPreferredSize( new Dimension( 100 , 100 ));
g.drawImage( this.image , 0 , 0 , 100 , 100 , null );
}
}
class MYcomposedImagePanel extends JLayeredPane {
public MYcomposedImagePanel( Image img ) {
this.setLayout( null );
MYimagePanel myImgPane = new MYimagePanel( img );
this.add( myImgPane );
this.setLayer( myImgPane , 1 );
this.setBounds( 0, 0 , 100 , 100 );
//this.setPreferredSize( new Dimension( 100 , 100 ));
}
}
public class ClippingProblem extends JFrame {
public static boolean forceProblem = true;
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
// Creating Frame
JFrame frame = new ClippingProblem();
Container contentPane = frame.getContentPane();
contentPane.setLayout( null );
// ScrollPane viewport
JLayeredPane imagesPane = new JLayeredPane();
imagesPane.setLayout( null );
imagesPane.setLocation(0, 0);
imagesPane.setPreferredSize( new Dimension(2000,2000));
// ScrollPane
JScrollPane scrollPane = new JScrollPane( imagesPane );
scrollPane.setBounds(0, 0, 1000 , 700 );
scrollPane.getViewport().setScrollMode(JViewport.SIMPLE_SCROLL_MODE);
contentPane.add( scrollPane );
// Add Images
int offset = 0;
MYcomposedImagePanel composedImage;
MYimagePanel myImagePanel;
ImageIcon icon = new ImageIcon( "image.png" );
for( int y = 0 ; y < 1900 ; y = y + 100 ) {
for( int x = 0 ; x < 1900 ; x = x + 100 ) {
if( forceProblem == true ) {
composedImage = new MYcomposedImagePanel( icon.getImage() );
composedImage.setBounds( x + offset , y , 100 , 100 );
imagesPane.add( composedImage );
} else {
myImagePanel = new MYimagePanel( icon.getImage() );
myImagePanel.setBounds( x + offset , y , 100 , 100 );
imagesPane.add( myImagePanel );
}
offset += 10;
}
// Set visible
frame.setVisible(true);
}
} ) ;
}
public ClippingProblem() {
setSize(1024, 768);
setTitle("Clipping Problem");
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setLocationRelativeTo(null);
}
}
Использованное изображение: а>