У меня есть TableView, который прокручивается по вертикали, и я хотел бы, чтобы полоса прокрутки простиралась до верхней части родительской AnchorPane и находилась поверх квадрата-заполнителя в правом верхнем углу. Ниже показано, как это выглядит по умолчанию. Обратите внимание, что мой узел-заполнитель белого цвета, а не столбец таблицы в правом верхнем углу.
а ниже этой строки то, что я хочу, корректно реализованное другой программой.
Я смог добиться этого, выполнив
Platform.runLater(() ->
{
ScrollBar someScrollBar = (ScrollBar) someTable.lookup(".scroll-bar:vertical");
someScrollBar.setTranslateY(-12);
someScrollBar.setScaleY(1.2);
}
);
где someTable — это TableView, созданный в FXML, на который ссылается функция инициализации контроллера.
Это выглядит хорошо, как это, но это не масштабируется правильно. Если содержащее AnchorPane изменяет размер по вертикали, это выглядит ужасно.
Может ли кто-нибудь предложить лучший способ сделать это?
Большое вам спасибо за ваше время.




Пользовательский макет полос прокрутки не поддерживается. И мой первоначальный комментарий вам нужен собственный TableViewSkin с пользовательским TableHeaderRow: последний отвечает за управление .. ну, tableHeader, к сожалению, только часть истории.
layoutChildren(), он немедленно сбрасываетсяИтак, в конце концов, нам нужно
Пример написан для fx11, должен работать для fx10, но не для fx9, потому что последний не позволяет предоставить собственный VirtualFlow:
public class TableWithoutCorner extends Application {
/**
* Custom TableHeaderRow that requests a larger vbar height
* if needed.
*/
private static class MyTableHeader extends TableHeaderRow {
private Region cornerAlias;
private ScrollBar vBar;
private TableViewSkinBase skin;
public MyTableHeader(TableViewSkinBase skin) {
super(skin);
this.skin = skin;
}
@Override
protected void layoutChildren() {
super.layoutChildren();
adjustCornerLayout();
}
private void adjustCornerLayout() {
checkAlias();
// tbd: check also if corner is visible
if (!vBar.isVisible()) {
vBar.getProperties().remove("DELTA");
} else {
vBar.getProperties().put("DELTA", getHeight());
}
}
private void checkAlias() {
if (cornerAlias == null) {
cornerAlias = (Region) lookup(".show-hide-columns-button");
}
if (vBar == null) {
vBar = (ScrollBar) skin.getSkinnable().lookup(".scroll-bar:vertical");
}
}
}
/**
* Custom VirtualFlow that respects additinal height for its
* vertical ScrollBar.
*/
private static class MyFlow extends VirtualFlow {
private ScrollBar vBar;
private Region clip;
public MyFlow() {
// the scrollbar to adjust
vBar = (ScrollBar) lookup(".scroll-bar:vertical");
// the clipped container to use for accessing viewport dimensions
clip = (Region) lookup(".clipped-container");
}
/**
* Overridden to adjust vertical scrollbar's height and y-location
* after calling super.
*/
@Override
protected void layoutChildren() {
super.layoutChildren();
adjustVBar();
}
/**
* Adjusts vBar height and y-location by the height as
* requested by the table header.
*/
protected void adjustVBar() {
if (vBar.getProperties().get("DELTA") == null) return;
double delta = (double) vBar.getProperties().get("DELTA");
vBar.relocate(clip.getWidth(), - delta);
vBar.resize(vBar.getWidth(), clip.getHeight() + delta);
}
}
/**
* Boilerplate: need custom TableViewSkin to inject a custom TableHeaderRow and
* custom VirtualFlow.
*/
private static class MyTableViewSkin<T> extends TableViewSkin<T> {
public MyTableViewSkin(TableView<T> control) {
super(control);
}
@Override
protected TableHeaderRow createTableHeaderRow() {
return new MyTableHeader(this);
}
@Override
protected VirtualFlow<TableRow<T>> createVirtualFlow() {
return new MyFlow();
}
}
private Parent createContent() {
TableView<Locale> table = new TableView<>(FXCollections.observableArrayList(Locale.getAvailableLocales())) {
@Override
protected Skin<?> createDefaultSkin() {
return new MyTableViewSkin(this);
}
};
TableColumn<Locale, String> col = new TableColumn<>("Name");
col.setCellValueFactory(new PropertyValueFactory<>("displayName"));
table.getColumns().addAll(col);
return table;
}
@Override
public void start(Stage stage) throws Exception {
stage.setScene(new Scene(createContent()));
//stage.setTitle(FXUtils.version());
stage.show();
}
public static void main(String[] args) {
launch(args);
}
@SuppressWarnings("unused")
private static final Logger LOG = Logger
.getLogger(TableWithoutCorner.class.getName());
}
вам нужен собственный TableViewSkin с пользовательским TableHeaderRow: последний отвечает за управление .. ну, tableHeader :)