5

私は今学期を卒業するためにCapStoneプロジェクトに取り組んでおり、私が取り組んでいるプログラムでいくつかの問題に遭遇しました。このコードは、SerendipityBooksellersProgram用です。私は架空の書店のPOSシステムを設計することになっています。

現時点では、ユーザーが特定の行の「数量」セルを編集したときに、「合計」列(列5)を更新しようとして立ち往生しています。「数量」フィールドは、セル内の値が在庫量よりも大きくないことも検証する必要があります(列1)。

また、Javaチュートリアルの「テーブルの使用方法」を読み、昨夜Googleで何時間も検索して、CellEditorListenerを機能させる方法に関する役立つコードを見つけようとしました。

以前、CashierPanelクラスにCellEditorListenerを実装させて、次の方法で登録しようとしました。

CellEditorListener listener = null;
table.getDefaultEditor(String.class).addCellEditorListener(listener);

この方法も機能しませんでした。

これが私が今持っているコードです(217行):

/* Imported Dependencies */

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.util.ArrayList;

import javax.swing.DefaultCellEditor;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.JTextField;
import javax.swing.event.CellEditorListener;
import javax.swing.event.ChangeEvent;
import javax.swing.table.AbstractTableModel;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.TableColumn;

import net.miginfocom.swing.MigLayout;

public class CashierPanel extends JFrame {

    /**
    * Serial Version UID
    */
    private static final long serialVersionUID = 6922826100788686147L;

    /* Private Fields */
    private JPanel searchPanel;
    private JLabel isbn_lbl;
    private JLabel title_lbl;
    private JTextField isbn_tf;
    private JTextField title_tf;
    private JTextField currentCell;
    private BookCollection myCollection;
    private CashierTableModel tableData;

    private CellEditorListener listener;

    /**
    * Constructor
    */
    public CashierPanel() {

        // Set window properties
        setTitle("Testing this table");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setSize(585, 300);
        setLayout(new BorderLayout());

        tableData = new CashierTableModel();
        final JTable table = new JTable(tableData);
        table.setRowHeight(25);
        table.getTableHeader().setReorderingAllowed(false);
        table.setCellSelectionEnabled(true);
        table.putClientProperty("terminateEditOnFocusLost", Boolean.TRUE);
        add(new JScrollPane(table), BorderLayout.CENTER);

        // Define a cell renderer to render text to center of cell
        DefaultTableCellRenderer centerRenderer = new DefaultTableCellRenderer();
        centerRenderer.setHorizontalAlignment(JLabel.CENTER);

        // Set column properties
        int[] sizes = new int[] { 50, 50, 87, 275, 50, 50 };
        String[] columnNames = new String[] { "Qty", "Stock", "ISBN", "Title",
                "Price", "Total" };
        TableColumn[] column = new TableColumn[columnNames.length];
        for (byte i = 0; i < columnNames.length; i++) {
            column[i] = table.getColumnModel().getColumn(i);
            column[i].setMinWidth(sizes[i]);
            column[i].setHeaderValue(columnNames[i]);
            if (i != 3) {
                column[i].setMaxWidth(sizes[i]);
                column[i].setCellRenderer(centerRenderer);
            }
        }

        /* Make the quantity field only display digits */
        currentCell = new JTextField();
        column[0].setCellEditor(new DefaultCellEditor(currentCell));
        // Add a key listener to the current cell
        currentCell.addKeyListener(new KeyAdapter() {
            public void keyTyped(KeyEvent e) {
                // if the key typed is not a digit or backspace
                if (!Character.isDigit(e.getKeyChar())
                        && e.getKeyChar() != KeyEvent.VK_BACK_SPACE) {
                    currentCell.setEditable(false);
                    currentCell.setBackground(Color.WHITE);
                } else {
                    currentCell.setEditable(true);
                }
            }
        });

        /* Search panel */
        searchPanel = new JPanel();
        myCollection = new BookCollection();

        isbn_lbl = new JLabel("Search by ISBN:");
        isbn_tf = new JTextField();
        isbn_tf.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent event) {
                ArrayList<BookData> data = myCollection.select(isbn_tf.getText(), BookCollection.FIELD_ISBN);
                tableData.addNewRow();
                table.setValueAt((new BigDecimal("1").toString()), tableData.getRowCount() - 1, 0);
                table.setValueAt(Integer.toString(data.get(0).getQtyOnHand()), tableData.getRowCount() - 1, 1);
                table.setValueAt(data.get(0).getIsbn(),tableData.getRowCount() - 1, 2);
                table.setValueAt(data.get(0).getTitle(),tableData.getRowCount() - 1, 3);
                table.setValueAt(data.get(0).getRetail().setScale(2, RoundingMode.HALF_UP).toString(), tableData.getRowCount() - 1, 4);
                table.setValueAt(table.getValueAt(tableData.getRowCount() - 1, 4), tableData.getRowCount() - 1, 5);
                isbn_tf.setText("");
            }
        });

        title_lbl = new JLabel("Search by Title:");
        title_tf = new JTextField();
        title_tf.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent event) {
                ArrayList<BookData> data = myCollection.select(title_tf.getText(), BookCollection.FIELD_TITLE);
                tableData.addNewRow();
                table.setValueAt((new BigDecimal("1").toString()), tableData.getRowCount(), 0);
                table.setValueAt(Integer.toString(data.get(0).getQtyOnHand()), tableData.getRowCount() - 1, 1);
                table.setValueAt(data.get(0).getIsbn(), tableData.getRowCount(), 2);
                table.setValueAt(data.get(0).getTitle(), tableData.getRowCount(), 3);
                table.setValueAt(data.get(0).getRetail().setScale(2, RoundingMode.HALF_UP).toString(), tableData.getRowCount(), 4);
                table.setValueAt(data.get(0).getRetail().setScale(2, RoundingMode.HALF_UP).toString(), tableData.getRowCount(), 5);
                title_tf.setText("");
            }
        });

        searchPanel.setLayout(new MigLayout("", "[78px][202px,grow]", "[24px][29px]"));
        searchPanel.add(isbn_lbl, "cell 0 0, alignx left, aligny center");
        searchPanel.add(isbn_tf, "cell 1 0,growx, aligny center");
        searchPanel.add(title_lbl, "cell 0 1, alignx left, aligny center");
        searchPanel.add(title_tf, "cell 1 1, growx, aligny center");

        add(searchPanel, BorderLayout.SOUTH);

    };

    public static void main(String[] args) {
        CashierPanel myCashierPanel = new CashierPanel();
        myCashierPanel.setVisible(true);
    }
}

