从ResultSet创建动态GUI - Java

时间:2014-02-03 19:35:26

标签: java mysql swing resultset

我正在尝试通过从结果集中获取值并使用它来生成核对表来动态创建Java GUI。我已经创建了一个小型演示程序来演示我所做的:

SQL命令

CREATE USER 'test'@'localhost' IDENTIFIED BY 'testpw';
CREATE DATABASE combotest;
USE combotest;

CREATE TABLE combotable (
id INT(5) NOT NULL PRIMARY KEY auto_increment,
type VARCHAR(50) NOT NULL);

INSERT INTO combotable (id, type) VALUES
(default, 'Label'),
(default, 'Textfield'),
(default, 'Combo'),
(default, 'Label'),
(default, 'Textfield'),
(default, 'Combo'),
(default, 'Combo');

GRANT SELECT ON combotest.* TO 'test'@'localhost';

为了您的方便,如果您想自己测试一下,我已经将所有SQL命令放在上面。

现在,对于我的Java代码:

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.sql.*;  
import javax.swing.*;

public class resToComboDemo implements ActionListener {

//JDBC Variables
static Connection connect = null;
static Statement statement = null;
static ResultSet res = null;

@SuppressWarnings("rawtypes")
//Other Variables
JComboBox comboBox;
JButton submit;
JFrame frame;
JLabel label;
JTextField textField;
Container pane;

public static void main(String[] args) throws SQLException {
    new resToComboDemo();
}

public resToComboDemo() throws SQLException {
    try {
        Class.forName("com.mysql.jdbc.Driver");
        // Setup the connection with the DB

        connect = DriverManager
                .getConnection("jdbc:mysql://localhost/combotest?"
                        + "user=test&password=testpw");

        statement = connect.createStatement();
        //Note: in this specific case I do realize that "order by id" is not necessary. I want it there, though.
        res = statement.executeQuery("SELECT * FROM combotable ORDER BY id");

        createStuff(res);

    } catch (Exception e) {
        JOptionPane.showMessageDialog(null, "Error 1: "+e, "Error!", JOptionPane.ERROR_MESSAGE);
    } finally {
        connect.close();
    }

}

@SuppressWarnings({"rawtypes", "unchecked" })
public void createStuff (ResultSet res) throws SQLException {

    frame = new JFrame("Testing dynamic gui");
    Dimension sD = Toolkit.getDefaultToolkit().getScreenSize();
    int width = sD.width;
    int height = sD.height - 45;
    frame.setSize(width,height);

    pane = frame.getContentPane();

    pane.setLayout(new GridLayout(0, 2));

    while (res.next()) {
        Object[] options = { "Pass", "Fail"};
        String type = res.getString("type");

        JLabel label = new JLabel("<html><small>"+type+"</small></html>");
        JLabel blank = new JLabel(" ");
        blank.setBackground(Color.black);
        blank.setOpaque(true);

        if (type.equals("Label")) {
            label.setBackground(Color.black);
            label.setForeground(Color.white);
            label.setOpaque(true);
            pane.add(label);
            pane.add(blank);

        } else if (type.equals("Combo")) {
            pane.add(label);
            comboBox = new JComboBox(options);
            pane.add(comboBox);

        } else if (type.equals("Textfield")) {
            pane.add(label);
            textField = new JTextField(20);
            pane.add(textField);

        }   
    }

     JLabel blank2 = new JLabel(" ");
     pane.add(blank2);

     submit = new JButton("Submit");
     submit.addActionListener(this);
     pane.add(submit);

     frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
     frame.setVisible(true);

}

@Override
public void actionPerformed(ActionEvent e) {

}
}

现在,在这里创建GUI一切都很顺利。但是,我需要能够将Combobox和Textfield组件视为自己独立的实体。意思是,我希望能够从每个不同的组件获得用户输入。现在,如果我要从文本字段请求信息,它只是给我从上一个文本字段的信息。这是完美的,因为这是java读取它的方式。我没问题。

