JavaFX 表格组件详解及案例

1. 表格组件简介

  1. TableView:用于显示数据的二维表格,支持列排序、选择、编辑等功能。
  2. TreeTableView:类似于 TableView,但支持分层数据展示,适合树形结构的数据。

2. TableView 的常用方法

2.1 构造方法
  • TableView():创建一个空的表格视图。
  • TableView(ObservableList items):使用指定的初始数据创建表格。
2.2 列定义相关方法
  • TableColumn:定义表格的一列,其中 S 是表的模型类,T 是列的数据类型。

常见方法如下:

方法名说明示例
setCellValueFactory()设置列的值工厂,用于指定列数据的来源。nameColumn.setCellValueFactory(new PropertyValueFactory<>("name"));
setCellFactory()自定义单元格的渲染方式(如格式化日期)。priceColumn.setCellFactory(column -> new CurrencyCell());
setSortable(boolean value)设置列是否支持排序,默认值为 truenameColumn.setSortable(false);
setPrefWidth(double value)设置列的首选宽度。priceColumn.setPrefWidth(150);
2.3 表格数据操作方法
方法名说明示例
setItems(ObservableList items)设置表格的数据集合。tableView.setItems(dataList);
getItems()获取表格当前的数据集合。ObservableList items = tableView.getItems();
refresh()刷新表格以更新显示内容。tableView.refresh();
2.4 选择相关方法
方法名说明示例
getSelectionModel()获取表格的选择模型,用于处理选中行的操作。tableView.getSelectionModel().getSelectedItem();
setSelectionMode()设置选择模式(单选或多选)。tableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);

3. 常见问题及解决方案

3.1 列数据绑定问题

问题: 数据类的字段未使用 JavaFX 属性(Property),导致表格数据无法自动更新。

解决方法: 使用 SimpleStringProperty 等属性包装字段,并在数据类中提供 getter 和 setter 方法。

public class Person {private final SimpleStringProperty name;private final SimpleIntegerProperty age;public Person(String name, int age) {this.name = new SimpleStringProperty(name);this.age = new SimpleIntegerProperty(age);}public String getName() {return name.get();}public void setName(String name) {this.name.set(name);}public int getAge() {return age.get();}public void setAge(int age) {this.age.set(age);}
}

3.2 表格刷新问题

问题: 更新数据后表格未自动刷新。

解决方法: 调用 refresh() 方法强制刷新表格。

tableView.refresh();

3.3 自定义单元格样式

问题: 默认样式不满足需求,例如高亮特定数据行。

解决方法: 使用 setCellFactory 自定义渲染逻辑。

statusColumn.setCellFactory(column -> new TableCell<Person, String>() {@Overrideprotected void updateItem(String item, boolean empty) {super.updateItem(item, empty);if (item != null && !empty) {setText(item);setStyle(item.equals("Active") ? "-fx-background-color: lightgreen;" : "-fx-background-color: lightcoral;");} else {setText(null);setStyle("");}}
});

4. 完整案例

以下是一个完整的 TableView 案例,展示如何定义表格、绑定数据、自定义单元格渲染,以及处理常见问题。

