JTable -> TableModeListener

У меня есть этот JTable с моделью DefaultTableModel. На столе у ​​меня есть несколько компонентов свинга, JComboBox и JCheckBox, установленных в определенном столбце через DefaultCellEditor и DefaultCellRenderer. TableModelListener был добавлен в таблицу для фиксации изменений в редактируемых столбцах. В остальных столбцах будут отображаться сведения о выбранном компоненте, т. е. код элемента -> цена элемента, количество элементов, классификация элементов и т. д.

У меня есть эта проблема, когда если выбранный элемент JComboBox(itemCode) изменяется, элементы другого JComboBox(itemClassification) меняются. Но вместе с изменением другого JComboBox мне нужно отображать цену товара в той же таблице. Это изменение перезапускает метод valueChanged, который создает бесконечный цикл valueChanged.

Как я могу избавиться от бесконечного цикла?


person Cyril Horad    schedule 09.04.2011    source источник


Ответы (3)


Один из способов — проверить событие обновления, чтобы увидеть, для какого столбца это событие, и игнорировать столбцы, которые автоматически обновляются:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.event.*;
import javax.swing.table.*;

public class TableProcessing extends JPanel implements TableModelListener
{
    public TableProcessing()
    {
        String[] columnNames = {"Item", "Quantity", "Price", "Cost"};
        Object[][] data =
        {
            {"Bread", new Integer(1), new Double(1.11), new Double(1.11)},
            {"Milk", new Integer(1), new Double(2.22), new Double(2.22)},
            {"Tea", new Integer(1), new Double(3.33), new Double(3.33)},
            {"Cofee", new Integer(1), new Double(4.44), new Double(4.44)}
        };

        DefaultTableModel model = new DefaultTableModel(data, columnNames)
        {
            //  Returning the Class of each column will allow different
            //  renderers to be used based on Class
            @Override
            public Class getColumnClass(int column)
            {
                return getValueAt(0, column).getClass();
            }

            //  The Cost is not editable
            @Override
            public boolean isCellEditable(int row, int column)
            {
                return (column == 3) ? false : true;
            }
        };
        model.addTableModelListener( this );

        JTable table = new JTable( model );
        table.setPreferredScrollableViewportSize(table.getPreferredSize());

        JScrollPane scrollPane = new JScrollPane( table );
        add( scrollPane );

        String[] items = { "Bread", "Milk", "Tea", "Coffee" };
        JComboBox<String> editor = new JComboBox<String>( items );

        DefaultCellEditor dce = new DefaultCellEditor( editor );
        table.getColumnModel().getColumn(0).setCellEditor(dce);
    }

    /*
     *  The cost is recalculated whenever the quantity or price is changed
     */
    public void tableChanged(TableModelEvent e)
    {
        if (e.getType() == TableModelEvent.UPDATE)
        {
            int row = e.getFirstRow();
            int column = e.getColumn();

            if (column == 1 || column == 2)
            {
                TableModel model = (TableModel)e.getSource();
                int quantity = ((Integer)model.getValueAt(row, 1)).intValue();
                double price = ((Double)model.getValueAt(row, 2)).doubleValue();
                Double value = new Double(quantity * price);
                model.setValueAt(value, row, 3);
            }
        }
    }

    private static void createAndShowGUI()
    {
        JFrame frame = new JFrame("Table Model Listener");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.add(new TableProcessing());
        frame.pack();
        frame.setLocationByPlatform( true );
        frame.setVisible( true );
    }

    public static void main(String[] args) throws Exception
    {
        EventQueue.invokeLater( () -> createAndShowGUI() );
/*
        EventQueue.invokeLater(new Runnable()
        {
            public void run()
            {
                createAndShowGUI();
            }
        });
*/
    }
}
person camickr    schedule 09.04.2011
comment
@Hovercraft, я никогда не знаю, какой подход лучше. Преимущество удаления/добавления прослушивателя заключается в том, что событие не запускается. Хорошо, когда есть несколько вариантов. - person camickr; 09.04.2011

Вы можете удалить прослушиватель при внесении изменений из самого прослушивателя, а затем повторно добавить его, когда закончите с изменениями. Или вы можете дать прослушивателю логическое свойство, которое активирует и деактивирует код прослушивателя, и установить его соответствующим образом внутри прослушивателя.

person Hovercraft Full Of Eels    schedule 09.04.2011
comment
+1, всегда хороший трюк, чтобы избежать бесконечных циклов внутри слушателей. - person camickr; 09.04.2011

На самом деле... У меня есть и свой ответ. Я просто захожу в API TableModelListener и нахожу TableModelEvent.getColumn()

Спасибо, парни...

person Cyril Horad    schedule 09.04.2011
comment
Я дал вам этот ответ ранее. Думаю, в следующий раз вам не понадобится моя помощь, так что не буду больше отвечать. - person camickr; 10.04.2011
comment
@camickr: Прошу прощения за это... Мне нужна ваша помощь, кто-нибудь поможет с любой проблемой, с которой я столкнулся. - person Cyril Horad; 11.04.2011