gpt4 book ai didi

javafx - Java 11 和 Java 8 中的拖放工作方式不同

转载 作者:行者123 更新时间:2023-12-04 01:06:45 25 4
gpt4 key购买 nike

我写了一个程序,它在 javafx 中使用拖放。在 JavaFX8 中,它完美运行。

在 JavaFX11 中,拖放功能不正常:我没有得到不同的鼠标光标,我没有得到我正在拖动的行的重影图像,而且拖放有问题——它们不会触发鼠标释放,然后每次单击 table 时都会触发放置。

这是最小的可运行示例,它演示了我面临的问题。在 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"
};
}

最佳答案

虽然 JavaFX 中的拖放具有适用于所有平台的通用 API(当然,与 API 的其余部分一样),但其内部实现取决于平台,并且在 Windows、Mac 或 Linux 上完全不同。
但是,当从 JavaFX 8 迁移到 JavaFX 11 时,这应该不是问题。
OP 发布的示例在带有 JavaFX 8 和 11 的 Windows 和 Mac 上的工作方式相同,如果在 Linux 上不是这种情况,则可能与最新版本的 JavaFX for Linux 中所做的更改有关。
根据releases note ,下Important Changes部分我们可以看到:

Switch default GTK version to 3

JavaFX will now use GTK 3 by default on Linux platforms where the gtk3 library is present. Prior to JavaFX 11, the GTK 2 library was the default. This matches the default for AWT in JDK 11. See JDK-8198654 for more information.


虽然这个 change基本上是 JavaFX 代码中的两行差异,DND 的实现细节没有任何变化,GTK 3 实现可能与 GTK 2 相比有所改变,并且没有考虑这些变化。
已针对 dialogs 报告了与 GTK 相关的类似问题, windowsWayland crashes .
解决方法
到目前为止,所有这些问题的唯一已知解决方法是使用 GTK 2 运行应用程序,可以使用系统属性进行设置: jdk.gtk.version .
所以这个选项可以在命令行上添加:
java -Djdk.gtk.version=2 ...
运行应用程序。
正如评论中发布的那样,这似乎解决了拖放问题。
报告问题
当然,这证实了这是一个问题,因此应该在 OpenJFX 问题 tracker 上提交。 , 提供重现它的示例代码、系统详细信息(操作系统版本、Java 版本、JavaFX 版本...)。

关于javafx - Java 11 和 Java 8 中的拖放工作方式不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53333364/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com