TableView 从排序中排除底行(总计)
题
我有一个简单的 TableView(Java FX 2.0,但我认为问题相当普遍),它获得默认的排序功能。然而,该表的最后一行有总计,因此我想从排序算法中排除最后一行。
我找到了 Swing JTable 的解决方案,其中包括 为总行创建一个单独的表 - 这可以转置为 TableView,但似乎有点麻烦。我尝试过实现我自己的比较器,但我认为不可能创建一个既可以升序又可以降序运行的比较器。
解决方案
使用 JavaFX 8 可以定义排序策略,并且问题更容易解决。假设包含总计的行是 TOTAL
:
table.sortPolicyProperty().set(t -> {
Comparator<Row> comparator = (r1, r2) ->
r1 == TOTAL ? 1 //TOTAL at the bottom
: r2 == TOTAL ? -1 //TOTAL at the bottom
: t.getComparator() == null ? 0 //no column sorted: don't change order
: t.getComparator().compare(r1, r2); //columns are sorted: sort accordingly
FXCollections.sort(t.getItems(), comparator);
return true;
});
其他提示
基于lolsvemir的答案,我成功实现了与自定义比较器的“总计”。
我使用列的sorttypeproperty设置了比较器:
col.setComparator(new BigDecimalColumnComparator(col.sortTypeProperty()));
自定义比较器:
public class BigDecimalColumnComparator implements Comparator<BigDecimal> {
private final ObjectProperty<TableColumn.SortType> sortTypeProperty;
public BigDecimalColumnComparator(ObjectProperty<TableColumn.SortType> sortTypeProperty) {
this.sortTypeProperty = sortTypeProperty;
}
@Override
public int compare(BigDecimal o1, BigDecimal o2) {
TableColumn.SortType sortType = sortTypeProperty.get();
if (sortType == null) {
return 0;
}
if (o1 instanceof TotalBigDecimal) {
if (sortType == TableColumn.SortType.ASCENDING) {
return 1;
} else {
return -1;
}
} else if (o2 instanceof TotalBigDecimal) {
if (sortType == TableColumn.SortType.ASCENDING) {
return -1;
} else {
return 1;
}
}
return o1.compareTo(o2);
}
}
. 如果您在比较器的Compareto()方法中总是返回0?这意味着所有元素“是相同的”,并且不会发生任何地方的交换?
我有同样的问题,并且由于表view属性来调用Cell Factory,在表中可见的数量,而不仅可以用于TableView.setItems()方法设置的列表索引, imo最好通过Cell Factory为需要拥有它的每列设置摘要值。
这是这种小区工厂类的简单示例:
public class LocalDateTableCell<T> implements
Callback<TableColumn<T, LocalDate>, TableCell<T, LocalDate>> {
@Override
public TableCell<T, LocalDate> call(TableColumn<T, LocalDate> p) {
TableCell<T, LocalDate> cell = new TableCell<T, LocalDate>() {
@Override
protected void updateItem(LocalDate item, boolean empty) {
super.updateItem(item, empty);
if (item == null || empty) {
setText(null);
} else {
setText(DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM)
.format(item));
}
if (getTableView().getItems().size() == getIndex())
setText("Test"); // This shows in the summary row
}
};
return cell;
}
}
.
这种方法的优点是,这种行根本不涉及排序(因为它不在表项列表中)也没有选择。此外,通过覆盖Tablerow工厂易于为整个摘要行设置不同的背景颜色,并将其与其他行区分开来。
但是,滚动下滚动时,此额外行不会可见,因为TableView滚动实现不知道附加行,并且仅向视口底部的项目列表中滚动到最后一行。可以通过覆盖tableviewskin.getItemcount()方法来解决,例如在以下示例中。覆盖TableView本身不是必需的,但如果经常使用此类表,则建议使用。
public class MyTableViewSkin<T> extends TableViewSkin<T> {
private TableView<T> tableView;
public MyTableViewSkin(final TableView<T> tableView) {
super(tableView);
this.tableView = tableView;
}
@Override
public int getItemCount() {
return tableView.getItems() == null || tableView.getItems().size() == 0 ?
0 : tableView.getItems().size() + 1;
}
}
public class MyTableView<S> extends TableView<S> {
public MyTableView() {
super();
setSkin(new MyTableViewSkin<>(this));
}
}
.
这可能看起来像一个黑客,但是像魅力一样。