之前老想着用一个Jtable上实现冻结列的效果,弄得非常复杂还出错。现在我用两个Jtable合并起来,第一个table作为冻结的列,第二个作为非冻结的列,这样就好办了。为了使外界看起来是一个Table,就需要改进如下几点:
1.ListSelected事件统一处理,选中某个表中任何一行,另外一个表的对应行也需要被选中。
2.Mouse事件,Key事件统一处理,repaint事件也统一处理。
3.其他事件均统一处理。
public class MergeTable {
private JTable actualTable;
private JTable fixTable;
private FixTableModel fixModel;
private ActualTableModel actualModel;
private int fixColumnCount;
private String[] columnHeader;
private JScrollPane mainScroll;
private Object[][] tableData;
public MergeTable(int fixColumnCount, String[] columnHeader,
Object[][] tableData) {
this.fixColumnCount = fixColumnCount;
this.columnHeader = columnHeader;
this.tableData = tableData;
this.initComponents();
}
private void initComponents() {
actualTable = new JTable();
fixTable = new JTable();
actualModel = new ActualTableModel();
fixModel = new FixTableModel();
actualTable.setModel(actualModel);
actualTable.setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
actualTable.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
fixTable.setModel(fixModel);
fixTable.setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
fixTable.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
actualTable.getSelectionModel().addListSelectionListener(
new ListSelectionListener() {
public void valueChanged(ListSelectionEvent e) {
checkSelection(false);
}
});
fixTable.getSelectionModel().addListSelectionListener(
new ListSelectionListener() {
public void valueChanged(ListSelectionEvent e) {
checkSelection(true);
}
});
actualTable.addMouseListener(new MouseAdapter() {
public void mouseClicked(MouseEvent e) {
if (e.getClickCount() > 1) {
if (e.getSource().equals(actualTable)) {
MouseListener[] mls = (MouseListener[]) fixTable
.getListeners(MouseListener.class);
for (int i = 0; i < mls.length; i++)
mls[i].mouseClicked(e);
}
}
}
});
fixTable.addMouseListener(new MouseAdapter() {
public void mouseClicked(MouseEvent e) {
if (e.getClickCount() > 1) {
if (e.getSource().equals(fixTable)) {
System.out.println("Click fix column");
} else {
System.out.println("Click actual column");
}
}
}
});
mainScroll = new JScrollPane();
mainScroll.getViewport().add(actualTable);
JViewport fixPort = new JViewport();
fixPort.setPreferredSize(fixTable.getPreferredSize());
fixPort.add(fixTable);
mainScroll.setRowHeaderView(fixPort);
mainScroll.setCorner(JScrollPane.UPPER_LEFT_CORNER, fixTable
.getTableHeader());
}
/**
* @param tableData
*/
public void setData(Object[][] tableData) {
this.tableData = tableData;
}
public JScrollPane getTableContentPane() {
return mainScroll;
}
/**
* @param isFixedTable
*/
private void checkSelection(boolean isFixedTable) {
int fixedSelectedIndex = fixTable.getSelectedRow();
int selectedIndex = actualTable.getSelectedRow();
if (fixedSelectedIndex != selectedIndex) {
if (isFixedTable) {
actualTable.setRowSelectionInterval(fixedSelectedIndex,
fixedSelectedIndex);
} else {
fixTable.setRowSelectionInterval(selectedIndex, selectedIndex);
}
}
}
private class FixTableModel extends AbstractTableModel {
/**
*
*/
private static final long serialVersionUID = 1L;
public int getColumnCount() {
return fixColumnCount;
}
public int getRowCount() {
return tableData.length;
}
public Object getValueAt(int rowIndex, int columnIndex) {
if (tableData[rowIndex][columnIndex] != null)
return tableData[rowIndex][columnIndex];
return "";
}
public String getColumnName(int column) {
return columnHeader[column];
}
}
private class ActualTableModel extends AbstractTableModel {
/**
*
*/
private static final long serialVersionUID = 1L;
public int getColumnCount() {
return columnHeader.length - fixColumnCount;
}
public int getRowCount() {
return tableData.length;
}
public Object getValueAt(int rowIndex, int columnIndex) {
if (tableData[rowIndex][columnIndex + fixColumnCount] != null)
return tableData[rowIndex][columnIndex + fixColumnCount];
return "";
}
public String getColumnName(int column) {
return columnHeader[column + fixColumnCount];
}
}
}
简单的测试程序
package com;
import java.awt.BorderLayout;
import java.awt.Dimension;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
public class MergeTableFrame extends JFrame {
/**
*
*/
private static final long serialVersionUID = 1L;
private MergeTable table;
/**
*
*/
public MergeTableFrame() {
super("MergeTable");
setPreferredSize(new Dimension(300, 200));
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
JPanel panel = new JPanel();
panel.setLayout(new BorderLayout());
getContentPane().add(panel, BorderLayout.CENTER);
String[] columns = new String[] { "Name", "Number", "English", "Chinese", "Maths", "Integration" };
Object[][] data = new Object[][] {
{ "Hank", "101", "84", "74", "65", "76" },
{ "Ryo", "102", "63", "77", "88", "39" },
{ "Ken", "103", "98", "36", "26", "98" },
{ "Luna", "104", "98", "76", "87", "66" },
{ "Mei", "105", "76", "46", "67", "45" },
{ "Mike", "106", "54", "68", "63", "44" } };
table = new MergeTable(1, columns, data);
panel.add(table.getTableContentPane());
}
public static void main(String args[]) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
MergeTableFrame frame = new MergeTableFrame();
frame.setVisible(true);
frame.pack();
}
});
}
}
分享到:
相关推荐
7. **数据格式控制**:调整字体、背景色、网格线颜色,设定行高、列宽,改变列顺序,隐藏或冻结列,以提升数据的可读性和美观性。 8. **筛选与排序**:通过多种方式筛选和排序记录,如选定内容筛选、排除筛选、按...
时间表详细列出每个任务的开始和结束日期,而里程碑是项目中的关键事件或阶段完成的标志,如需求冻结、初步设计完成、系统上线等。 #### 风险管理与变更控制 风险管理涉及到识别潜在的问题和威胁,制定应对策略,...
- **良好习惯**:保持工作表整洁,使用对齐和缩进,避免字符串内插入空格,使用公式引用数据,正确处理错误值,使用辅助列行,善用拆分窗口和冻结窗格,组合工作表,添加序号列等。 2. **常用函数** - **运算符及...
- 进行产业和市场分析,细化产品诉求,初步制定产品设计任务书和平台架构方案。 - 对标杆车和参考车进行分析,研究政策法规,造型预认可并确定单一造型方向。 3. **概念开发阶段 (G7)**: - 在概念开发阶段,...
在进行模糊查询时,我们通常会使用SQL语句的LIKE操作符,配合通配符(% 和 _)来实现。 默认情况下,Delphi的模糊查询可能会遇到性能问题,尤其是在处理大量数据时。例如,使用“%”通配符在索引字段前可能导致全表...
里程碑是项目中的重要事件或决策点,如需求冻结、初步设计审查、软件发布等,它们有助于监控项目的进度和控制质量。 三、资源管理 资源管理涉及人员、资金、设备和技术等的规划和分配。在软件项目中,合理配置人力...
例如,可能需要读取不同类型的单元格数据、实现自定义数据读取等功能。 ##### 2.1 单元格各类型数据读取 - **基本类型**:除了上面提到的字符串和数值类型外,还有布尔型、错误类型等。 - **日期类型**:对于日期...
- **环境要求**:确保伺服驱动单元安装在符合特定温湿度条件的环境中,使用温度范围为0~55℃,相对湿度不超过90%,并且无冻结现象;存储时温度范围为-20~65℃,相对湿度不超过90%。此外,还应注意避免腐蚀性、易燃...
通过本节的学习,读者可以初步掌握如何使用C#语言创建和运行一个简单的应用程序。 #### C#程序设计基础 **4.1 值类型** 值类型直接存储数据值,包括基本数据类型(如int、float等)和枚举类型。值类型存储在栈中...
- **项目工作分解结构**:对整个项目进行细致划分,明确每个阶段的目标和任务,如需求分析、系统设计、编码实现、测试部署等。 - **软件生命周期模型**:采用瀑布模型作为本次项目的主要开发模型,具体分为以下几个...
在软件开发中,里程碑可以是需求冻结、设计审查、代码审查、测试完成或者产品发布等。这些节点帮助团队跟踪进度,确保项目按预定的时间表进行。 在“Milestone1-for-Group1-main”这个压缩文件中,我们可以预期找到...
新品引进流程图是企业在引入新产品时所遵循的一种系统化步骤,旨在确保新产品的引入过程高效、规范,并最终实现商业目标。这份文档详细阐述了从产品概念到市场投放的全过程,涵盖了多个关键环节,对于理解与优化企业...
- **描述**:通过报表导入的方式,实现个人用户的批量开户。 - **操作**: - **下载模板**:下载指定格式的报表模板。 - **导入信息**:按照模板要求填写相关信息并导入系统。 - **导出报表**:将开户信息及...
它通过与其他模块如生产计划与控制(PP)及财务会计(FI)等的高度集成,实现了数据的一致性和业务流程的自动化。 **1.3 物料需求计划** 物料需求计划(Material Requirements Planning, MRP)是一种确保物料及时...