1

私はこの問題の解決策を設計しようとすることに圧倒されています。これは経験不足の副産物です。

私の目標は、XML 入力ファイルを読み取り、XML からの情報を保存し、2 つのコンボ ボックスに XML からのデータを入力することです。2 番目のコンボ ボックスの内容は、最初の選択に基づいて変化します。

この XML 構造を考えると、次のようになります。

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<Category xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<Node>
    <ID>Unique string</ID>
    <Name>Unique string</Name>
    <Code>Generic string<Code>
    <Kind>Generic string</Kind>
    <Frame>Generic string</Frame>
            ...
</Node>
    ...
</Category>

最初のコンボ ボックス: [種類] セクションで見つかった一意の値のみを含める必要があります。

2 番目のコンボ ボックス: Kind が最初のコンボ ボックスで選択された Kind と等しいすべてのノードのすべての Name エントリが含まれます。

XML ソースについて: 外部で維持および生成されます。ID セクションの値は常に一意になります。Name セクションの値は常に一意になります。スキーマは (おそらく) 変更されることはありません。今後、Kind セクションに新しい一意の値が表示される可能性があります。

私の提案する解決策: XML ソースからノードを表すクラス XMLNode を作成します。クラス XMLNode のメンバーは、各ノードのタグに対応します。すべてのノードをループし、ノードごとに XMLNode を作成します。ノードのループ中: Keys = XMLNode.ID および vals = XMLNode を使用して、ハッシュ マップに XMLNode オブジェクトを追加します。一意の種類の配列を作成します。

Kind エントリの配列からコンボ ボックス 1 を設定します。それぞれの Name データからコンボ ボックス 2 に入力します。

これは適切なアプローチですか、それともより優れた/より簡単な/よりエレガントなソリューションを見落としていますか? 私が正しい方向に進んでいる場合、提案されたソリューションに明らかな欠陥はありますか?

4

2 に答える 2

1

コード

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

public class ComboBoxTwo extends JFrame implements ActionListener, ItemListener {

    private static final long serialVersionUID = 1L;
    private JComboBox mainComboBox;
    private JComboBox subComboBox;
    private Hashtable<Object, Object> subItems = new Hashtable<Object, Object>();

    public ComboBoxTwo() {
        String[] items = {"Select Item", "Color", "Shape", "Fruit"};
        mainComboBox = new JComboBox(items);
        mainComboBox.addActionListener(this);
        mainComboBox.addItemListener(this);
        //prevent action events from being fired when the up/down arrow keys are used
        //mainComboBox.putClientProperty("JComboBox.isTableCellEditor", Boolean.TRUE);
        getContentPane().add(mainComboBox, BorderLayout.WEST);
        subComboBox = new JComboBox();//  Create sub combo box with multiple models
        subComboBox.setPrototypeDisplayValue("XXXXXXXXXX"); // JDK1.4
        subComboBox.addItemListener(this);
        getContentPane().add(subComboBox, BorderLayout.EAST);
        String[] subItems1 = {"Select Color", "Red", "Blue", "Green"};
        subItems.put(items[1], subItems1);
        String[] subItems2 = {"Select Shape", "Circle", "Square", "Triangle"};
        subItems.put(items[2], subItems2);
        String[] subItems3 = {"Select Fruit", "Apple", "Orange", "Banana"};
        subItems.put(items[3], subItems3);
//      mainComboBox.setSelectedIndex(1);
    }

    @Override
    public void actionPerformed(ActionEvent e) {
        String item = (String) mainComboBox.getSelectedItem();
        Object o = subItems.get(item);
        if (o == null) {
            subComboBox.setModel(new DefaultComboBoxModel());
        } else {
            subComboBox.setModel(new DefaultComboBoxModel((String[]) o));
        }
    }

    @Override
    public void itemStateChanged(ItemEvent e) {
        if (e.getStateChange() == ItemEvent.SELECTED) {
            if (e.getSource() == mainComboBox) {
                if (mainComboBox.getSelectedIndex() != 0) {
                    FirstDialog firstDialog = new FirstDialog(ComboBoxTwo.this,
                            mainComboBox.getSelectedItem().toString(), "Please wait,  Searching for ..... ");
                }
            } 
        }
    }

    private class FirstDialog extends JDialog {

        private static final long serialVersionUID = 1L;

        FirstDialog(final Frame parent, String winTitle, String msgString) {
            super(parent, winTitle);
            setModalityType(Dialog.ModalityType.APPLICATION_MODAL);
            JLabel myLabel = new JLabel(msgString);
            JButton bNext = new JButton("Stop Processes");
            add(myLabel, BorderLayout.CENTER);
            add(bNext, BorderLayout.SOUTH);
            bNext.addActionListener(new ActionListener() {

                @Override
                public void actionPerformed(ActionEvent evt) {
                    setVisible(false);
                }
            });
            javax.swing.Timer t = new javax.swing.Timer(1000, new ActionListener() {

                @Override
                public void actionPerformed(ActionEvent e) {
                    setVisible(false);
                }
            });
            t.setRepeats(false);
            t.start();
            setLocationRelativeTo(parent);
            setSize(new Dimension(400, 100));
            setVisible(true);
        }
    }

    public static void main(String[] args) {
        JFrame frame = new ComboBoxTwo();
        frame.setDefaultCloseOperation(EXIT_ON_CLOSE);
        frame.pack();
        frame.setLocationRelativeTo(null);
        frame.setVisible(true);
    }
}
于 2012-10-16T16:27:12.223 に答える
0

クラス NodeImporter とクラス Node になりました。Node はノードを表し、NodeImporter は XML を解析します。NodeImporter では、XML ソースが解析され、HashMap の HashMap として表されます。外側の HashMap (Key,Value) は (Kind,(HashMap(Key,Value)) です。内側の HashMap (Key,Value) は、(Kind,(UniqueID,Node)) の最終結果の (UinqueID,Node) です。最終結果を「filteredMap」と呼びます. NodeImporter のすべてのフィールドとメソッドはプライベートです, コンストラクターとfilteredMapのゲッターを除く. コンボボックスを構築するためにデータを必要とするクラスは、NodeImporterのインスタンスからfilteredMapを取得します.次に、外側の HashMap のキーを取得して、最初の ComboBoxModel を構築します.2 番目のコンボ ボックスの HashMap として使用する内側の HashMap を簡単に取得できます。

セットアップ用の擬似コード:

クラスメンバーで構築されたクラス NodeImporter:

arrayList(String) uniqueKinds = null  
arrayList(Node) allNodes  = null  
HashMap(String, HashMap(String,Node))) filteredNodes = null  

クラス NodeImporter コンストラクター:

open XML  
while xml source has next  
  {  
  add next node to allNodes, key = node.uniqueId, Val = node  
  if next node.kind not in uniqueKinds, add node.kind to uniqueKinds  
  }



ClassNodeImporter method makeFilteredeMap:  

    private boolean makeFilteredeMap() {
        if (uniqueKinds.isEmpty()) {
            return false;
        } else {
            for (String k : uniqueKinds) {
                HashMap<String, Node> aMap = new HashMap<String, Node>();
                for (Node n : allNodes) {
                    if (n.getKind().equals(k)) {
                        aMap.put(n.getCode(), n);
                    }
                }
                filteredNodes.put(k, aMap);
            }
            return true;
        }
    }
于 2012-10-25T15:41:28.140 に答える