繁体   English   中英

Java / Swing:系统剪贴板的所有权

[英]Java/Swing: ownership of the system clipboard

我正在编写一个小型Java程序,该程序应该运行一个将图像复制到系统剪贴板的外部程序(即Windows 7“剪裁工具”),等待其完成,将图像从剪贴板保存到磁盘并复制剪贴板的URL(可从中访问图像)。 简而言之,应该:

  1. 运行外部工具并等待它
  2. 从剪贴板复制图像
  3. 将字符串复制到剪贴板

这,我的程序完全可以做到。 但是,我想使用Swing / AWT来呈现用户界面。 我使用的是系统任务栏图标,但为简单起见,它也可能是框架中的JButton。 单击该按钮时,应执行上述过程。 第一次执行此操作时,它应能正常工作。 图像被复制,粘贴到磁盘,并且字符串被复制到剪贴板。 然后,第二次单击该按钮,好像我的程序没有意识到剪贴板已更新,因为它从第一次开始仍然看到自己的字符串。 之后,我的剪贴板处理类才失去所有权,实际上,第二次尝试执行该过程都会失败。

import java.awt.Toolkit;
import java.awt.datatransfer.Clipboard;
import java.awt.datatransfer.ClipboardOwner;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.StringSelection;
import java.awt.datatransfer.Transferable;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.image.BufferedImage;
import java.io.IOException;

import javax.swing.JButton;
import javax.swing.JFrame;

public class Main {
    private static BufferedImage image; //the image from clipboard to be saved

    public static void main(String[] args) throws InterruptedException, IOException {
        new GUI();
    }

    public static void run(String filename) throws IOException, InterruptedException {
        CBHandler cbh = new CBHandler();

        //run tool, tool will copy an image to system clipboard
        Process p = Runtime.getRuntime().exec("C:\\Windows\\system32\\SnippingTool.exe");
        p.waitFor();

        //copy image from clipboard
        image = cbh.getClipboard();
        if(image == null) {
            System.out.println("No image found in clipboard.");
            return;
        }

        //save image to disk...

        //copy file link to clipboard
        String link = "http://somedomain.com/" + filename;
        cbh.setClipboard(link);
    }
}

class CBHandler implements ClipboardOwner {
    public BufferedImage getClipboard() {
        Transferable t = Toolkit.getDefaultToolkit().getSystemClipboard().getContents(null);

        try {
            if(t.isDataFlavorSupported(DataFlavor.imageFlavor))
                return (BufferedImage) t.getTransferData(DataFlavor.imageFlavor);
        }
        catch(Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public void setClipboard(String str) {
        StringSelection strsel = new StringSelection(str);
        Toolkit.getDefaultToolkit().getSystemClipboard().setContents(strsel, this);
    }

    @Override
    public void lostOwnership(Clipboard arg0, Transferable arg1) {
        System.out.println("Lost ownership!");
    }
}

class GUI extends JFrame {
    public GUI() {
        JButton button = new JButton("Run");
        button.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent arg0) {
                try {
                    Main.run("saveFile.png");
                } catch (IOException e) {
                    e.printStackTrace();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });

        add(button);
        pack();
        setVisible(true);
    }
}

如果尝试运行它,请注意,在第二次运行中,仅在尝试复制映像之后才调用lostOwnership方法。 我猜这是问题的根源,我不知道为什么会这样,只是它只有在由Swing事件触发时才会发生。 任何帮助解决此问题的方法将不胜感激。

一个猜想:您正在AWT事件分发线程上进行整个处理(调用另一个进程)(例如,直接从ActionListener或类似对象进行)。

剪贴板更改消息也将由EDT上的VM处理...,但仅在单击按钮之后。

道德:不要在EDT上执行长时间运行的工作(以及具有应在事件队列中排队的影响的工作),而应该为此启动一个新线程。

了解丢失所有权问题的关键在于这一行

Toolkit.getDefaultToolkit().getSystemClipboard().setContents(strsel, this);

您传入的第二个参数是ClipboardOwner。 剪贴板的JavaDocs.setContents说

如果存在与参数所有者不同的现有所有者,则通过对该所有者调用ClipboardOwner.lostOwnership()来通知该所有者不再拥有剪贴板内容的所有权。 setContents()的实现可以自由地从此方法直接调用lostOwnership()。 例如,lostOwnership()可以稍后在另一个线程上调用。 这同样适用于在此剪贴板上注册的FlavorListeners。

好吧,这是怎么回事? 当您传递所有者时,剪贴板现在具有对该对象的引用。 在这种情况下,它是CBHandler。 然后,您创建一个新的并尝试再次设置内容。 然后剪贴板返回到旧所有者(您的原始实例),并告诉它“嘿,您不再是所有者了”。

public synchronized void setContents(Transferable contents, ClipboardOwner owner) {
    final ClipboardOwner oldOwner = this.owner;
    final Transferable oldContents = this.contents;

    this.owner = owner;
    this.contents = contents;

    if (oldOwner != null && oldOwner != owner) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                oldOwner.lostOwnership(Clipboard.this, oldContents);
            }
        });
    }
    fireFlavorsChanged();
}

您必须提供有关其他问题的更多详细信息“好像我的程序没有意识到剪贴板已更新,因为它从头开始仍然看到自己的字符串。”

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM