我有一个简单的 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));
  }
}
.

这可能看起来像一个黑客,但是像魅力一样。

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top