JavaFx TableView疑难详解
TableView是个十分有用的控件,适应性和灵活性非常强,可以对它进行任意的修改,比如界面样式、功能。本文将从一步步提问的方式讲解TableView
欢迎加入我的javafx探讨群:518914410
本文版权原创 by cmlanche.com, 文章链接:http://cmlanche.com/2017/06/08/JavaFx-TableView详解/
创建已知列的TableView
已知列的表格的创建,需要把TableView的TableColumn关联到模型的属性,TableView是个模板类,其实是TableView,这个T就是模型,例如下代码:
// MyModel.java
public class MyModel{
private String name;
private String url;
// getters, setters
...
}
// init your tableView
TableColumn<MyModel, String> t1 = new TableColumn();
// 关联MyModel中的name属性
t1.setCellValueFactory(new PropertyValueFactory<>("name"));
t1.setCellFactory(p->{
// 创建此列的Cell的时候的回调,允许让你自己去创建
});
特别要说明的是,
setCellValueFactory和setCellFactory不是冲突的,我用的时候一直以为是冲突,就是只能用其中一个,另外一个就失效了,其实不是,setCellFactory它的意图是在创建这列的时候要做的事情,你可以改变TableCell的任何内容,包括UI和Value,而setCellValueFactory呢,它的重点是关联属性,从你传递给它的Model中通过对应属性的getter来获取值。在setCellFactory中的TableCell有个回调,叫updateItem,它可以获取到你设置到此Cell的值,这个值是跟setCellValueFactory所关联的属性有关。创建动态列的TableView
参考:https://community.oracle.com/thread/2474328
因为列是不定的,模型是没有属性对应的,创建列的时候你根本不知道列是什么,看如下实现代码:
column.setCellValueFactory(param -> {
ObservableList<VarCell> values = param.getValue();
if (columnIndex >= values.size()) {
return new SimpleObjectProperty<>(null);
} else {
return new SimpleObjectProperty<>(param.getValue().get(columnIndex));
}
});
创建动态列的tableview,它的模型是一个
ObservableList<T>,你的setCellValueFactory不能使用PropertyValueFactory,而是如上代码所示,通过列的索引来获取此列的值[列拖动] 如何捕获列拖动事件?
列拖动是tablview一个默认的自带的效果,但是并没有专门的事件给你去监听它,而是监听列的变化,方法:给tableview的columns添加Listener,判断变动列的状态是否是replaced的状态,例如:
tableView.getColumns().addListener(new ListChangeListener<TableColumn<ObservableList<VarCell>, ?>>() {
@Override
public void onChanged(Change<? extends TableColumn<ObservableList<VarCell>, ?>> change) {
change.next();
if (change.wasReplaced()) {
// 表示当前拖动过了
}
}
});
[列拖动] 如何防止第一列被拖动?
在上一问的基础上,实现第一列不允许被拖动的功能。
参考:https://stackoverflow.com/questions/30645606/javafx-restrict-column-rearrangement-on-drag-and-drop
tableView.getColumns().addListener(new ListChangeListener<TableColumn<ObservableList<String>, ?>>() {
private boolean suspended; @Override
public void onChanged(Change<? extends TableColumn<ObservableList<String>, ?>> change) {
change.next(); if (change.wasReplaced() && !suspended) {
List<TableColumn<ObservableList<String>, ?>> oldList = new ArrayList<>(change.getRemoved());
List<TableColumn<ObservableList<String>, ?>> newList = new ArrayList<>(tableView.getColumns()); // first column changed => revert to original list
if (oldList.get(0) != newList.get(0)) {
this.suspended = true;
tableView.getColumns().setAll(oldList);
this.suspended = false;
}
}
}
});
上面的代码中有三个关键的地方,是tableview原本提供的api,一个是
change.wasReplaced表示当前的变动是否被替换了,第二个是change.getRemoved,表示获取要移除掉的列,进一步的意思就是原来的列,也就是此前的tablecolumns,第三个是tableview.getColumns这个是获取现在列,有了这些信息,就可以判断,oldList.get(0)!=newList(0),表示如果新老列的第一列不相同,表示是第一列是变动的,但是我们不允许变动,因此,调用tableview.getColumns().setAll(oldList)用来恢复原来的列。这样就禁止拖动第一列了。[列拖动] 如何禁用列拖动效果?
给列设置一个属性:
column.impl_setReorderable(false);impl_setReorderable前面带impl_前缀,表示它是一个将来可能会被删除的方法,但是为了解决目前无法解决的问题,暂时把impl的私有方法改为了public方法,参考我的博客中的如何自定义Taborder的文章,是一样的道理。[行拖动] 如何拖动行,进行换行?
参考:https://stackoverflow.com/questions/28603224/sort-tableview-with-drag-and-drop-rows
已经测试过的代码:
tableView.setRowFactory(tv -> {
TableRow<ObservableList<String>> row = new TableRow<>(); row.setOnDragDetected(event -> {
log.info("row drag detected");
if (!row.isEmpty()) {
Integer index = row.getIndex();
Dragboard db = row.startDragAndDrop(TransferMode.MOVE);
db.setDragView(row.snapshot(null, null));
ClipboardContent cc = new ClipboardContent();
cc.put(SERIALIZED_MIME_TYPE, index);
db.setContent(cc);
event.consume();
}
}); row.setOnDragOver(event -> {
log.info("row drag over");
Dragboard db = event.getDragboard();
if (db.hasContent(SERIALIZED_MIME_TYPE)) {
if (row.getIndex() != ((Integer) db.getContent(SERIALIZED_MIME_TYPE)).intValue()) {
event.acceptTransferModes(TransferMode.COPY_OR_MOVE);
event.consume();
}
}
}); row.setOnDragDropped(event -> {
log.info("row drag dropped");
Dragboard db = event.getDragboard();
if (db.hasContent(SERIALIZED_MIME_TYPE)) {
int draggedIndex = (Integer) db.getContent(SERIALIZED_MIME_TYPE);
ObservableList<String> draggedPerson = tableView.getItems().remove(draggedIndex); int dropIndex; if (row.isEmpty()) {
dropIndex = tableView.getItems().size();
} else {
dropIndex = row.getIndex();
} tableView.getItems().add(dropIndex, draggedPerson); event.setDropCompleted(true);
tableView.getSelectionModel().select(dropIndex);
event.consume();
}
}); return row;
});
修改TableView样式
使用css,参考如下我的测试代码
.table-view {
-fx-border-width: 1px;
-fx-border-color: #CACACA;
-fx-background-color: transparent;
} .table-view:focused {
-fx-background-color: transparent;
} .table-view .table-cell {
-fx-font-size: 12px;
} .table-view .filler {
-fx-background-color: #BDE8FF;
} .table-view .text {
-fx-text-fill: red;
} .table-view .column-header {
-fx-background-color: #BDE8FF;
-fx-pref-height: 37px;
-fx-border-width: 1px;
-fx-border-color: #D9D9D9;
-fx-border-insets: -2px -2px 0px -2px;
} .table-view .column-header-background .label {
-fx-text-fill: #363739;
-fx-font-weight: normal;
-fx-font-size: 12px;
} .table-row-cell {
/*行高*/
-fx-cell-size: 35px;
} .table-row-cell .cell {
-fx-alignment: center;
-fx-text-fill: #333333;
} .table-view .table-cell:selected {
-fx-text-fill: white;
} .table-view .table-column .column-header {
-fx-background-color: #363739;
} .cell {
/*-fx-border-width: 0px 1px 0 0;*/
/*-fx-border-color: #CACACA;*/
} .table-view .scroll-bar {
-fx-background-color: transparent;
} .viewport {
-fx-background-color: white;
}
如何自定义列头,比如自己设置一个可编辑的列头呢?
答案是给你的TableColumn设置
setGraphic,可编辑的列头的话,你让你的graphic中有编辑框,双击显示编辑框,按enter键确认编辑,如下是一个我的实现:package com.itestin.ui.datamgt.table; import com.itestin.ui.recordNreplay.logic.CommonLogic;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.SimpleBooleanProperty;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.geometry.Pos;
import javafx.scene.control.Label;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TextField;
import javafx.scene.control.Tooltip;
import javafx.scene.input.KeyCode;
import javafx.scene.layout.StackPane; /**
* Created by cmlanche on 2017/6/1.
*/
public class EditColumn extends StackPane { private EditColumnCallback callback; private Label label;
private TextField textField; private StringProperty title;
private BooleanProperty editing;
private BooleanProperty editable;
private BooleanProperty textFieldFocus;
private TableColumn tableColumn; private String oldTitle; public EditColumn(TableColumn tableColumn) {
super();
this.tableColumn = tableColumn;
this.init();
} private void init() {
this.setMaxWidth(120);
this.setAlignment(Pos.CENTER);
this.setStyle("-fx-background-color: #D9D9D9;");
label = new Label();
label.setStyle("-fx-font-size: 12px; -fx-text-fill: #333333;");
textField = new TextField();
textField.getStyleClass().add("tablefx-header-editor");
label.textProperty().bindBidirectional(titleProperty());
textField.textProperty().bindBidirectional(titleProperty());
label.setTooltip(new Tooltip());
label.getTooltip().textProperty().bindBidirectional(label.textProperty());
this.getChildren().addAll(label, textField);
editingProperty().addListener((observable, oldValue, newValue) -> {
if (isEditable()) {
if (newValue) {
label.setVisible(false);
textField.setVisible(true);
textField.setFocusTraversable(true);
textField.requestFocus();
} else {
label.setVisible(true);
textField.setVisible(false);
}
}
});
textField.textProperty().addListener((observable, oldValue, newValue) -> {
textField.setStyle("-fx-border-color: #25B3FA;");
});
this.setStyle("-fx-background-color: transparent;");
textField.setOnKeyPressed(event -> {
if (isEditing()) {
if (event.getCode() == KeyCode.ENTER) {
event.consume(); // 放置对tabview的其他列产生影响,不让消息透传
oldTitle = textField.getText();
setEditing(false);
// 提交编辑
if (callback != null) {
callback.editCommit(tableColumn, textField.getText());
}
} else if (event.getCode() == KeyCode.ESCAPE) {
setTitle(oldTitle);
setEditing(false);
// 取消编辑
if (callback != null) {
callback.cancelEdit();
}
} else if (event.getCode() == KeyCode.TAB) {
setEditing(false);
}
}
});
textFieldFocusProperty().addListener((observable, oldValue, newValue) -> {
if (newValue) {
textField.setFocusTraversable(true);
textField.requestFocus();
}
});
setEditing(false);
} public String getTitle() {
return titleProperty().get();
} public StringProperty titleProperty() {
if (title == null) {
title = new SimpleStringProperty();
}
return title;
} public void setTitle(String title) {
this.oldTitle = title;
this.titleProperty().set(title);
} public boolean isEditing() {
return editingProperty().get();
} public BooleanProperty editingProperty() {
if (editing == null) {
editing = new SimpleBooleanProperty(true);
}
return editing;
} public void setEditing(boolean editing) {
this.editingProperty().set(editing);
} public void setEditCallback(EditColumnCallback callback) {
this.callback = callback;
} public boolean isEditable() {
return editableProperty().get();
} public BooleanProperty editableProperty() {
if (editable == null) {
editable = new SimpleBooleanProperty(true);
}
return editable;
} public void setEditable(boolean editable) {
this.editableProperty().set(editable);
} public boolean isTextFieldFocus() {
return textFieldFocusProperty().get();
} public BooleanProperty textFieldFocusProperty() {
if (textFieldFocus == null) {
textFieldFocus = new SimpleBooleanProperty();
}
return textFieldFocus;
} public void setTextFieldFocus(boolean textFieldFocus) {
this.textFieldFocusProperty().set(textFieldFocus);
}
}
JavaFx TableView疑难详解的更多相关文章
- IOS详解TableView——选项抽屉(天猫商品列表)
在之前的有篇文章讲述了利用HeaderView来写类似QQ好友列表的表视图. 这里写的天猫抽屉其实也可以用该方法实现,具体到细节每个人也有所不同.这里采用的是点击cell对cell进行运动处理以展开“ ...
- tableview 详解I
在开发iphone的应用时基本上都要用到UITableView,这里讲解一下UITableView的使用方法及代理的调用情况 UITableView使用详解 - (void)viewDidLoad { ...
- JavaFx ObservableList的使用详解
原文地址:JavaFx ObservableList的使用详解 | Stars-One的杂货小窝 最近在研究MVVM模式,发现可以将之前写的FxRecyclerView控件改造一下,便是开始尝试,尝试 ...
- TableView 常用技巧与功能详解
分割线顶格iOS8 UITableview分割线顶格的做法 //iOS8 Cell分割线顶格 if ([_tableView respondsToSelector:@selector(setSepar ...
- 服务器.htaccess 详解以及 .htaccess 参数说明(转载)
htaccess文件(或者”分布式配置文件”)提供了针对目录改变配置的方法, 即,在一个特定的文档目录中放置一个包含一个或多个指令的文件, 以作用于此目录及其所有子目录.作为用户,所能使用的命令受到限 ...
- .htaccess详解及.htaccess参数说明【转】
目录(?)[-] htaccess 详解 htaccess rewrite 规则详细说明 RewriteEngine OnOff RewriteBase URL-path RewriteCond Te ...
- iPhone应用开发 UITableView学习点滴详解
iPhone应用开发 UITableView学习点滴详解是本文要介绍的内容,内容不多,主要是以代码实现UITableView的学习点滴,我们来看内容. -.建立 UITableView DataTab ...
- Cocos2d-x 3.X手游开发实例详解
Cocos2d-x 3.X手游开发实例详解(最新最简Cocos2d-x手机游戏开发学习方法,以热门游戏2048.卡牌为例,完整再现手游的开发过程,实例丰富,代码完备,Cocos2d-x作者之一林顺和泰 ...
- XML解析之SAX详解
XML解析之SAX详解 本文属于作者原创 http://www.cnblogs.com/ldnh/ XML解析的五个步骤 1.打开文档 (void)parserDidStartDocument:(NS ...
随机推荐
- Android -- 贝塞尔实现水波纹动画(划重点!!)
1,昨天看到了一个挺好的ui效果,是使用贝塞尔曲线实现的,就和大家来分享分享,还有,在写博客的时候我经常会把自己在做某种效果时的一些问题给写出来,而不是像很多文章直接就给出了解决方法,这里给大家解释一 ...
- 使用Eclipse Memory Analyzer Tool(MAT)分析线上故障(一) - 视图&功能篇
Eclipse Memory Analyzer Tool(MAT)相关文章目录: 使用Eclipse Memory Analyzer Tool(MAT)分析线上故障(一) - 视图&功能篇 使 ...
- [系统启动]Printk与sched_clock_init的一点分析
作者:Younger Liu, 本作品采用知识共享署名-非商业性使用-相同方式共享 3.0 未本地化版本许可协议进行许可. 在分析Linu内核启动的过程中,发现一段"不平常" ...
- Linux(ubuntu)安装MediaWiki
本篇文档所述步骤,作者完全验证过.一切OK. 作者:http://gaoxingf.blog.51cto.com/612518/188132,Younger Liu 本作品采用知识共享署名-非商业性使 ...
- selenium + python 登录页面,输入账号、密码,元素定位问题
示例简介: 要求:登录QQ邮箱,输入账号.密码 出现问题:页面中含有iframe框架,因此直接进行元素的查找与操作,出现找不到元素的现象,首先需进行iframe框架的转换,使用switch_to_fr ...
- web前端技术框架选型参考
一.出发点 随着Web技术的不断发展,前端架构框架.UI框架.构建工具.CSS预处理等层出不穷,各有千秋.太多的框架在形成初期,都曾在web领域 掀起过一场技术浪潮,可有些却仅仅是昙花一现,随着他们用 ...
- java内存模型二
数据依赖性 如果两个操作访问同一个变量,且这两个操作中有一个为写操作,此时这两个操作之间就存在数据依赖性.数据依赖分下列三种类型: 名称 代码示例 说明 写后读 a = 1;b = a; 写一个变量之 ...
- window 远程在Linux(centOS7.0)上安装JDK以及配置环境变量
本人是在windows 7 上安装了虚拟机,虚拟机安装的是linux(centOS7.0)系统现在在Windows 上安装SecureCRT 远程虚拟机的linux系统,安装JDK以及配置环境变量. ...
- javaWeb基础核心之一Servlet
既然是做JAVA开发的,先从一些基本的整理起来,算是知识回顾,加深记忆. 第一篇想到那理到哪,可能有点乱,不是太会排版,见谅,估计可能也就我自己看的懂. servlet在百度百科上的定义是这样的: S ...
- python requests 模拟登陆网站,抓取数据
抓取页面数据的时候,有时候我们需要登陆才可以获取页面资源,那么我们需要登陆以后才可以跳转到对应的资源页面,那么我们需要通过模拟登陆,登陆成功以后再次去抓取对应的数据. 首先我们需要通过手动方式来登陆一 ...