- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
此问题是 Custom object drag-and-drop from FX to Swing 的后续问题 .
我正在为一个将 JavaFX 用于某些图形用户界面的 Swing 应用程序开发一个插件。我们添加了拖放功能以改善用户体验。首先,我们为我们的Scene
使用外部 JavaFX 窗口 (Stage
),现在我们想通过 JFXPanel
将它直接嵌入到 Swing 应用程序中>.
现在,奇怪的是,无论是在 Stage
中加载完全相同的 Scene
还是在一个 JFXPanel
。
在尝试将具有自定义 MIME 类型的自定义 Java 对象(以序列化形式)从 JavaFX 应用程序拖到 Swing 应用程序时,我已经遇到了一些问题。但是,我的问题在我上面提到的问题中得到了解决。现在,使用嵌入式JavaFX应用程序,遇到了一些新问题,所以我想问问有没有人有类似的问题或知道这种情况的解决方案。
我写了一个 MVCE,它是一个简单的 Java 应用程序,一侧有一个支持拖动的 JFXPanel
,另一侧有一个支持拖放的 JPanel
:
public class MyApp {
public static final DataFormat FORMAT = new DataFormat(
// this works fine in a separate window
//"JAVA_DATAFLAVOR:application/x-my-mime-type; class=java.lang.String",
"application/x-my-mime-type; class=java.lang.String");
public static final DataFlavor FLAVOR;
static {
try {
FLAVOR = new DataFlavor("application/x-my-mime-type; class=java.lang.String");
} catch (ClassNotFoundException ex) {
throw new RuntimeException(ex);
}
}
public static void main(String[] args) {
new MyApp().run();
}
private void run() {
JFrame frame = new JFrame();
frame.setLayout(new GridLayout(1, 2));
frame.add(buildFX());
frame.add(buildSwing());
frame.setSize(300, 300);
frame.setVisible(true);
}
private JFXPanel buildFX() {
BorderPane parent = new BorderPane();
parent.setOnDragDetected(event -> {
Dragboard dragboard = parent.startDragAndDrop(TransferMode.COPY);
ClipboardContent content = new ClipboardContent();
content.put(FORMAT, "Test");
dragboard.setContent(content);
event.consume();
});
JFXPanel panel = new JFXPanel();
panel.setScene(new Scene(parent));
return panel;
}
@SuppressWarnings("serial")
private JPanel buildSwing() {
JPanel panel = new JPanel();
panel.setBackground(Color.ORANGE);
panel.setTransferHandler(new TransferHandler() {
@Override
public boolean canImport(TransferSupport support) {
return support.isDataFlavorSupported(FLAVOR);
}
@Override
public boolean importData(TransferSupport support) {
if (!canImport(support)) return false;
try {
String data = (String) support.getTransferable().getTransferData(FLAVOR);
System.out.println(data);
return true;
} catch (UnsupportedFlavorException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return false;
}
});
return panel;
}
}
根据另一个问题的回答,在DataFormat
中使用前缀JAVA_DATAFLAVOR:
是Swing正确处理MIME类型所必需的。但是,当在 JFXPanel
(示例中禁用)中使用这样的 DataFormat
时,Java 似乎在从中拖动时试图构造一个 DataFlavor
FX 应用程序无法解析带有前缀的 MIME 类型:
Exception in thread "AWT-EventQueue-0" java.lang.IllegalArgumentException: failed to parse:JAVA_DATAFLAVOR:application/x-my-mime-type; class=java.lang.String
at java.awt.datatransfer.DataFlavor.<init>(Unknown Source)
at javafx.embed.swing.SwingDnD$DnDTransferable.getTransferDataFlavors(SwingDnD.java:394)
at sun.awt.datatransfer.DataTransferer.getFormatsForTransferable(Unknown Source)
at sun.awt.dnd.SunDragSourceContextPeer.startDrag(Unknown Source)
at java.awt.dnd.DragSource.startDrag(Unknown Source)
at java.awt.dnd.DragSource.startDrag(Unknown Source)
at java.awt.dnd.DragGestureEvent.startDrag(Unknown Source)
at javafx.embed.swing.SwingDnD.startDrag(SwingDnD.java:280)
at javafx.embed.swing.SwingDnD.lambda$null$66(SwingDnD.java:247)
at java.awt.event.InvocationEvent.dispatch(Unknown Source)
at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
at java.awt.EventQueue.access$500(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
at java.awt.EventQueue.dispatchEvent(Unknown Source)
at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.run(Unknown Source)
仅使用没有前缀的纯 MIME 类型,拖放操作有效,我什至可以接收到正确的 DataFlavor
(java.awt.datatransfer.DataFlavor[mimetype =application/x-my-mime-type;representationclass=java.lang.String]
),但是丢弃的数据总是null
。正如在另一个问题中看到的那样,在两个单独的窗口中使用第二种方法,我什至无法接收到 DataFlavor
,但现在它以某种方式工作到这个有限的点。
最佳答案
可能对传输的工作方式存在一些误解。
尝试直接将传输数据作为字符串检索可能适用于“text/plain”或其他标准文本类型,并且正如您所注意到的,对于自定义未注册类型的特定情况有一些怪癖。但我认为自定义解决方法的努力是不合理的。
由于您完全控制自定义 MIME 类型的内容结构以及同一应用程序中数据生产者和消费者的两端,我建议不要处理内部工具包依赖于实现的前缀或类映射。可能更好的做法是只定义您的 MIME 类型,而没有不相关的元数据和格式错误的前缀(因为它应该是)。
定义“application/x-my-mime”类型并正确解码数据就足够了。
下面根据您的示例进行了更正,应该可以将数据很好地放入 Java 8 中的 Swing 框架。
package jfxtest;
import java.awt.Color;
import java.awt.GridLayout;
import java.awt.datatransfer.DataFlavor;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.util.Collections;
import javafx.embed.swing.JFXPanel;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.input.DataFormat;
import javafx.scene.input.TransferMode;
import javafx.scene.layout.BorderPane;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
import javax.swing.TransferHandler;
import javax.swing.TransferHandler.TransferSupport;
public class MyApp {
final static String MY_MIME_TYPE = "application/x-my-mime";
public static final DataFormat FORMAT = new DataFormat(MY_MIME_TYPE);
public static final DataFlavor FLAVOR = new DataFlavor(MY_MIME_TYPE, "My Mime Type");
private void startDrag(Node node) {
node.startDragAndDrop(TransferMode.COPY).setContent(
Collections.singletonMap(FORMAT, "Test"));
}
private boolean processData(TransferSupport support) {
try (InputStream in = (InputStream) support.getTransferable().getTransferData(FLAVOR)) {
Object transferred = new ObjectInputStream(in).readObject();
System.out.println("transferred: " + transferred + " (" + transferred.getClass() + ")");
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
public static void main(String[] args) {
new MyApp().run();
}
private void run() {
JFrame frame = new JFrame();
frame.setLayout(new GridLayout(1, 2));
frame.add(buildSwing());
SwingUtilities.invokeLater(() -> {
frame.add(buildFX());
});
frame.setSize(300, 300);
frame.setVisible(true);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
private JFXPanel buildFX() {
BorderPane parent = new BorderPane();
parent.setOnDragDetected(event -> {
startDrag(parent);
event.consume();
});
JFXPanel panel = new JFXPanel();
panel.setScene(new Scene(parent));
return panel;
}
private JPanel buildSwing() {
JPanel panel = new JPanel();
panel.setBackground(Color.ORANGE);
panel.setTransferHandler(new TransferHandler() {
private static final long serialVersionUID = 1L;
@Override
public boolean canImport(TransferSupport support) {
return support.isDataFlavorSupported(FLAVOR);
}
@Override
public boolean importData(TransferSupport support) {
if (canImport(support)) {
return processData(support);
}
return false;
}
});
return panel;
}
}
输出:传输:测试(类java.lang.String)
这里的重要摘录是:
...
final static String MY_MIME_TYPE = "application/x-my-mime";
public static final DataFormat FORMAT = new DataFormat(MY_MIME_TYPE);
public static final DataFlavor FLAVOR = new DataFlavor(MY_MIME_TYPE, "My Mime Type");
private void startDrag(Node node) {
node.startDragAndDrop(TransferMode.COPY).setContent(
Collections.singletonMap(FORMAT, "Test"));
}
private boolean processData(TransferSupport support) {
try (InputStream in = (InputStream) support.getTransferable().getTransferData(FLAVOR)) {
Object transferred = new ObjectInputStream(in).readObject();
System.out.println("transferred: " + transferred + " (" + transferred.getClass() + ")");
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
...
请注意,出于说明目的,数据检索非常简单,对于实际应用程序,可能希望添加更严格的流读取、处理错误等。
第一个示例传输一个序列化对象(这通常是一件好事和简单的事情,因为您可以传输任何可序列化的东西,但很难传输/接受,比如第 3 方 JSON)。在不太可能的情况下,当您希望为自定义 MIME 而不是序列化对象生成真实文本或其他任意内容时,以下内容应该可以完成工作:
package jfxtest;
import java.awt.Color;
import java.awt.GridLayout;
import java.awt.datatransfer.DataFlavor;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import javafx.embed.swing.JFXPanel;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.input.DataFormat;
import javafx.scene.input.TransferMode;
import javafx.scene.layout.BorderPane;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
import javax.swing.TransferHandler;
import javax.swing.TransferHandler.TransferSupport;
public class MyApp {
final static String MY_MIME_TYPE = "application/x-my-mime";
public static final DataFormat FORMAT = new DataFormat(MY_MIME_TYPE);
public static final DataFlavor FLAVOR = new DataFlavor(MY_MIME_TYPE, "My Mime Type");
private void startDrag(Node node) {
node.startDragAndDrop(TransferMode.COPY).setContent(
// put a ByteBuffer to transfer the content unaffected
Collections.singletonMap(FORMAT, StandardCharsets.UTF_8.encode("Test")));
}
private boolean processData(TransferSupport support) {
try (InputStream in = (InputStream) support.getTransferable().getTransferData(FLAVOR)) {
byte[] textBytes = new byte[in.available()];
in.read(textBytes);
String transferred = new String(textBytes, StandardCharsets.UTF_8);
System.out.println("transferred text: " + transferred);
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
public static void main(String[] args) {
new MyApp().run();
}
private void run() {
JFrame frame = new JFrame();
frame.setLayout(new GridLayout(1, 2));
frame.add(buildSwing());
SwingUtilities.invokeLater(() -> {
frame.add(buildFX());
});
frame.setSize(300, 300);
frame.setVisible(true);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}
private JFXPanel buildFX() {
BorderPane parent = new BorderPane();
parent.setOnDragDetected(event -> {
startDrag(parent);
event.consume();
});
JFXPanel panel = new JFXPanel();
panel.setScene(new Scene(parent));
return panel;
}
private JPanel buildSwing() {
JPanel panel = new JPanel();
panel.setBackground(Color.ORANGE);
panel.setTransferHandler(new TransferHandler() {
private static final long serialVersionUID = 1L;
@Override
public boolean canImport(TransferSupport support) {
return support.isDataFlavorSupported(FLAVOR);
}
@Override
public boolean importData(TransferSupport support) {
if (canImport(support)) {
return processData(support);
}
return false;
}
});
return panel;
}
}
输出:传输的文本:测试
这里最重要的部分是:
...
private void startDrag(Node node) {
node.startDragAndDrop(TransferMode.COPY).setContent(
// put a ByteBuffer to transfer the content unaffected
Collections.singletonMap(FORMAT, StandardCharsets.UTF_8.encode("Test")));
}
private boolean processData(TransferSupport support) {
try (InputStream in = (InputStream) support.getTransferable().getTransferData(FLAVOR)) {
byte[] textBytes = new byte[in.available()];
in.read(textBytes);
String transferred = new String(textBytes, StandardCharsets.UTF_8);
System.out.println("transferred text: " + transferred);
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
...
再次说明,流、错误等在这里的处理很简单。
需要注意的一件事是,还有一个预定义的“application/x-java-serialized-object”(DataFlavor.javaSerializedObjectMimeType
)用于更通用和更容易的反序列化。但长期自定义 MIME 似乎更灵活,整体处理起来更直接。
关于java - 将自定义对象从嵌入式 FX (JFXPanel) 拖放到 Swing,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52599005/
我是 MooTools 的新手。我注意到 Morph 效果类似于 Tween 效果。 唯一的区别是 tween 用于一个属性,而 morph 用于多个属性。 有人能告诉我这些效果最适合做什么吗,即一种
和有什么区别和 ?我是否需要两者,或者其中一个是另一个的新替代品? 最佳答案 脚本是将一些 AS3 代码放入 MXML 类中的地方。声明是一种将 MXML 中定义的公共(public)属性放在类上的
您好,我有一个可以在 GUI(Java FX)和命令行上运行的应用程序。当作为 GUI 运行时,我在文本区域显示状态。这工作得很好。 但问题是,当我尝试显示来自某个不同(非 javafx)类的错误(通
我正在为我的局域网应用程序创建包含日志的表。当我尝试运行此应用程序时,它抛出 NullPointerException,但 fx:id 和 fx:controller 是正确的。我正在使用 JavaF
我想将纹理应用于自定义形状。我认为这会起作用: .myTextureShape { -fx-shape: "M0 0 L0 50 L25 25 L50 50 L50 0 Z"; -fx-backgro
我正在使用 JavaFX,我想自定义一个按钮。我看到了几个可以设计样式的特征。其中我发现了两个我不认识的。 .button { -fx-padding: 5 22 5 22; -f
我想在一秒钟内淡入一个节点。然后保持 10 秒钟。然后 fadeOut 再持续 3 秒。一种链接方式如下: dojo.fx.chain([ dojo.fadeIn({node:myNode, d
我是 Java Fx 的真正初学者,并且在将我在网络上找到的两个 png 文件加载到我的应用程序中时遇到问题。有人可以告诉我可能出了什么问题吗?下面是我正在使用的代码: Button fBut
在 JavaFX CSS 中,一个 Label 似乎有 2 个 CSS 填充属性。 Label具有 Labeled 的所有属性, 在这之下我们有 -fx-label-padding. 但是,Label
由于我的规则已经到位,但这些错误很流畅。我试图了解网络上可用的解决方案,但没有一个对我有用。 Jul 18, 2016 1:39:17 PM javafx.scene.CssStyleHelper c
afterburner.fx for JavaFX 8 是一个简约的(3 类)JavaFX MVP 框架,基于配置约定和依赖注入(inject),由亚当·比恩。 afterburner.fx 使用 M
更新: 我发现问题出在 Inno Setup 上。安装程序正在创建,但 Inno Setup 会自动尝试运行需要管理员权限的安装程序。由于未授予这些权限,因此安装程序失败。 如果我将权限授予使用 In
我可以在我的IntelliJ项目中包含Controsfx,但我不能运行该程序。我也尝试了另一种解决方案:将此添加到VM选项中。当我运行该程序时,它出现运行时错误:
I could include the controlsfx into my project in IntelliJ but I can't run the program.I also alr
我想通过鼠标拖动或缩放按钮实现实时折线图的放大功能(图表每 50 毫秒移动一次)。我想放大特定区域..但是我面临的问题是每次图形中的(x,y)坐标值发生变化时。有什么办法可以放大这个图表吗? Java
我进行了搜索,但找不到任何与我在 Fx 中遇到的问题相近的内容。我正在使用 Java Fx/JDK 8,并且在调整场景大小时遇到问题。下面的代码在场景图中一次仅加载一个屏幕,并在屏幕之间切换。问题
在我的程序中,我必须通过单击单选按钮来更改示例字符串的颜色。但每次我点击它们时我都会收到错误。我不断收到此错误。我还使用 e(fx)clipse 来对此进行编码。 Caused by: java.la
更新 2019.05.14 4:54PM EST - 好的 - 这是说明我的问题的代码 - 可能花了我太长时间才得到这个,而且可能太长了,但同样,我是 Java 新手。无论如何 - 它有效,打开表单,
我有一个 FX 应用程序,从一开始,它就开始播放音乐,这很有效。现在我希望当我打开一个方法来在其上播放另一个声音时。 我将第二个声音编码为第一个,但它不起作用。我尝试创建一个新的Thread,但没有改
我最近正在使用 blob 图像进行图像上传,并且它有效但现在当我尝试在搜索中搜索特定的人时,它没有显示我试图将 blob 图像转换为字节并以 java fx 形式显示在图像字段中 这是我的代码 @F
我是一名优秀的程序员,十分优秀!