我不能为我的生活弄清楚如何分别获得每个组件的输入。也许通过获取结果集并将结果添加到某种类型的数组?我尝试过多次不同的口味,但我不能按照我需要的方式来实现。你们当中有些人会要求我告诉你我尝试过的东西......老实说,这不值得。

并且,在任何人问之前:不,我不会使用FlowLayout。 :)

非常感谢任何帮助!

3 个答案:

答案 0 :(得分:1)

根据您的目标,可能有几种方法可以实现这一目标......

如果您只是执行批量更新,则可以使用Map键入行的id并映射到Component

这样,当您想要将值保存回数据库时,您只需迭代Map个键值,提取与每个键关联的Component,然后提取值。 Component ...

我可能会考虑创建一个包含interface的包装器,它有一个简单的getText方法并将组件包装在其中,使得包装器的实现负责提取文本,但这只是我;)< / p>

如果要在更新单个组件时执行更新,则需要交换映射,以便Component将密钥和id映射到它。

这意味着当某种事件发生时会触发并更新(即ActionEvent),您可以从事件中提取source并查找id in Map基于导致事件的Component ...

现在......坦率地说,我只想使用JTable并创建一个自定义TableModel,可以对所有这些进行建模。

这将要求您创建表的POJO,在单个对象中维护idtypevalue。这将在表中定义一个基本行。

唯一的问题是你需要创建一个(合理的)复杂TableCellEditor,它可以取type并为表返回一个合适的编辑器。并非不可能,它只是表格的正常使用之外的额外复杂性。

这样您所需的所有信息都可以在表格中单行的单个对象中使用。

请查看How to use tables了解详情

同样,您可以使用与上述Map提示类似的想法......

您还可以简单地创建一个自包含的“编辑器”(从JPanel扩展而来),该编辑器维护有关idtype的信息,您可以从中提取值并简单地保留这些的列表....例如......

答案 1 :(得分:0)

您只能引用您创建的最后一个文本字段或组合框,因为您正在重用包含它们的变量。我会将它们放在一个ArrayList中,在创建它们时存储每个新文本字段和combbox,然后你可以在完成后返回并从所有文件中获取输入。

----------(在OP对上段的回应之后)

不,没有“推荐你的地方” - 这是你的一套要求,找到已经存在的代码就可以了。 Java和Swing为您提供了工具,您需要自己将它们放在一起。

你没有显示你的“actionPerformed”例程,但让我们假设它一分钟。当一个动作完成时,框架会调用它,并传递一个“ActionEvent”对象。通过它的方法,我们发现它有“getSource()”,因此它将为您提供对生成事件的组件的引用。

让我们进一步思考我们拥有什么 - UI中的一组组件,以及可以生成事件的组件对我们来说很有趣。在这种情况下,我们希望从生成事件的组件中检索某些内容。

如果我们有组件(来自actionEvent.getSource())并且我们想对它做一些事情,那么我们最好在actionPerformed()方法中执行类似下面的操作:

Component sourceComponent = actionEvent.getSource();
if (sourceComponent instanceof JComboBox)
  { JComboBox sourceBox = (JComboBox) sourceComponent;
    // get the value from the combo box here
  }
else if (sourceComponent instanceof JTextField)
  { JTextField sourceTextField = (JTextField) sourceComponent;
    // get the value from the text field here
  }
// or else do nothing -- our action was not one of these.

通过这种方式,您甚至不需要保留组件列表 - UI保留对所有组件的引用,并且您只需在actionEvent发生时使用该引用。

现在,这不是唯一的,甚至是最好或最简单的方法。如果你想用自己的类扩展JComboBox和JTextField,你可以让这些类都实现一个定义getValue()getText之类的接口;那么你就不需要丑陋的instance of运算符,通常可以通过更好的设计和规划来消除它。

答案 2 :(得分:0)

询问包含组件的Container(窗格)怎么样? getComponents()方法并循环遍历子组件并检查JComobox和JTextField执行所需的强制转换并检索值

如果您反对将子组件添加到一种列表中

,那么这只是一个想法