4.1 数据模型
import javafx.beans.property.SimpleIntegerProperty;
import javafx.beans.property.SimpleStringProperty;public class Person {private final SimpleStringProperty name;private final SimpleIntegerProperty age;private final SimpleStringProperty status;public Person(String name, int age, String status) {this.name = new SimpleStringProperty(name);this.age = new SimpleIntegerProperty(age);this.status = new SimpleStringProperty(status);}public String getName() {return name.get();}public void setName(String name) {this.name.set(name);}public SimpleStringProperty nameProperty() {return name;}public int getAge() {return age.get();}public void setAge(int age) {this.age.set(age);}public SimpleIntegerProperty ageProperty() {return age;}public String getStatus() {return status.get();}public void setStatus(String status) {this.status.set(status);}public SimpleStringProperty statusProperty() {return status;}
}
4.2 主界面代码
import javafx.application.Application;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.control.*;
import javafx.scene.layout.HBox;
import javafx.scene.layout.VBox;
import javafx.stage.Modality;
import javafx.stage.Stage;import java.util.stream.Collectors;public class TableViewExample extends Application {private ObservableList<Person> masterData = FXCollections.observableArrayList(new Person("Alice", 30, "Active"),new Person("Bob", 25, "Inactive"),new Person("Charlie", 35, "Active"));private ObservableList<Person> filteredData = FXCollections.observableArrayList(masterData);@Overridepublic void start(Stage primaryStage) {// 创建 TableView 和列TableView<Person> tableView = new TableView<>();TableColumn<Person, String> nameColumn = new TableColumn<>("Name");nameColumn.setCellValueFactory(cellData -> cellData.getValue().nameProperty());TableColumn<Person, Integer> ageColumn = new TableColumn<>("Age");ageColumn.setCellValueFactory(cellData -> cellData.getValue().ageProperty().asObject());TableColumn<Person, String> statusColumn = new TableColumn<>("Status");statusColumn.setCellValueFactory(cellData -> cellData.getValue().statusProperty());tableView.getColumns().addAll(nameColumn, ageColumn, statusColumn);tableView.setItems(filteredData);// 搜索框TextField searchField = new TextField();searchField.setPromptText("Search...");searchField.textProperty().addListener((obs, oldText, newText) -> {filteredData.setAll(masterData.stream().filter(person -> person.getName().toLowerCase().contains(newText.toLowerCase())|| person.getStatus().toLowerCase().contains(newText.toLowerCase())).collect(Collectors.toList()));});// 添加按钮Button addButton = new Button("Add");addButton.setOnAction(e -> {masterData.add(new Person("New Person", 20, "Active"));filteredData.setAll(masterData);});// 删除按钮Button deleteButton = new Button("Delete");deleteButton.setOnAction(e -> {Person selectedPerson = tableView.getSelectionModel().getSelectedItem();if (selectedPerson != null) {masterData.remove(selectedPerson);filteredData.setAll(masterData);} else {showAlert(Alert.AlertType.WARNING, "No Selection", "Please select a row to delete.");}});// 修改按钮Button editButton = new Button("Edit");editButton.setOnAction(e -> {Person selectedPerson = tableView.getSelectionModel().getSelectedItem();if (selectedPerson != null) {// 弹出编辑窗口showEditDialog(primaryStage, selectedPerson);} else {showAlert(Alert.AlertType.WARNING, "No Selection", "Please select a row to edit.");}});// 布局HBox controls = new HBox(10, searchField, addButton, deleteButton, editButton);VBox root = new VBox(10, controls, tableView);Scene scene = new Scene(root, 600, 400);primaryStage.setTitle("JavaFX TableView Example");primaryStage.setScene(scene);primaryStage.show();}// 编辑对话框private void showEditDialog(Stage parentStage, Person selectedPerson) {// 新建一个编辑窗口Stage editDialog = new Stage();editDialog.initModality(Modality.WINDOW_MODAL);editDialog.initOwner(parentStage);editDialog.setTitle("Edit Person");// 创建输入框TextField nameField = new TextField(selectedPerson.getName());TextField statusField = new TextField(selectedPerson.getStatus());// 保存按钮Button saveButton = new Button("Save");saveButton.setOnAction(e -> {selectedPerson.setName(nameField.getText());selectedPerson.setStatus(statusField.getText());// 更新表格filteredData.setAll(masterData);editDialog.close();});// 布局VBox editLayout = new VBox(10, new Label("Name:"), nameField, new Label("Status:"), statusField, saveButton);Scene editScene = new Scene(editLayout, 300, 200);editDialog.setScene(editScene);editDialog.show();}// 显示警告框private void showAlert(Alert.AlertType alertType, String title, String message) {Alert alert = new Alert(alertType);alert.setTitle(title);alert.setHeaderText(null);alert.setContentText(message);alert.showAndWait();}public static void main(String[] args) {launch(args);}
}

5. 样式自定义

使用 JavaFX CSS 文件可以进一步定制表格的样式。

样式示例
.table-view .column-header-background {-fx-background-color: #2c3e50;
}.table-view .column-header .label {-fx-text-fill: white;
}.table-row-cell:odd {-fx-background-color: #f4f4f4;
}.table-row-cell:even {-fx-background-color: #ffffff;
}

在项目中添加 styles.css 并通过 scene.getStylesheets().add("styles.css"); 应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/887521.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

cocoscreater3.8.4生成图集并使用

1.安装texturepacker&#xff0c;去官网下载https://www.codeandweb.com/texturepacker 2.将图片拖动进来&#xff0c;即可自动生成精灵表&#xff0c;这里输出选用cocos2d-x&#xff0c;打包用免费版的“基本”就行&#xff0c;高级模式是收费的&#xff0c;然后点击“发布精…

解决SSL VPN客户端一直提示无法连接服务器的问题

近期服务器更新VPN后&#xff0c;我的win10电脑一致无法连接到VPN服务器&#xff0c; SSL VPN客户端总是提示无法连接到服务端。网上百度尝试了各种方法后&#xff0c;终于通过以下设置方式解决了问题&#xff1a; 1、首先&#xff0c;在控制面板中打开“网络和共享中心”窗口&…

从零开始学GeoServer源码(二)添加支持arcgis切片功能

文章目录 参考文章环境背景1、配置打包好的程序1.1、下载GeoServer的war包1.2、下载GeoWebCache1.3、拷贝jar包1.4、修改配置文件1.4.1、拷贝geowebcache-arcgiscache-context.xml1.4.2、修改geowebcache-core-context.xml1.4.3、修改geowebcache-servlet.xml 1.5、配置切片信息…

【Docker】Centos7 Jenkins 踩坑笔记

文章目录 1. docker pull 超时2. 初始化找不到 initialAdminPassword 1. docker pull 超时 docker pull 命令拉不下来 docker pull jenkins/jenkins:lts-jdk17 Error response from daemon: Get "https://registry-1.docker.io/v2/": 编辑docker配置 sudo mkdir -…

Java中的JSONObject详解

文章目录 Java中的JSONObject详解一、引言二、JSONObject的创建与基本操作1、创建JSONObject2、添加键值对3、获取值 三、JSONObject的高级特性1、遍历JSONObject2、从字符串创建JSONObject3、JSONObject与JSONArray的结合使用4、更新和删除键值对 四、错误处理1. 键值存在性检…

【大数据学习 | Spark-Core】Spark中的join原理

join是两个结果集之间的链接&#xff0c;需要进行数据的匹配。 演示一下join是否存在shuffle。 1. 如果两个rdd没有分区器&#xff0c;分区个数一致 &#xff0c;会发生shuffle。但分区数量不变。 scala> val arr Array(("zhangsan",300),("lisi",…

111.有效单词

class Solution {public boolean isValid(String word) {if(word.length()<3){return false;}int countV0,countC0;//分别统计原音和辅音for(int i0;i<word.length();i){if(Character.isLetterOrDigit(word.charAt(i))){if(word.charAt(i)a||word.charAt(i)e||word.charA…

安装python拓展库pyquery相关问题

我采用的是离线whl文件安装, 从官方库地址: https://pypi.org/, 下载whl文件, 然后在本地电脑上执行pip install whl路径文件名.whl 但是在运行时报错如下图 大体看了看, 先是说了说找到了合适的 lxml>2.1, 在我的python库路径中, 然后我去看了看我的lxml版本, 是4.8.0, 对…

如何启动多个libvirtd进程

导语:如何启动多个libvirtd,咋一想这不简单,多运行几个libvirtd不就完事,其实不然?为什么启动多个libvirtd,有何应用场景?当前libvirt代码架构是否支持启动多个libvirtd? 一、如何启动libvirtd # /usr/local/sbin/libvirtd --listen --listen 必须参数,监控tcp/ip c…

架构-微服务-服务网关

文章目录 前言一、网关介绍1. 什么是API网关2. 核心功能特性3. 解决方案 二、Gateway简介三、Gateway快速入门1. 基础版2. 增强版3. 简写版 四、Gateway核心架构1. 基本概念2. 执行流程 五、Gateway断言1. 内置路由断言工厂2. 自定义路由断言工厂 六、过滤器1. 基本概念2. 局部…

idea怎么打开两个窗口,运行两个项目

今天在开发项目的时候&#xff0c;前端希望运行一下以前的项目&#xff0c;于是就需要开两个 idea 窗口&#xff0c;运行两个项目 这里记录一下如何设置&#xff1a;首先依次点击&#xff1a; File -> Settings -> Appearance & Behavior ->System Settings 看到如…

零碎04 MybatisPlus自定义模版生成代码

目录 背景 动手开干 需要的依赖包&#xff0c;需要注意mybatis-plus-generator的3.5版本是没有兼容历史版本的。 定义一个CodeGenerator类&#xff0c;负责生成代码和配置属性 Entity模版 背景 MybatisPlus代码生成使用默认的velocity模版解决不了定制化的需求&#xff0…

CentOS8.5.2111(7)完整的Apache综合实验

一、实验目标 1.掌握Linux系统中Apache服务器的安装与配置&#xff1b; 2.掌握个人主页、虚拟目录、基于用户和主机的访问控制及虚拟主机的实现方法。 二、实验要求 练习使用linux系统下WEB服务器的配置方法。 三、实验背景 重庆工程学院为筹备“重庆工程大学”特申请了c…

Cocos编辑器

1、下载 下载地址&#xff1a;https://www.cocos.com/creator-download 2、编辑器界面介绍 官方链接&#xff1a;https://docs.cocos.com/creator/3.8/manual/zh/editor/ 3、项目结构 官方链接&#xff1a;https://docs.cocos.com/creator/3.8/manual/zh/getting-started/…

Easyui 实现订单拆分开票功能

Easyui 实现订单拆分开票功能 需求 1、实现一个订单开具多分发票功能&#xff1b; 2、支持拆行&#xff1b; 3、支持拆数量&#xff1b; 流程设计 1、操作页面展示订订单头信息&#xff0c;订单明细信息 2、点击新增发票按钮弹出一个弹出框用于创建一张拆分发票&#xff0c;弹…

解决 java -jar 报错:xxx.jar 中没有主清单属性

问题复现 在使用 java -jar xxx.jar 命令运行 Java 应用程序时&#xff0c;遇到了以下错误&#xff1a; xxx.jar 中没有主清单属性这个错误表示 JAR 文件缺少必要的启动信息&#xff0c;Java 虚拟机无法找到应用程序的入口点。本文将介绍该错误的原因以及如何通过修改 pom.xm…

推荐一款龙迅HDMI2.0转LVDS芯片 LT6211UX LT6211UXC

龙迅的HDMI2.0转LVDS芯片LT6211UX和LT6211UXC是两款高性能的转换器芯片&#xff0c;它们在功能和应用上有所差异&#xff0c;同时也存在一些共同点。以下是对这两款芯片的详细比较和分析&#xff1a; 一、LT6211UX 主要特性&#xff1a; HDMI2.0至LVDS和MIPI转换器。HDMI2.0输…

flink学习(7)——window

概述 窗口的长度(大小): 决定了要计算最近多长时间的数据 窗口的间隔: 决定了每隔多久计算一次 举例&#xff1a;每隔10min,计算最近24h的热搜词&#xff0c;24小时是长度&#xff0c;每隔10分钟是间隔。 窗口的分类 1、根据window前是否调用keyBy分为键控窗口和非键控窗口…

C语言解析命令行参数

原文地址&#xff1a;C语言解析命令行参数 – 无敌牛 欢迎参观我的个人博客&#xff1a;无敌牛 – 技术/著作/典籍/分享等 C语言有一个 getopt 函数&#xff0c;可以对命令行进行解析&#xff0c;下面给出一个示例&#xff0c;用的时候可以直接copy过去修改&#xff0c;很方便…

精密工装夹具加工:打造高精度产品

在现代制造业中&#xff0c;精密工装夹具加工扮演着关键角色&#xff0c;是打造高精度产品不可缺少的环节。 精密工装夹具的设计与制造&#xff0c;首先依赖于对加工工艺的深入理解与精准把握。工程师们需要根据待加工产品的形状、尺寸、精度要求以及加工设备的特性&#xff0c…