我编写了一个程序,该程序利用javafx中的拖放功能。在JavaFX8中,它可以完美运行。
在JavaFX11中,拖放操作不起作用:我没有得到不同的鼠标光标,没有得到我要拖动的行的幻影,并且拖放有问题-它们没有t会在鼠标释放时触发,然后在每次单击表格时触发该下降。
这是最小的可运行示例,它演示了我所面临的问题。在Java 8 JVM上运行,可以按需运行。在Java 11 JVM上没有。我在Ubuntu 18.04上。
我可以更改代码以适应Java 11,但是我不知道自己在做什么错。
Java 11版本
java version "11.0.1" 2018-10-16 LTS
Java(TM) SE Runtime Environment 18.9 (build 11.0.1+13-LTS)
Java HotSpot(TM) 64-Bit Server VM 18.9 (build 11.0.1+13-LTS, mixed mode)
Java 8版本
openjdk version "1.8.0_181"
OpenJDK Runtime Environment (build 1.8.0_181-8u181-b13-1ubuntu0.18.04.1-b13)
OpenJDK 64-Bit Server VM (build 25.181-b13, mixed mode)
DND11.java
import javafx.application.Application;
import javafx.collections.FXCollections;
import javafx.scene.Scene;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableRow;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.input.ClipboardContent;
import javafx.scene.input.DataFormat;
import javafx.scene.input.Dragboard;
import javafx.scene.input.TransferMode;
import javafx.stage.Stage;
public class DND11 extends Application {
public TableView<Person> getTable () {
DataFormat DRAGGED_PERSON = new DataFormat ( "application/example-person" );
TableColumn <Person, String> firstNameColumn = new TableColumn <> ( "First Name" );
TableColumn <Person, String> LastNameColumn = new TableColumn <> ( "Last Name" );
firstNameColumn.setCellValueFactory( new PropertyValueFactory <Person, String>( "firstName" ) );
LastNameColumn.setCellValueFactory( new PropertyValueFactory <Person, String>( "lastName" ) );
TableView <Person> tableView = new TableView <> ();
tableView.getColumns().addAll( firstNameColumn, LastNameColumn );
tableView.setColumnResizePolicy( TableView.CONSTRAINED_RESIZE_POLICY );
tableView.setEditable( false );
tableView.setItems( FXCollections.observableArrayList( Person.generatePersons ( 10 ) ) );
tableView.getSelectionModel().setSelectionMode( SelectionMode.MULTIPLE );
tableView.setRowFactory( tv -> {
TableRow <Person> row = new TableRow <>();
row.setOnDragDetected( event -> {
if ( !row.isEmpty() ) {
Dragboard db = row.startDragAndDrop( TransferMode.COPY );
ClipboardContent cc = new ClipboardContent();
cc.put( DRAGGED_PERSON, row.getItem() );
tableView.getItems().remove( row.getItem() );
db.setContent( cc );
}
});
row.setOnDragOver( event -> {
Dragboard db = event.getDragboard();
event.acceptTransferModes( TransferMode.COPY );
});
row.setOnDragDropped( event -> {
Dragboard db = event.getDragboard();
Person person = (Person)event.getDragboard().getContent( DRAGGED_PERSON );
if ( person != null ) {
tableView.getItems().remove( person );
int dropIndex = row.getIndex();
tableView.getItems().add( dropIndex, person );
}
event.setDropCompleted( true );
event.consume();
});
return row;
});
return tableView;
}
@Override
public void start ( Stage stage ) throws Exception {
stage.setScene( new Scene( getTable(), 800, 400 ) );
stage.show();
}
public static void main ( String[] args ) {
launch( args );
}
}
Person.java
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
public class Person implements Serializable {
private static final long serialVersionUID = 1L;
private String firstName, lastName;
public Person ( String firstName, String lastName ) {
this.firstName = firstName;
this.lastName = lastName;
}
public String getFirstName() {
return firstName;
}
public String getLastName() {
return lastName;
}
public static List <Person> generatePersons ( int number ) {
List<Person> retMe = new ArrayList<Person> ( number );
for ( int k = 0; k < number; k++ ) {
retMe.add ( new Person ( randomFirstName(), randomLastName() ) );
}
return retMe;
}
private static Random rand = new Random();
private static String randomFirstName() {
return firstNames [ Math.abs( rand.nextInt() ) % firstNames.length ];
}
private static String randomLastName() {
return lastNames [ Math.abs( rand.nextInt() ) % lastNames.length ];
}
private static String[] firstNames = new String[] {
"ANTON","ANTONE","ANTONIA","NTONIO","ANTONY","ANTWAN","ARCHIE","ARDEN","ARIEL","ARLEN",
"ARMAND","ARMANDO","ARNOLD","ARNOLDO","ARNULF","ARON","ARRON","ART","ARTHUR","ARTURO",
"DARRICK","DARRIN","DARRON","DARRYL","DARWIN","DARYL","DAVE","DAVID","DAVIS","DEAN",
};
private static String[] lastNames = new String[] {
"SMITH","JOHNSON","WILLIAMS","BROWN","JONES","MILLER","DAVIS","GARCIA","RODRIGUEZ",
"WILSON","MARTINEZ","ANDERSON","TAYLOR","THOMAS","HERNANDEZ","MOORE","MARTIN","JACKSON"
};
}
答案 0 :(得分:6)
尽管JavaFX中的拖放功能在所有平台上都有一个通用的API(当然,其余API也是这样),但其内部实现取决于平台,并且在Windows,Mac或Linux上完全不同。
但是,当从JavaFX 8迁移到JavaFX 11时,这应该不是问题。
OP发布的示例在Windows和Mac上以及JavaFX 8和11上均相同,如果在Linux上不是这样,则可能与最新版Linux的JavaFX中所做的更改有关
根据releases note,在Important Changes
部分下,我们可以看到:
将默认GTK版本切换为3
在存在gtk3库的Linux平台上,JavaFX现在默认情况下将使用GTK 3。在JavaFX 11之前,GTK 2库是默认库。这与JDK 11中AWT的默认值匹配。有关更多信息,请参见JDK-8198654。
虽然此change在JavaFX代码中基本上是两行差异,而DND的实现细节没有任何变化,但GTK 3的实现可能已从GTK 2进行了更改,而这些更改尚未采取考虑在内。
已经报告了dialogs,windows或Wayland crashes与GTK相关的类似问题。
解决方法
到目前为止,解决所有这些问题的唯一已知解决方法是使用GTK 2运行该应用程序,该应用程序可以使用系统属性jdk.gtk.version
进行设置。
因此可以在命令行上添加此选项:
java -Djdk.gtk.version=2 ...
运行该应用程序。
如评论中所述,这似乎解决了拖放问题。
报告问题
肯定会确认这是一个问题,因此应将其提交OpenJFX的问题tracker,并提供用于重现它的示例代码,系统详细信息(操作系统版本,Java版本,JavaFX版本)。 ..)。