java - Java剪贴板不与系统共享
问题描述
我有一个格式为 HTML 的 JEditorPane 字段,当我将它复制到剪贴板时,系统正在丢失所有换行符。
JEditorPane 中的文本:
<html>
<head>
</head>
<body>
<p>
Bonjour,<br><br>Merci de faire les actions ci-dessous:<br><br>1-
Sauvegarder le fichier AAAA<br>2- Remplacer le contenu du
fichier AAAA par le contenu fichier
BBBBB <br>3- Relancer au step planté<br><br>Cordialement,<br>
</p>
</body>
</html>
使用 Ctrl + C 复制并将其粘贴到 Notepad++ 中时,文本为:
Bonjour, Merci de faire les actions ci-dessous: 1- Sauvegarder le fichier AAAA 2- Remplacer le contenu du fichier AAAA par le contenu fichier BBBB 3- Relancer au step planté Cordialement,
所以,我截取了 Ctrl+C:
public void keyPressed(KeyEvent e) {
//Modify clipboard Ctrl+C
if ((e.getKeyCode() == KeyEvent.VK_C) && ((e.getModifiers() & KeyEvent.CTRL_MASK) != 0)) {
super.keyPressed(e);
genericFunctions.setClipboardText(edSolution.getText());//see function bellow
}
我试过修改它:
static void setClipboardText(String s) {
Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
StringSelection stringSelection = new StringSelection(s);
clipboard.setContents(stringSelection, stringSelection);
}
但剪贴板内的文本没有改变。
奇怪的是,如果我在下面使用这个函数:
static void setClipboardText(String s) {
Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
Clipboard selection = Toolkit.getDefaultToolkit().getSystemSelection();
StringSelection stringSelection = new StringSelection(s);
StringSelection stringS = new StringSelection(s);
clipboard.setContents(stringSelection, stringSelection);
selection.setContents(stringS, stringS);//error AWT-EventQueue-0" java.lang.NullPointerException"
}
即使我收到 NullPointerException,剪贴板也已使用新值进行了很好的修改。
有人能告诉我为什么剪贴板只在错误之后而不是之前刷新吗?我怎样才能宣布剪贴板已更改并应该阅读它的系统?
JDK8 v1.8.0_251
解决方案
我推测“默认”复制操作仅将编辑器中的文本复制为“纯文本” - 或者至少,这是接收者可用的选项之一,并且将其作为首选项(我不可以轻松访问 Java 源代码以深入了解默认实现)
如将 JTextArea 复制为“text/html”DataFlavor并将jTable 行及其网格线复制到 excel/word 文档中所述,您可以覆盖默认的“复制”操作并插入您自己的工作流程/格式。
这是一个非常简单的概念示例。当我使用它复制和粘贴文本时,我得到了完整的 HTML 文本。
如果您不想简单地获取带有换行符的文本,则必须自己解释内容并生成合适的输出String
,然后您可以将其DataFlavor.stringFlavor
用作主要数据风格,但这超出了范围。
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Toolkit;
import java.awt.datatransfer.Clipboard;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.event.ActionEvent;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.StringBufferInputStream;
import java.io.StringReader;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import javax.swing.AbstractAction;
import javax.swing.JEditorPane;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
public class Test {
public static void main(String[] args) {
new Test();
}
public Test() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
JFrame frame = new JFrame();
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class TestPane extends JPanel {
private JEditorPane editorPane;
public TestPane() {
URL url = getClass().getResource("Text.html");
editorPane = new JEditorPane();
editorPane.setContentType("html/text");
try {
editorPane.setPage(url);
} catch (IOException ex) {
ex.printStackTrace();
}
editorPane.getActionMap().put("copy", new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
String text = editorPane.getText();
Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
clipboard.setContents(new HtmlSelection(text), null);
}
});
setLayout(new BorderLayout());
add(new JScrollPane(editorPane));
}
@Override
public Dimension getPreferredSize() {
return new Dimension(200, 200);
}
}
private static class HtmlSelection implements Transferable {
private static List<DataFlavor> htmlFlavors = new ArrayList<>(3);
static {
try {
htmlFlavors.add(new DataFlavor("text/html;class=java.lang.String"));
htmlFlavors.add(new DataFlavor("text/html;class=java.io.Reader"));
htmlFlavors.add(new DataFlavor("text/html;charset=unicode;class=java.io.InputStream"));
} catch (ClassNotFoundException ex) {
ex.printStackTrace();
}
}
private String html;
public HtmlSelection(String html) {
this.html = html;
}
public DataFlavor[] getTransferDataFlavors() {
return (DataFlavor[]) htmlFlavors.toArray(new DataFlavor[htmlFlavors.size()]);
}
public boolean isDataFlavorSupported(DataFlavor flavor) {
return htmlFlavors.contains(flavor);
}
public Object getTransferData(DataFlavor flavor) throws UnsupportedFlavorException {
if (String.class.equals(flavor.getRepresentationClass())) {
return html;
} else if (Reader.class.equals(flavor.getRepresentationClass())) {
return new StringReader(html);
} else if (InputStream.class.equals(flavor.getRepresentationClass())) {
return new StringBufferInputStream(html);
}
throw new UnsupportedFlavorException(flavor);
}
}
}
更新
好的,所以经过一番挖掘,我发现它JEditorPane
正在使用默认功能从JTextComponent
将内容复制Document
到剪贴板......
public void exportToClipboard(JComponent comp, Clipboard clipboard,
int action) throws IllegalStateException {
if (comp instanceof JTextComponent) {
JTextComponent text = (JTextComponent)comp;
int p0 = text.getSelectionStart();
int p1 = text.getSelectionEnd();
if (p0 != p1) {
try {
Document doc = text.getDocument();
String srcData = doc.getText(p0, p1 - p0);
StringSelection contents =new StringSelection(srcData);
// this may throw an IllegalStateException,
// but it will be caught and handled in the
// action that invoked this method
clipboard.setContents(contents, null);
if (action == TransferHandler.MOVE) {
doc.remove(p0, p1 - p0);
}
} catch (BadLocationException ble) {}
}
}
}
问题在于,它没有考虑样式标记。喜悦。
因此,在阅读了更多内容后,我遇到了 -用 Java 复制到剪贴板
所以,我修改了原始代码以使用相同的概念......
editorPane.getActionMap().put("copy", new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
String text = editorPane.getText();
text = formattedPlainText(text);
StringSelection selection = new StringSelection(text);
Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
clipboard.setContents(selection, selection);
}
protected String formattedPlainText(String text) {
final StringBuilder sb = new StringBuilder();
HTMLEditorKit.ParserCallback parserCallback = new HTMLEditorKit.ParserCallback() {
public boolean readyForNewline;
@Override
public void handleText(final char[] data, final int pos) {
String s = new String(data);
sb.append(s.trim());
readyForNewline = true;
}
@Override
public void handleStartTag(final HTML.Tag t, final MutableAttributeSet a, final int pos) {
if (readyForNewline && (t == HTML.Tag.DIV || t == HTML.Tag.BR || t == HTML.Tag.P)) {
sb.append("\n");
readyForNewline = false;
}
}
@Override
public void handleSimpleTag(final HTML.Tag t, final MutableAttributeSet a, final int pos) {
handleStartTag(t, a, pos);
}
};
try {
new ParserDelegator().parse(new StringReader(text), parserCallback, false);
} catch (IOException ex) {
ex.printStackTrace();
}
return sb.toString();
}
});
我现在可以当我复制文本时得到这个......
Bonjour,
Merci de faire les actions ci-dessous:
1- Sauvegarder le fichier AAAA
2- Remplacer le contenu du fichier AAAA par le contenu fichier BBBBB
3- Relancer au step planté
Cordialement,
卡维特
这两个示例都将忽略选择并复制整个文本 - 这是我时间的限制。StyledDocument
如果需要,将花费更多的精力从
推荐阅读
- random - 如何从标准正态分布中抽取大小为 n 的样本?
- java - 查询手机中的所有图像适用于 api 18 但不适用于 api 30
- r - 在 R 中,根据左侧表中指示变量的值有条件地左连接两个表
- c++ - 混淆构造函数之间的交互和 C++ 中的复制/移动语义
- sql - 在 Presto 中如何检索具有唯一元素的数组
- anylogic - 读取和写入数据到文件
- xamarin.android - Android 应用在前台未显示 Azure 通知中心通知
- ios - 设置 UINavigationBar 高度与屏幕大小成比例
- python - 为什么我的 tkinter 代码会发生这种情况?
- javascript - 实现 React 的 UseEffect 只监听增量/减量