如何创建自定义JTable?

时间:2014-06-15 09:21:42

标签: java jtable

我正在用Java创建一个聊天应用程序。他们可以添加朋友并与他们聊天。

以下是我的添加好友JFrame想法:

enter image description here

我尝试谷歌搜索Multi jpanel in one jscrollpane,但我什么也没找到。我最终得到了自定义JTable。我想创建一个自定义JTable,其中JLabel位于每个广告位的不同位置。用户只需在JTable中选择一个广告位,然后他们就可以使用下面的JButton与他们聊天。

是否可以在Java中执行此操作。是的,请分享您的想法。感谢。

2 个答案:

答案 0 :(得分:1)

这是一个提案。但它包含一些缺陷

  • 大小传播是固定的(可能是更好的基于客户端或服务器的大小调整)
  • 没有事件委托给基础Component
  • 性能,因为重量级面板实例经常在绘制循环中创建

但这是如何做到的。代码分为几部分。

值类

只是一个简单的类来表示每个单元格的面板内的日期。

class Value {
    public final String text;
    public final boolean flag;
    public Value(String text, boolean flag) {
        this.text = text;
        this.flag = flag;
    }
}

我的个人小组类

表示可以在谷歌的窗口构建器之类的gui编辑器中建模。此面板使用该值并相应地显示它。

public class MyPanel extends JPanel {
    public MyPanel(Value v) {
        JLabel lblNewLabel = new JLabel(v.text);
        add(lblNewLabel);
        JCheckBox chckbxSomeValue = new JCheckBox("some value");
        chckbxSomeValue.setSelected(v.flag);
        add(chckbxSomeValue);
    }
}

表格单元格渲染器类

只返回一些显示所需值的面板实例。

import javax.swing.JTable;
import javax.swing.table.TableCellRenderer;

class MyPanelCellRenderer implements TableCellRenderer {
    @Override
    public Component getTableCellRendererComponent(JTable table, Object value,
            boolean isSelected, boolean hasFocus, int row, int column) {
        return new MyPanel((Value)value); // maybe performance problem
    }   
}

自定义表格模型

import javax.swing.table.DefaultTableModel;

class MyTableModel extends DefaultTableModel {
    public MyTableModel() {
        super(new Object[][] {
                        new Object[] { 1, new Value("asdf", true) },
                        new Object[] { 2, new Value("qwer", false) } },
                        new String[] {"Id", "MyPanel" });
    }

    Class[] columnTypes = new Class[] { Integer.class, Value.class };

    MyTableModel(Object[][] data, Object[] columnNames) {
        super(data, columnNames);
    }

    public Class getColumnClass(int columnIndex) {
        return columnTypes[columnIndex];
    }
}

一个框架类

import java.awt.BorderLayout;

public class MyFrame {
    JFrame frame;
    private JTable table;

    public MyFrame() {
        frame = new JFrame();
        frame.setBounds(100, 100, 450, 300);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        table = new JTable();
        table.setModel(new MyTableModel());
        table.setFillsViewportHeight(true);
        table.getColumnModel()
            .getColumn(1)
            .setCellRenderer(new MyPanelCellRenderer());
        table.setRowHeight(40); // static sizing

        JScrollPane scrollPane = new JScrollPane();
        frame.getContentPane().add(scrollPane, BorderLayout.CENTER);
        scrollPane.setViewportView(table);
    }
}

主要功能

import java.awt.EventQueue;

public class MyApp {
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                try {
                    MyFrame window = new MyFrame();
                    window.frame.setVisible(true);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }
}

最终结果

The result

面板MyPanel是使用eclipse和google的窗口构建器

创建的

<code>MyPanel</code> creation

答案 1 :(得分:0)

这是一种完全不同的方法,既不使用表也不使用列表。它使用面板作为排列在另一个面板内的列表的项目。该解决方案不需要任何进一步的修改来将任何事件委托给底层控件。事件是本地处理的。这个想法来自this post。我现在把逻辑分成两部分。

List of <code>JPanels</code> arranged inside on a `JPanel smaller resized version

JPanelList类

此类是JPanel并维护此类的列表。 ATM可以使用方法addPaneladdPanels添加。

public class JPanelList extends JPanel {
    private static final long serialVersionUID = 1L;
    private JPanel mainList;
    private List<JPanel> panels = new ArrayList<JPanel>();
    public JPanelList() { this(new ArrayList<JPanel>()); }
    public JPanelList(List<JPanel> panels) {
        setLayout(new BorderLayout());
        mainList = new JPanel(new GridBagLayout());
        GridBagConstraints gbc = new GridBagConstraints();
        gbc.gridwidth = GridBagConstraints.REMAINDER;
        gbc.weightx = 1;
        gbc.weighty = 1;
        mainList.add(new JPanel(), gbc);
        add(new JScrollPane(mainList));
        addPanels(panels);
    }
    public void addPanels(List<JPanel> panels) {
        for (JPanel panel : panels)
            addPanel(panel);
    }
    public void addPanel(JPanel panel) {
        GridBagConstraints gbc = new GridBagConstraints();
        gbc.gridwidth = GridBagConstraints.REMAINDER;
        gbc.weightx = 1;
        gbc.fill = GridBagConstraints.HORIZONTAL;
        mainList.add(panel, gbc, 0);
        panels.add(panel);
        validate();
        repaint();
    }
}

具有主函数

的JListFrame测试类
public class JPanelListFrame {
    private JFrame frame;
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                try {
                    JPanelListFrame window = new JPanelListFrame();
                    window.frame.setVisible(true);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }
    public JPanelListFrame() {
        frame = new JFrame();
        frame.setBounds(100, 100, 210, 192);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        final JPanelList panelList = new JPanelList();
        frame.getContentPane().add(panelList, BorderLayout.CENTER);
        JButton btnAddMypanel = new JButton("Add MyPanel");
        btnAddMypanel.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent e) {
                // just add a MyPanel with a Value containing a
                // "asdf" and a random boolean
                panelList.addPanel(new MyPanel(new Value("asdf",
                        (int) (2 * Math.random()) % 2 == 0)));
            }
        });
        panelList.add(btnAddMypanel, BorderLayout.SOUTH);
    }
}

这种解决方案的一个缺点是选择机制丢失了。选择元素是关键要求时,JList可能更合适。另一件事是物品不会被渲染,因为它们将在JListJTable内呈现,例如使用花哨的边框。这可以通过在将面板添加到mainList的{​​{1}}之前以某种方式添加边框装饰器来解决。