Полоса прокрутки в 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...

Другие вопросы по тегам