Используя только ListView, это очень сложно, и я не уверен, что это вообще возможно, потому что ListView показывает полосы прокрутки так, как необходимо, и, кажется, нет доступа к полосам прокрутки (см. это сообщение ).
Но есть обходной путь, использующий ScrollPane:
1. Поместите ListView в ScrollPane
2. Измените размер ListView так, чтобы он всегда был достаточно большим, чтобы отображать содержимое (если недостаточно места, он будет расширен в пределах ScrollPane)
3. Сделать полосы прокрутки ScrollPane видимыми только при наведении на них мышью
Пример, который я протестировал:
Класс Application для запуска тестового приложения:
package listViewScrollbar;
import java.net.URL;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
public class ListViewScrollbarApplication extends Application {
public static void main(String[] args) {
launch(args);
}
@Override
public void start(Stage primaryStage) {
try {
URL fxmlUrl = getClass().getResource("ListViewScrollbarTest.fxml");
FXMLLoader fxmlLoader = new FXMLLoader(fxmlUrl);
ListViewScrollbarController controller = new ListViewScrollbarController();
fxmlLoader.setController(controller);
Parent root = fxmlLoader.load();
Scene scene = new Scene(root, 200, 300);
scene.getStylesheets().add(getClass().getResource("ListViewScrollbar.css").toExternalForm());
primaryStage.setScene(scene);
primaryStage.show();
}
catch (Exception e) {
e.printStackTrace();
}
}
}
Файл fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.ListView?>
<?import javafx.scene.control.ScrollPane?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.BorderPane?>
<AnchorPane xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1">
<children>
<BorderPane layoutX="-89.0" layoutY="-123.0" prefHeight="300.0" prefWidth="400.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<center>
<ScrollPane fx:id="scrollPane" BorderPane.alignment="CENTER">
<content>
<ListView fx:id="listView" prefHeight="500.0" prefWidth="500.0" />
</content>
</ScrollPane>
</center>
</BorderPane>
</children>
</AnchorPane>
Пример контроллера:
package listViewScrollbar;
import java.net.URL;
import java.util.ResourceBundle;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.ListView;
import javafx.scene.control.ScrollPane;
import javafx.scene.control.ScrollPane.ScrollBarPolicy;
public class ListViewScrollbarController implements Initializable {
@FXML
private ListView<String> listView;
@FXML
private ScrollPane scrollPane;
@Override
public void initialize(URL location, ResourceBundle resources) {
//just add some text for the example
ObservableList<String> strings = FXCollections.observableArrayList("String 1",
"A very long String ...............................................................", "String 3", "Another String");
listView.setItems(strings);
//here you enable and disable the scroll bars depending on the mouse hovering the list view
scrollPane.setOnMouseEntered(e -> {
scrollPane.setHbarPolicy(ScrollBarPolicy.ALWAYS);
scrollPane.setVbarPolicy(ScrollBarPolicy.ALWAYS);
});
scrollPane.setOnMouseExited(e -> {
scrollPane.setHbarPolicy(ScrollBarPolicy.NEVER);
scrollPane.setVbarPolicy(ScrollBarPolicy.NEVER);
});
}
}
Теперь, пока размер ListView достаточно велик, чтобы показать содержимое, которое должен делать этот обходной путь.