Полоса прокрутки в JScrollPane не работает на JTables
У меня есть JScrollPane с количеством огромных JTables.
Пока я прокручиваю JScrollPane с помощью мыши, он просто останавливается, когда указатель мыши проходит JTable. (т.е. полоса прокрутки работает только в области B, но не в A.)
Как я могу избежать этого? Благодарю.
Вот код:
/*
* test.java
*
* Created on Jun 26, 2011, 7:39:43 PM
*/
package collector;
import javax.swing.JScrollPane;
/**
*
* @author Bhathiya
*/
public class test extends javax.swing.JFrame {
/** Creates new form test */
public test() {
initComponents();
jScrollPane2.setWheelScrollingEnabled( false );
jScrollPane3.setWheelScrollingEnabled( false );
}
/** This method is called from within the constructor to
* initialize the form.
* WARNING: Do NOT modify this code. The content of this method is
* always regenerated by the Form Editor.
*/
@SuppressWarnings("unchecked")
// <editor-fold defaultstate="collapsed" desc="Generated Code">
private void initComponents() {
jScrollPane1 = new javax.swing.JScrollPane();
jPanel1 = new javax.swing.JPanel();
jScrollPane2 = new javax.swing.JScrollPane();
jTable1 = new javax.swing.JTable();
jScrollPane3 = new javax.swing.JScrollPane();
jTable2 = new javax.swing.JTable();
setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
jScrollPane1.setHorizontalScrollBarPolicy(javax.swing.ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
jScrollPane2.setHorizontalScrollBarPolicy(javax.swing.ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
jScrollPane2.setVerticalScrollBarPolicy(javax.swing.ScrollPaneConstants.VERTICAL_SCROLLBAR_NEVER);
jTable1.setModel(new javax.swing.table.DefaultTableModel(
new Object [][] {
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null}
},
new String [] {
"Title 1", "Title 2", "Title 3", "Title 4"
}
));
jTable1.setAutoscrolls(false);
jScrollPane2.setViewportView(jTable1);
jScrollPane3.setHorizontalScrollBarPolicy(javax.swing.ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
jScrollPane3.setVerticalScrollBarPolicy(javax.swing.ScrollPaneConstants.VERTICAL_SCROLLBAR_NEVER);
jTable2.setModel(new javax.swing.table.DefaultTableModel(
new Object [][] {
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null},
{null, null, null, null}
},
new String [] {
"Title 1", "Title 2", "Title 3", "Title 4"
}
));
jTable2.setAutoscrolls(false);
jTable2.setCursor(new java.awt.Cursor(java.awt.Cursor.DEFAULT_CURSOR));
jScrollPane3.setViewportView(jTable2);
javax.swing.GroupLayout jPanel1Layout = new javax.swing.GroupLayout(jPanel1);
jPanel1.setLayout(jPanel1Layout);
jPanel1Layout.setHorizontalGroup(
jPanel1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(jPanel1Layout.createSequentialGroup()
.addGroup(jPanel1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.TRAILING)
.addComponent(jScrollPane3, javax.swing.GroupLayout.Alignment.LEADING, 0, 0, Short.MAX_VALUE)
.addComponent(jScrollPane2, javax.swing.GroupLayout.Alignment.LEADING, javax.swing.GroupLayout.PREFERRED_SIZE, 374, javax.swing.GroupLayout.PREFERRED_SIZE))
.addContainerGap(151, Short.MAX_VALUE))
);
jPanel1Layout.setVerticalGroup(
jPanel1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(jPanel1Layout.createSequentialGroup()
.addComponent(jScrollPane2, javax.swing.GroupLayout.PREFERRED_SIZE, 298, javax.swing.GroupLayout.PREFERRED_SIZE)
.addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED, 131, Short.MAX_VALUE)
.addComponent(jScrollPane3, javax.swing.GroupLayout.PREFERRED_SIZE, 168, javax.swing.GroupLayout.PREFERRED_SIZE)
.addGap(194, 194, 194))
);
jScrollPane1.setViewportView(jPanel1);
javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane());
getContentPane().setLayout(layout);
layout.setHorizontalGroup(
layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addComponent(jScrollPane1, javax.swing.GroupLayout.DEFAULT_SIZE, 544, Short.MAX_VALUE)
);
layout.setVerticalGroup(
layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addComponent(jScrollPane1, javax.swing.GroupLayout.DEFAULT_SIZE, 300, Short.MAX_VALUE)
);
pack();
}// </editor-fold>
/**
* @param args the command line arguments
*/
public static void main(String args[]) {
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
new test().setVisible(true);
}
});
}
// Variables declaration - do not modify
private javax.swing.JPanel jPanel1;
private javax.swing.JScrollPane jScrollPane1;
private javax.swing.JScrollPane jScrollPane2;
private javax.swing.JScrollPane jScrollPane3;
private javax.swing.JTable jTable1;
private javax.swing.JTable jTable2;
// End of variables declaration
}
3 ответа
Вероятно, это связано с тем, что каждая таблица имеет свой собственный JScrollPane (это по крайней мере, в случае вашей таблицы C). Когда мышь находится внутри таблицы, самая внутренняя панель прокрутки - это та, которая реагирует на колесо мыши. Я не думаю, что вы должны изменить это: это нормальное поведение.
ИЗДАНО:
Это некрасиво, но я не уверен, что есть другой способ сделать это. Я смог заставить ваш пример работать, удалив слушателя, добавленного на панель прокрутки с помощью его пользовательского интерфейса (даже если прокрутка колесика отключена):
for (MouseWheelListener listener : jScrollPane2.getMouseWheelListeners()) {
jScrollPane2.removeMouseWheelListener(listener);
}
// same for jScrollPane3
Это не элегантно, но наличие нескольких прокручиваемых таблиц внутри прокручиваемой панели тоже не подходит.
У меня есть JScrollPane с количеством огромных JTables.
Я предполагаю, что каждая из дочерних таблиц также добавляется в JScrollPane. Если это так, то событие прокрутки передается на эти панели прокрутки, а не на панель прокрутки. Так что вам нужно использовать:
scrollPane.setWheelScrollingEnabled( false );
на каждой из полос прокрутки, содержащихся в главной полосе прокрутки.
Обновить:
Проблема в том, что, хотя колесо прокрутки было отключено, область прокрутки перехватывает все события mouseScrolled, и они не передаются в родительскую панель прокрутки. Поэтому решение состоит в том, чтобы не добавлять таблицу в область прокрутки. Вместо этого используйте JPanel с BorderLayout. Что-то вроде:
import java.awt.*;
import java.util.*;
import javax.swing.*;
import javax.swing.table.*;
public class TableMultiple extends JPanel
{
public TableMultiple()
{
setLayout( new BorderLayout() );
Box box = Box.createVerticalBox();
box.add(Box.createVerticalStrut(50));
JPanel panel = new JPanel( new BorderLayout() );
JTable table = new JTable(10, 5);
panel.add(table.getTableHeader(), BorderLayout.NORTH);
panel.add(table);
box.add( panel );
box.add(Box.createVerticalStrut(50));
JTable table2 = new JTable(100, 5);
table2.setPreferredScrollableViewportSize(table.getPreferredSize());
JScrollPane scrollPane2 = new JScrollPane( table2 );
box.add( scrollPane2 );
box.add(Box.createVerticalStrut(50));
JTable table3 = new JTable(10, 5);
box.add( table3 );
box.add(Box.createVerticalStrut(50));
add( new JScrollPane(box) );
}
private static void createAndShowUI()
{
JFrame frame = new JFrame("TableMultiple");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add( new TableMultiple() );
frame.setSize(400, 500);
frame.setLocationRelativeTo( null );
frame.setVisible( true );
}
public static void main(String[] args)
{
EventQueue.invokeLater(new Runnable()
{
public void run()
{
createAndShowUI();
}
});
}
}
Я бы просто сделал это с помощью визуального редактора Eclipse. Проект VE только что умер. Мне нравится кодировать только важные вещи Swing, а не каждый JLabel...