【问题标题】:My custom "paste from clipboard" action我的自定义“从剪贴板粘贴”操作
【发布时间】:2013-12-19 01:42:39
【问题描述】:

我想找到一种方法来执行一种自定义的“从剪贴板粘贴”操作。为简单起见,我们假设剪贴板的内容是文本(不是文件)。每当您按Ctrl+V 时,它都会将该内容(即文本)插入到当前打开的具有焦点的文件中。

我有一个用于捕捉 global 热键的应用。请注意,这不是一个窗口应用程序,它是一个控制台应用程序,它可以全局 捕获热键。假设我有Ctrl+U 的热键。所以我想做的是当我按下Ctrl+U 时,我想在当前打开的文件中插入一些预定义的文本。就像Ctrl+V 一样!与标准 Ctrl+V 的区别在于我要插入 预定义 文本,并且热键 不同

我该怎么做?

我更喜欢跨平台解决方案,但首先我要为 Linux 做这个,特别是 Ubuntu。语言并不重要,但 Java 或 Scala 会更好。当然,我知道解决方案是 Java 使用本机操作系统的 API。

【问题讨论】:

  • 嗨,Alex,我有一些澄清问题。当前打开的文件是在您的 java 应用程序中打开的,还是在某些随机文本编辑器中打开的?另外,您是否已经有了热键事件的代码,或者这也应该是答案的一部分?
  • @Bucco,在任何文本编辑器中。正如我所说,我有热键的代码。

标签: java c++ linux windows clipboard


【解决方案1】:

我希望这个骇人听闻的解决方案能奏效,但它仍然未经测试,我不确定如何捕捉热键的事件。

这段代码背后的想法是以下五个步骤:

  1. 获取剪贴板中的旧文本并临时保存
  2. 将我们预定义的文本粘贴到剪贴板中
  3. 触发全局粘贴事件
  4. 释放全局粘贴事件
  5. 将剪贴板重置为旧文本

这应该会给你一个新剪贴板的外观(如果不是,希望它会激发你想出一个更好、更少hackish的解决方案)。

事不宜迟,这是我的代码。首先,我有一个简单的辅助方法来设置剪贴板的值(我们这样做了两次)。

public static void setClipboard(String s) {
    StringSelection contents = new StringSelection(s);
    Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
    clipboard.setContents(contents, contents);
}

然后,我有一个主要方法,我按顺序执行五个步骤。

public static void main(String[] args) {

    // Step 1 ) get old text
    String oldText = "";
    try {
        oldText = (String) Toolkit.getDefaultToolkit().getSystemClipboard().getData(DataFlavor.stringFlavor); 
    } catch (UnsupportedFlavorException ufe) {
        ufe.printStackTrace();
    } catch (IOException ioe) {
        ioe.printStackTrace();
    }

    // Step 2 ) paste our text in clipboard
    setClipboard("This lorem ipsum predefined string blows my mind.");

    // Step 3 ) trigger paste event
    Robot robot = null;
    try {
        robot = new Robot();
    } catch (AWTException awte) {
        awte.printStackTrace();
    }
    robot.keyPress(KeyEvent.VK_CONTROL);
    robot.keyPress(KeyEvent.VK_V);

    // Step 4 ) Release paste event
    robot.keyRelease(KeyEvent.VK_CONTROL);
    robot.keyRelease(KeyEvent.VK_V);

    // Step 5 ) Reset clipboard
    setClipboard(oldText);

}

[编辑]:

这里有一些代码来测试剪贴板中的内容类型 - 图像、文本等。unicode 错误来自剪贴板的旧内容无法用纯文本表示的事实细绳。要修复此错误,您必须检查旧内容是否为图像,旧内容是否为文本,并相应地保存它们。

public static int kindOfContents() {
    Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
    Transferable contents = clipboard.getContents(null);

    if(contents.isDataFlavorSupported(DataFlavor.stringFlavor)) {
        // String, save temporarily as string and write back as string
        return 0;
    } else if(contents.isDataFlavorSupported(DataFlavor.imageFlavor)) {
        // Image, save temporarily as BufferedImage and write back as image
        return 1;
    } else if(contents.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
        // List of files, save temporarily as java.util.List interface and write back as the file lists
        return 2;
    }

}

如果内容是文本,那么为了保存和写入内容,您将使用旧方法,为方便起见,在下面重新粘贴。

// Step 1 ) get old text
String oldText = "";
try {
    oldText = (String) Toolkit.getDefaultToolkit().getSystemClipboard().getData(DataFlavor.stringFlavor); 
} catch (UnsupportedFlavorException ufe) {
    ufe.printStackTrace();
} catch (IOException ioe) {
    ioe.printStackTrace();
}

// Step 5 ) Reset clipboard
setClipboard(oldText);

但是,如果内容是图像,那么为了临时保存和重写,您需要执行以下操作。请注意,用于编写图像的代码不是我的,而是取自Setting images to Clipboard - Java 接受的答案

// Step 1 ) get old image
BufferedImage img = null;
try {
    img = (BufferedImage) Toolkit.getDefaultToolkit().getSystemClipboard().getData(DataFlavor.imageFlavor);
} catch (UnsupportedFlavorException ufe) {
    ufe.printStackTrace();
} catch (IOException ioe) {
    ioe.printStackTrace();
}

取自Setting images to Clipboard - Java

// Step 5 ) Reset clipboard
ImageTransferable transferable = new ImageTransferable( image );
Toolkit.getDefaultToolkit().getSystemClipboard().setContents(transferable, null);

static class ImageTransferable implements Transferable
{
    private Image image;

    public ImageTransferable (Image image)
    {
        this.image = image;
    }

    public Object getTransferData(DataFlavor flavor)
        throws UnsupportedFlavorException
    {
        if (isDataFlavorSupported(flavor))
        {
            return image;
        }
        else
        {
            throw new UnsupportedFlavorException(flavor);
        }
    }

    public boolean isDataFlavorSupported (DataFlavor flavor)
    {
        return flavor == DataFlavor.imageFlavor;
    }

    public DataFlavor[] getTransferDataFlavors ()
    {
        return new DataFlavor[] { DataFlavor.imageFlavor };
    }
}

【讨论】:

  • 这是一个有趣的解决方案。
  • 出于某种原因,即使我正在编写英文文本,我也会收到java.awt.datatransfer.UnsupportedFlavorException: Unicode String。为什么?
  • @Alex 这是它试图保存旧内容的时候,不是标准文本。如果您不关心旧内容,您可以将步骤 1 和 5 注释掉。否则,您将需要查看它是字符串还是图像。我现在将添加更多代码来做到这一点。
  • 你确定之前剪贴板中的内容都是英文的吗? (而不仅仅是你将要放入的东西)
  • 是的,它现在可以工作了,我猜是我的小姐。但是,它的工作方式很奇怪:有时它会插入新文本,有时则不会。我不知道为什么会这样。
猜你喜欢
  • 2015-10-06
  • 2016-01-30
  • 2017-12-23
  • 2015-07-29
  • 2013-07-08
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多