class CashierTableModel extends AbstractTableModel implements CellEditorListener{

    /**
     * 
     */
    private static final long serialVersionUID = -1500952132717353189L;
    private ArrayList<Object[]> data = new ArrayList<Object[]>();

    public void addNewRow() {
        data.add(new String[] { null, null, null, null, null, null });
        fireTableDataChanged();
    }

    public int getRowCount() {
        return data.size();
    }

    public int getColumnCount() {
        return 6;
    }

    public Object getValueAt(int row, int col) {
        return data.get(row)[col];
    }

    public void setValueAt(Object value, int row, int col) {
        data.get(row)[col] = value;
        fireTableCellUpdated(row, col);
    }

    public boolean isCellEditable(int row, int col) {
        if (col > 0) {
            return false;
        } else {
            return true;
        }
    }

    public void editingCanceled(ChangeEvent e) {
        for (int i = 0; i <= getRowCount() - 1; i++) {
            BigDecimal[] data = new BigDecimal[3];
            data[0] = data[0].add((BigDecimal) getValueAt(i, 1));
            data[1] = data[1].add((BigDecimal) getValueAt(i, 4));
            data[2] = data[2].add(calculateTotal(data[0], data[1]));
            setValueAt(data[2], i, 5);
        }
    }

    public void editingStopped(ChangeEvent e) {
        for (int i = 0; i <= getRowCount() - 1; i++) {
            BigDecimal[] data = new BigDecimal[3];
            data[0] = data[0].add((BigDecimal) getValueAt(i, 1));
            data[1] = data[1].add((BigDecimal) getValueAt(i, 4));
            data[2] = data[2].add(calculateTotal(data[0], data[1]));
            setValueAt(data[2], i, 5);
        }
    }

    public BigDecimal calculateTotal(BigDecimal qty, BigDecimal retail) {
        BigDecimal total = new BigDecimal("0.00");
        total = total.add(qty.multiply(retail));
        return total;
    }
}

これは宿題なので、バグが解決された状態で、コードの動作バージョンを渡されたくありません。エディターリスナーを実際に呼び出させるために私が間違っていることについて、いくつかのガイダンスが必要です。

私が解決策に導くのに役立つ他の情報があれば、私に知らせてください。

これを見てくれてありがとう!

4

2 に答える 2

5

セルエディタに明示的なキーリスナーを自分で追加して、それらのイベントを処理することは想定されていません。

デフォルトでは、列タイプに基づいて(モデルがセルを編集可能として返す場合)、jtableは適切なエディターを自動的に選択します。独自のカスタムエディタ(色付けなど)が必要な場合は、セルエディタとして実装し、列モデルに設定する必要があります。

値が変更されるたびに(フォーカスが失われるとトリガーされる可能性があります)、jtableにエディターをインストールすると、jtableはモデルのsetValueAt()を自動的に呼び出します。別の従属列を更新するすべてのロジックは、ここに属する必要があります。

于 2012-04-21T22:54:54.140 に答える
5

@AKJの有用な回答を拡張して、このではセルエディターの通常の操作について説明し、このはエディターに依存するデータを更新するための1つのアプローチを示しています。

于 2012-04-22T02:31:36.143 に答える